From 52a30dc8e3b50540c75c897284068d0f0acad7f2 Mon Sep 17 00:00:00 2001 From: Rich-Harris Date: Tue, 18 Apr 2017 21:34:20 -0400 Subject: [PATCH 1/2] failing test for #495 --- .../_config.js | 36 +++++++++++++++++++ .../binding-input-text-deconflicted/main.html | 2 ++ 2 files changed, 38 insertions(+) create mode 100644 test/runtime/samples/binding-input-text-deconflicted/_config.js create mode 100644 test/runtime/samples/binding-input-text-deconflicted/main.html diff --git a/test/runtime/samples/binding-input-text-deconflicted/_config.js b/test/runtime/samples/binding-input-text-deconflicted/_config.js new file mode 100644 index 0000000000..0b705d3520 --- /dev/null +++ b/test/runtime/samples/binding-input-text-deconflicted/_config.js @@ -0,0 +1,36 @@ +export default { + solo: true, + + data: { + component: { + name: 'world' + } + }, + + html: ` +

Hello world!

+ + `, + + test ( assert, component, target, window ) { + const input = target.querySelector( 'input' ); + assert.equal( input.value, 'world' ); + + const event = new window.Event( 'input' ); + + input.value = 'everybody'; + input.dispatchEvent( event ); + + assert.equal( target.innerHTML, ` +

Hello everybody!

+ + ` ); + + component.set({ name: 'goodbye' }); + assert.equal( input.value, 'goodbye' ); + assert.equal( target.innerHTML, ` +

Hello goodbye!

+ + ` ); + } +}; diff --git a/test/runtime/samples/binding-input-text-deconflicted/main.html b/test/runtime/samples/binding-input-text-deconflicted/main.html new file mode 100644 index 0000000000..391a47c3f1 --- /dev/null +++ b/test/runtime/samples/binding-input-text-deconflicted/main.html @@ -0,0 +1,2 @@ +

Hello {{component.name}}!

+ \ No newline at end of file From b0095bda9d6cdfec7b60a219e87e46bf25e6a7e3 Mon Sep 17 00:00:00 2001 From: Rich-Harris Date: Tue, 18 Apr 2017 21:54:29 -0400 Subject: [PATCH 2/2] =?UTF-8?q?ensure=20var=20names=20in=20binding=20callb?= =?UTF-8?q?acks=20are=20unique=20=E2=80=94=20fixes=20#495?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/generators/dom/visitors/Component/Binding.js | 4 ++-- src/generators/dom/visitors/Element/Binding.js | 2 +- .../dom/visitors/shared/binding/getSetter.js | 13 ++++++++----- .../binding-input-text-deconflicted/_config.js | 9 ++++----- 4 files changed, 15 insertions(+), 13 deletions(-) diff --git a/src/generators/dom/visitors/Component/Binding.js b/src/generators/dom/visitors/Component/Binding.js index 7ad2e61cb2..12bf41cbf0 100644 --- a/src/generators/dom/visitors/Component/Binding.js +++ b/src/generators/dom/visitors/Component/Binding.js @@ -3,7 +3,7 @@ import flattenReference from '../../../../utils/flattenReference.js'; import getSetter from '../shared/binding/getSetter.js'; export default function visitBinding ( generator, block, state, node, attribute, local ) { - const { name, keypath } = flattenReference( attribute.value ); + const { name } = flattenReference( attribute.value ); const { snippet, contexts, dependencies } = block.contextualise( attribute.value ); if ( dependencies.length > 1 ) throw new Error( 'An unexpected situation arose. Please raise an issue at https://github.com/sveltejs/svelte/issues — thanks!' ); @@ -35,7 +35,7 @@ export default function visitBinding ( generator, block, state, node, attribute, prop }); - const setter = getSetter({ block, name, keypath, context: '_context', attribute, dependencies, value: 'value' }); + const setter = getSetter({ block, name, context: '_context', attribute, dependencies, value: 'value' }); generator.hasComplexBindings = true; diff --git a/src/generators/dom/visitors/Element/Binding.js b/src/generators/dom/visitors/Element/Binding.js index e88fe8df43..8771fe05e6 100644 --- a/src/generators/dom/visitors/Element/Binding.js +++ b/src/generators/dom/visitors/Element/Binding.js @@ -20,7 +20,7 @@ export default function visitBinding ( generator, block, state, node, attribute const bindingGroup = attribute.name === 'group' ? getBindingGroup( generator, keypath ) : null; const value = getBindingValue( generator, block, state, node, attribute, isMultipleSelect, bindingGroup, type ); - let setter = getSetter({ block, name, keypath, context: '_svelte', attribute, dependencies, value }); + let setter = getSetter({ block, name, context: '_svelte', attribute, dependencies, value }); let updateElement = `${state.parentNode}.${attribute.name} = ${snippet};`; const lock = block.getUniqueName( `${state.parentNode}_${attribute.name}_updating` ); let updateCondition = `!${lock}`; diff --git a/src/generators/dom/visitors/shared/binding/getSetter.js b/src/generators/dom/visitors/shared/binding/getSetter.js index 852ea83fa7..14c3799280 100644 --- a/src/generators/dom/visitors/shared/binding/getSetter.js +++ b/src/generators/dom/visitors/shared/binding/getSetter.js @@ -1,9 +1,10 @@ import deindent from '../../../../../utils/deindent.js'; -export default function getSetter ({ block, name, keypath, context, attribute, dependencies, value }) { +export default function getSetter ({ block, name, context, attribute, dependencies, value }) { + const tail = attribute.value.type === 'MemberExpression' ? getTailSnippet( attribute.value ) : ''; + if ( block.contexts.has( name ) ) { const prop = dependencies[0]; - const tail = attribute.value.type === 'MemberExpression' ? getTailSnippet( attribute.value ) : ''; return deindent` var list = this.${context}.${block.listNames.get( name )}; @@ -15,10 +16,12 @@ export default function getSetter ({ block, name, keypath, context, attribute, d } if ( attribute.value.type === 'MemberExpression' ) { + const alias = block.alias( name ); + return deindent` - var ${name} = ${block.component}.get( '${name}' ); - ${keypath} = ${value}; - ${block.component}._set({ ${name}: ${name} }); + var ${alias} = ${block.component}.get( '${name}' ); + ${alias}${tail} = ${value}; + ${block.component}._set({ ${name}: ${alias} }); `; } diff --git a/test/runtime/samples/binding-input-text-deconflicted/_config.js b/test/runtime/samples/binding-input-text-deconflicted/_config.js index 0b705d3520..49007dc883 100644 --- a/test/runtime/samples/binding-input-text-deconflicted/_config.js +++ b/test/runtime/samples/binding-input-text-deconflicted/_config.js @@ -1,6 +1,4 @@ export default { - solo: true, - data: { component: { name: 'world' @@ -21,14 +19,15 @@ export default { input.value = 'everybody'; input.dispatchEvent( event ); - assert.equal( target.innerHTML, ` + assert.equal( input.value, 'everybody' ); + assert.htmlEqual( target.innerHTML, `

Hello everybody!

` ); - component.set({ name: 'goodbye' }); + component.set({ component: { name: 'goodbye' } }); assert.equal( input.value, 'goodbye' ); - assert.equal( target.innerHTML, ` + assert.htmlEqual( target.innerHTML, `

Hello goodbye!

` );