prevent infinite loops caused by pathological component bindings (#398)

pull/407/head
Rich-Harris 8 years ago
parent 8ff2de90be
commit 1780876b99

@ -45,6 +45,7 @@ export default function createBinding ( generator, node, attribute, current, loc
component._bindings.push( function () { component._bindings.push( function () {
if ( ${local.name}._torndown ) return; if ( ${local.name}._torndown ) return;
${local.name}.observe( '${attribute.name}', function ( value ) { ${local.name}.observe( '${attribute.name}', function ( value ) {
if ( ${local.name}_updating ) return;
${local.name}_updating = true; ${local.name}_updating = true;
${setter} ${setter}
${local.name}_updating = false; ${local.name}_updating = false;
@ -54,7 +55,9 @@ export default function createBinding ( generator, node, attribute, current, loc
local.update.addBlock( deindent` local.update.addBlock( deindent`
if ( !${local.name}_updating && ${dependencies.map( dependency => `'${dependency}' in changed` ).join( '||' )} ) { if ( !${local.name}_updating && ${dependencies.map( dependency => `'${dependency}' in changed` ).join( '||' )} ) {
${local.name}_updating = true;
${local.name}._set({ ${attribute.name}: ${snippet} }); ${local.name}._set({ ${attribute.name}: ${snippet} });
${local.name}_updating = false;
} }
` ); ` );
} }

@ -0,0 +1,12 @@
<B bind:currentIdentifier />
<B bind:currentIdentifier />
<script>
import B from './B.html';
export default {
components: {
B
}
}
</script>

@ -0,0 +1,20 @@
{{#each list as item}}
<p>
<C identifier="{{item}}" bind:currentIdentifier>
{{item}}
</C>
</p>
{{/each}}
<script>
import C from './C.html';
export default {
data: () => ({
list: [1, 2, 3, 2, 1]
}),
components: {
C
}
}
</script>

@ -0,0 +1,23 @@
<span
on:click="toggle()"
class="{{isCurrentlySelected ? 'selected' : ''}}"
>
{{yield}}
</span>
<script>
export default {
computed: {
isCurrentlySelected: (currentIdentifier, identifier) => currentIdentifier === identifier
},
methods: {
toggle() {
const isCurrentlySelected = this.get('isCurrentlySelected')
this.set({
currentIdentifier: isCurrentlySelected ? null : this.get('identifier')
})
}
}
}
</script>

@ -0,0 +1,90 @@
export default {
html: `
<p><span class=''>1</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>3</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>3</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>3</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>3</span></p>
<p><span class='selected'>2</span></p>
<p><span class=''>1</span></p>
`,
test ( assert, component, target, window ) {
const click = new window.MouseEvent( 'click' );
const spans = target.querySelectorAll( 'span' );
spans[0].dispatchEvent( click );
assert.equal( component.get( 'currentIdentifier' ), 1 );
assert.htmlEqual( target.innerHTML, `
<p><span class='selected'>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class='selected'>1</span></p>
<p><span class='selected'>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class='selected'>1</span></p>
<p><span class='selected'>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class='selected'>1</span></p>
<p><span class='selected'>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class='selected'>1</span></p>
` );
spans[0].dispatchEvent( click );
assert.equal( component.get( 'currentIdentifier' ), null );
assert.htmlEqual( target.innerHTML, `
<p><span class=''>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class=''>1</span></p>
<p><span class=''>1</span></p>
<p><span class=''>2</span></p>
<p><span class=''>3</span></p>
<p><span class=''>2</span></p>
<p><span class=''>1</span></p>
` );
}
};

@ -0,0 +1,17 @@
<A bind:currentIdentifier />
<A bind:currentIdentifier />
<script>
import A from './A.html';
export default {
data() {
return {
currentIdentifier: 2
}
},
components: {
A
}
}
</script>
Loading…
Cancel
Save