store prop names as $$.props, prevent leaky bindings. fixes #2222

pull/2232/head
Richard Harris 6 years ago
parent 07ceb2e767
commit 9ea663e312

@ -407,6 +407,8 @@ export default function dom(
`); `);
} }
const prop_names = `[${props.map(v => JSON.stringify(v.export_name)).join(', ')}]`;
if (options.customElement) { if (options.customElement) {
builder.addBlock(deindent` builder.addBlock(deindent`
class ${name} extends @SvelteElement { class ${name} extends @SvelteElement {
@ -415,7 +417,7 @@ export default function dom(
${css.code && `this.shadowRoot.innerHTML = \`<style>${escape(css.code, { onlyEscapeAtSymbol: true }).replace(/\\/g, '\\\\')}${options.dev ? `\n/*# sourceMappingURL=${css.map.toUrl()} */` : ''}</style>\`;`} ${css.code && `this.shadowRoot.innerHTML = \`<style>${escape(css.code, { onlyEscapeAtSymbol: true }).replace(/\\/g, '\\\\')}${options.dev ? `\n/*# sourceMappingURL=${css.map.toUrl()} */` : ''}</style>\`;`}
@init(this, { target: this.shadowRoot }, ${definition}, create_fragment, ${not_equal}); @init(this, { target: this.shadowRoot }, ${definition}, create_fragment, ${not_equal}, ${prop_names});
${dev_props_check} ${dev_props_check}
@ -449,7 +451,7 @@ export default function dom(
constructor(options) { constructor(options) {
super(${options.dev && `options`}); super(${options.dev && `options`});
${should_add_css && `if (!document.getElementById("${component.stylesheet.id}-style")) @add_css();`} ${should_add_css && `if (!document.getElementById("${component.stylesheet.id}-style")) @add_css();`}
@init(this, options, ${definition}, create_fragment, ${not_equal}); @init(this, options, ${definition}, create_fragment, ${not_equal}, ${prop_names});
${dev_props_check} ${dev_props_check}
} }

@ -5,6 +5,7 @@ import { blankObject } from './utils.js';
import { children } from './dom.js'; import { children } from './dom.js';
export function bind(component, name, callback) { export function bind(component, name, callback) {
if (component.$$.props.indexOf(name) === -1) return;
component.$$.bound[name] = callback; component.$$.bound[name] = callback;
callback(component.$$.ctx[name]); callback(component.$$.ctx[name]);
} }
@ -53,7 +54,7 @@ function make_dirty(component, key) {
component.$$.dirty[key] = true; component.$$.dirty[key] = true;
} }
export function init(component, options, instance, create_fragment, not_equal) { export function init(component, options, instance, create_fragment, not_equal, prop_names) {
const parent_component = current_component; const parent_component = current_component;
set_current_component(component); set_current_component(component);
@ -64,6 +65,7 @@ export function init(component, options, instance, create_fragment, not_equal) {
ctx: null, ctx: null,
// state // state
props: prop_names,
update: noop, update: noop,
not_equal, not_equal,
bound: blankObject(), bound: blankObject(),

@ -0,0 +1,5 @@
<script>
let count = 0;
</script>
<button on:click='{() => count += 1}'>{count}</button>

@ -0,0 +1,27 @@
export default {
html: `
<button>0</button>
<p>count: undefined</p>
`,
async test({ assert, component, target, window }) {
const click = new window.MouseEvent('click');
const button = target.querySelector('button');
await button.dispatchEvent(click);
assert.equal(component.x, undefined);
assert.htmlEqual(target.innerHTML, `
<button>1</button>
<p>count: undefined</p>
`);
await button.dispatchEvent(click);
assert.equal(component.x, undefined);
assert.htmlEqual(target.innerHTML, `
<button>2</button>
<p>count: undefined</p>
`);
}
};

@ -0,0 +1,8 @@
<script>
import Counter from './Counter.svelte';
let x;
</script>
<Counter bind:count={x}/>
<p>count: {x}</p>
Loading…
Cancel
Save