From a793398d63a848b744c6e67c705aa8957a862dd2 Mon Sep 17 00:00:00 2001 From: Rich Harris Date: Tue, 11 Apr 2017 10:09:56 -0400 Subject: [PATCH 1/3] catch hardcoded names that should be aliases (#465) --- src/generators/Generator.js | 4 ++++ .../dom/visitors/Element/EventHandler.js | 4 ++-- .../dom/visitors/Element/meta/Window.js | 4 ++-- test/helpers.js | 15 ++++++++++++--- test/runtime/index.js | 9 +++++++-- 5 files changed, 27 insertions(+), 9 deletions(-) diff --git a/src/generators/Generator.js b/src/generators/Generator.js index 272e843142e3..83f726cab6f2 100644 --- a/src/generators/Generator.js +++ b/src/generators/Generator.js @@ -11,6 +11,8 @@ import getOutro from './shared/utils/getOutro.js'; import processCss from './shared/processCss.js'; import annotateWithScopes from './annotateWithScopes.js'; +const test = typeof global !== 'undefined' && global.__svelte_test; + export default class Generator { constructor ( parsed, source, name, options ) { this.parsed = parsed; @@ -238,6 +240,7 @@ export default class Generator { } getUniqueName ( name ) { + if ( test ) name = `${name}$`; let alias = name; for ( let i = 1; reservedNames.has( alias ) || this.importedNames.has( alias ) || this._usedNames.has( alias ); alias = `${name}_${i++}` ); this._usedNames.add( alias ); @@ -247,6 +250,7 @@ export default class Generator { getUniqueNameMaker ( params ) { const localUsedNames = new Set( params ); return name => { + if ( test ) name = `${name}$`; let alias = name; for ( let i = 1; reservedNames.has( alias ) || this.importedNames.has( alias ) || this._usedNames.has( alias ) || localUsedNames.has( alias ); alias = `${name}_${i++}` ); localUsedNames.add( alias ); diff --git a/src/generators/dom/visitors/Element/EventHandler.js b/src/generators/dom/visitors/Element/EventHandler.js index 2a51cdc1fff0..63b8f0b43def 100644 --- a/src/generators/dom/visitors/Element/EventHandler.js +++ b/src/generators/dom/visitors/Element/EventHandler.js @@ -32,7 +32,7 @@ export default function visitEventHandler ( generator, block, state, node, attri const declarations = usedContexts.map( name => { if ( name === 'root' ) { if ( shouldHoist ) state.usesComponent = true; - return 'var root = component.get();'; + return `var root = ${block.component}.get();`; } const listName = block.listNames.get( name ); @@ -52,7 +52,7 @@ export default function visitEventHandler ( generator, block, state, node, attri if ( state.usesComponent ) { // TODO the element needs to know to create `thing._svelte = { component: component }` - handlerBody.addLine( `var component = this._svelte.component;` ); + handlerBody.addLine( `var ${block.component} = this._svelte.component;` ); } declarations.forEach( declaration => { diff --git a/src/generators/dom/visitors/Element/meta/Window.js b/src/generators/dom/visitors/Element/meta/Window.js index c0f9f3d1817c..b1dc8f8ecfa1 100644 --- a/src/generators/dom/visitors/Element/meta/Window.js +++ b/src/generators/dom/visitors/Element/meta/Window.js @@ -22,7 +22,7 @@ export default function visitWindow ( generator, block, node ) { const flattened = flattenReference( attribute.expression.callee ); if ( flattened.name !== 'event' && flattened.name !== 'this' ) { // allow event.stopPropagation(), this.select() etc - generator.code.prependRight( attribute.expression.start, 'component.' ); + generator.code.prependRight( attribute.expression.start, `${block.component}.` ); } const handlerName = block.getUniqueName( `onwindow${attribute.name}` ); @@ -68,7 +68,7 @@ export default function visitWindow ( generator, block, node ) { block.builders.create.addBlock( deindent` var ${handlerName} = function ( event ) { - component.set({ + ${block.component}.set({ ${props} }); }; diff --git a/test/helpers.js b/test/helpers.js index 905233280c5d..4318d6d3644e 100644 --- a/test/helpers.js +++ b/test/helpers.js @@ -10,9 +10,18 @@ sourceMapSupport.install(); // for coverage purposes, we need to test source files, // but for sanity purposes, we need to test dist files -export const svelte = process.env.COVERAGE ? - require( '../src/index.js' ) : - require( '../compiler/svelte.js' ); +export function loadSvelte ( test ) { + if ( test ) global.__svelte_test = true; + + const resolved = process.env.COVERAGE ? + require.resolve( '../src/index.js' ) : + require.resolve( '../compiler/svelte.js' ); + + delete require.cache[ resolved ]; + return require( resolved ); +} + +export const svelte = loadSvelte(); export function exists ( path ) { try { diff --git a/test/runtime/index.js b/test/runtime/index.js index f4926388c1f9..e67192cbc66c 100644 --- a/test/runtime/index.js +++ b/test/runtime/index.js @@ -5,7 +5,9 @@ import * as fs from 'fs'; import * as acorn from 'acorn'; import * as babel from 'babel-core'; -import { addLineNumbers, loadConfig, svelte, env, setupHtmlEqual } from '../helpers.js'; +import { addLineNumbers, loadConfig, loadSvelte, env, setupHtmlEqual } from '../helpers.js'; + +let svelte; let showCompiledCode = false; let compileOptions = null; @@ -33,7 +35,10 @@ require.extensions[ '.html' ] = function ( module, filename ) { const Object_assign = Object.assign; describe( 'runtime', () => { - before( setupHtmlEqual ); + before( () => { + svelte = loadSvelte( true ); + return setupHtmlEqual(); + }); function runTest ( dir, shared ) { if ( dir[0] === '.' ) return; From 91a58a0f26e4652121071ea27b8a708456587a16 Mon Sep 17 00:00:00 2001 From: Rich Harris Date: Tue, 11 Apr 2017 11:02:20 -0400 Subject: [PATCH 2/3] fix all aliasing tests --- src/generators/dom/visitors/Component/Component.js | 2 +- src/generators/dom/visitors/Element/EventHandler.js | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/generators/dom/visitors/Component/Component.js b/src/generators/dom/visitors/Component/Component.js index 5446d5427609..5e210e976276 100644 --- a/src/generators/dom/visitors/Component/Component.js +++ b/src/generators/dom/visitors/Component/Component.js @@ -140,7 +140,7 @@ export default function visitComponent ( generator, block, state, node ) { if ( local.bindings.length ) { const initialData = block.getUniqueName( `${name}_initial_data` ); - statements.push( `var ${name}_initial_data = ${initialPropString};` ); + statements.push( `var ${initialData} = ${initialPropString};` ); local.bindings.forEach( binding => { statements.push( `if ( ${binding.prop} in ${binding.obj} ) ${initialData}.${binding.name} = ${binding.value};` ); diff --git a/src/generators/dom/visitors/Element/EventHandler.js b/src/generators/dom/visitors/Element/EventHandler.js index 63b8f0b43def..189bb536c4ed 100644 --- a/src/generators/dom/visitors/Element/EventHandler.js +++ b/src/generators/dom/visitors/Element/EventHandler.js @@ -37,8 +37,9 @@ export default function visitEventHandler ( generator, block, state, node, attri const listName = block.listNames.get( name ); const indexName = block.indexNames.get( name ); + const contextName = block.contexts.get( name ); - return `var ${listName} = ${_this}._svelte.${listName}, ${indexName} = ${_this}._svelte.${indexName}, ${name} = ${listName}[${indexName}];`; + return `var ${listName} = ${_this}._svelte.${listName}, ${indexName} = ${_this}._svelte.${indexName}, ${contextName} = ${listName}[${indexName}];`; }); // get a name for the event handler that is globally unique From 206937236965962e050993c66e81874e310a174e Mon Sep 17 00:00:00 2001 From: Rich Harris Date: Tue, 11 Apr 2017 11:35:26 -0400 Subject: [PATCH 3/3] remove hardcoded component reference --- src/generators/dom/visitors/Element/meta/Window.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/generators/dom/visitors/Element/meta/Window.js b/src/generators/dom/visitors/Element/meta/Window.js index 252ac1e2f7a0..b941721f0e25 100644 --- a/src/generators/dom/visitors/Element/meta/Window.js +++ b/src/generators/dom/visitors/Element/meta/Window.js @@ -109,19 +109,19 @@ export default function visitWindow ( generator, block, node ) { block.builders.create.addBlock( deindent` function ${observerCallback} () { if ( ${lock} ) return; - var x = ${bindings.scrollX ? `component.get( '${bindings.scrollX}' )` : `window.scrollX`}; - var y = ${bindings.scrollY ? `component.get( '${bindings.scrollY}' )` : `window.scrollY`}; + var x = ${bindings.scrollX ? `${block.component}.get( '${bindings.scrollX}' )` : `window.scrollX`}; + var y = ${bindings.scrollY ? `${block.component}.get( '${bindings.scrollY}' )` : `window.scrollY`}; window.scrollTo( x, y ); }; ` ); - if ( bindings.scrollX ) block.builders.create.addLine( `component.observe( '${bindings.scrollX}', ${observerCallback} );` ); - if ( bindings.scrollY ) block.builders.create.addLine( `component.observe( '${bindings.scrollY}', ${observerCallback} );` ); + if ( bindings.scrollX ) block.builders.create.addLine( `${block.component}.observe( '${bindings.scrollX}', ${observerCallback} );` ); + if ( bindings.scrollY ) block.builders.create.addLine( `${block.component}.observe( '${bindings.scrollY}', ${observerCallback} );` ); } else if ( bindings.scrollX || bindings.scrollY ) { const isX = !!bindings.scrollX; block.builders.create.addBlock( deindent` - component.observe( '${bindings.scrollX || bindings.scrollY}', function ( ${isX ? 'x' : 'y'} ) { + ${block.component}.observe( '${bindings.scrollX || bindings.scrollY}', function ( ${isX ? 'x' : 'y'} ) { if ( ${lock} ) return; window.scrollTo( ${isX ? 'x, window.scrollY' : 'window.scrollX, y' } ); }); @@ -133,7 +133,7 @@ export default function visitWindow ( generator, block, node ) { const handlerName = block.getUniqueName( `onlinestatuschanged` ); block.builders.create.addBlock( deindent` function ${handlerName} ( event ) { - component.set({ ${bindings.online}: navigator.onLine }); + ${block.component}.set({ ${bindings.online}: navigator.onLine }); }; window.addEventListener( 'online', ${handlerName} ); window.addEventListener( 'offline', ${handlerName} );