Allow binding of <details> open

pull/2855/head
Benjamin Milde 6 years ago
parent a6c05ed372
commit dabc9c3e53

@ -544,7 +544,7 @@ export default class Element extends Node {
message: `'group' binding can only be used with <input type="checkbox"> or <input type="radio">` message: `'group' binding can only be used with <input type="checkbox"> or <input type="radio">`
}); });
} }
} else if (name == 'files') { } else if (name === 'files') {
if (this.name !== 'input') { if (this.name !== 'input') {
component.error(binding, { component.error(binding, {
code: `invalid-binding`, code: `invalid-binding`,
@ -560,6 +560,14 @@ export default class Element extends Node {
message: `'files' binding can only be used with <input type="file">` message: `'files' binding can only be used with <input type="file">`
}); });
} }
} else if (name === 'open') {
if (this.name !== 'details') {
component.error(binding, {
code: `invalid-binding`,
message: `'${name}' binding can only be used with <details>`
});
}
} else if ( } else if (
name === 'currentTime' || name === 'currentTime' ||
name === 'duration' || name === 'duration' ||

@ -91,6 +91,12 @@ const events = [
name === 'playbackRate' name === 'playbackRate'
}, },
// details event
{
event_names: ['toggle'],
filter: (node: Element, name: string) =>
node.name === 'details'
},
]; ];
export default class ElementWrapper extends Wrapper { export default class ElementWrapper extends Wrapper {

@ -0,0 +1,69 @@
/* generated by Svelte vX.Y.Z */
import {
SvelteComponent,
detach,
element,
init,
insert,
listen,
noop,
safe_not_equal
} from "svelte/internal";
function create_fragment(ctx) {
var details, dispose;
return {
c() {
details = element("details");
details.innerHTML = `<summary>summary</summary>content
`;
dispose = listen(details, "toggle", ctx.details_toggle_handler);
},
m(target, anchor) {
insert(target, details, anchor);
details.open = ctx.open;
},
p(changed, ctx) {
if (changed.open) details.open = ctx.open;
},
i: noop,
o: noop,
d(detaching) {
if (detaching) {
detach(details);
}
dispose();
}
};
}
function instance($$self, $$props, $$invalidate) {
let { open } = $$props;
function details_toggle_handler() {
open = this.open;
$$invalidate('open', open);
}
$$self.$set = $$props => {
if ('open' in $$props) $$invalidate('open', open = $$props.open);
};
return { open, details_toggle_handler };
}
class Component extends SvelteComponent {
constructor(options) {
super();
init(this, options, instance, create_fragment, safe_not_equal, ["open"]);
}
}
export default Component;

@ -0,0 +1,7 @@
<script>
export let open;
</script>
<details bind:open={open}>
<summary>summary</summary>content
</details>

@ -0,0 +1,25 @@
export default {
html: `
<details><summary>toggle</summary></details>
`,
async test({ assert, component, target, window }) {
const details = target.querySelector('details');
const event = new window.Event('toggle');
details.open = true;
await details.dispatchEvent(event);
assert.equal(component.open, true);
assert.htmlEqual(target.innerHTML, `
<details open><summary>toggle</summary></details>
<p>hello!</p>
`);
details.open = false;
await details.dispatchEvent(event);
assert.equal(component.open, false);
assert.htmlEqual(target.innerHTML, `
<details open=""><summary>toggle</summary></details>
`);
}
};

@ -0,0 +1,9 @@
<script>
export let visible;
</script>
<details bind:open={visible}><summary>toggle</summary></details>
{#if visible}
<p>hello!</p>
{/if}
Loading…
Cancel
Save