move generate/index.js over to CodeBuilder

pull/183/head
Rich-Harris 8 years ago
parent a1225d5adf
commit 18baae69c1

@ -1,5 +1,6 @@
import MagicString, { Bundle } from 'magic-string'; import MagicString, { Bundle } from 'magic-string';
import { walk } from 'estree-walker'; import { walk } from 'estree-walker';
import CodeBuilder from '../utils/CodeBuilder.js';
import deindent from '../utils/deindent.js'; import deindent from '../utils/deindent.js';
import isReference from '../utils/isReference.js'; import isReference from '../utils/isReference.js';
import counter from './utils/counter.js'; import counter from './utils/counter.js';
@ -304,15 +305,17 @@ export default function generate ( parsed, source, options, names ) {
generator.addRenderer( generator.pop() ); generator.addRenderer( generator.pop() );
const topLevelStatements = []; const builders = {
main: new CodeBuilder(),
init: new CodeBuilder(),
set: new CodeBuilder()
};
const setStatements = [ deindent` builders.set.addLine( 'var oldState = state;' );
var oldState = state; builders.set.addLine( 'state = Object.assign( {}, oldState, newState );' );
state = Object.assign( {}, oldState, newState );
` ];
if ( templateProperties.computed ) { if ( templateProperties.computed ) {
const statements = []; const builder = new CodeBuilder();
const dependencies = new Map(); const dependencies = new Map();
templateProperties.computed.properties.forEach( prop => { templateProperties.computed.properties.forEach( prop => {
@ -334,7 +337,7 @@ export default function generate ( parsed, source, options, names ) {
const deps = dependencies.get( key ); const deps = dependencies.get( key );
deps.forEach( visit ); deps.forEach( visit );
statements.push( deindent` builder.addBlock( deindent`
if ( ${deps.map( dep => `( '${dep}' in newState && typeof state.${dep} === 'object' || state.${dep} !== oldState.${dep} )` ).join( ' || ' )} ) { if ( ${deps.map( dep => `( '${dep}' in newState && typeof state.${dep} === 'object' || state.${dep} !== oldState.${dep} )` ).join( ' || ' )} ) {
state.${key} = newState.${key} = template.computed.${key}( ${deps.map( dep => `state.${dep}` ).join( ', ' )} ); state.${key} = newState.${key} = template.computed.${key}( ${deps.map( dep => `state.${dep}` ).join( ', ' )} );
} }
@ -343,57 +346,49 @@ export default function generate ( parsed, source, options, names ) {
templateProperties.computed.properties.forEach( prop => visit( prop.key.name ) ); templateProperties.computed.properties.forEach( prop => visit( prop.key.name ) );
topLevelStatements.push( deindent` builders.main.addBlock( deindent`
function applyComputations ( state, newState, oldState ) { function applyComputations ( state, newState, oldState ) {
${statements.join( '\n\n' )} ${builder}
} }
` ); ` );
setStatements.push( `applyComputations( state, newState, oldState )` ); builders.set.addLine( `applyComputations( state, newState, oldState )` );
} }
setStatements.push( deindent` builders.set.addBlock( deindent`
dispatchObservers( observers.immediate, newState, oldState ); dispatchObservers( observers.immediate, newState, oldState );
if ( mainFragment ) mainFragment.update( newState, state ); if ( mainFragment ) mainFragment.update( newState, state );
dispatchObservers( observers.deferred, newState, oldState ); dispatchObservers( observers.deferred, newState, oldState );
` ); ` );
const importBlock = imports imports.forEach( ( declaration, i ) => {
.map( ( declaration, i ) => { if ( format === 'es' ) {
if ( format === 'es' ) { builders.main.addLine( source.slice( declaration.start, declaration.end ) );
return source.slice( declaration.start, declaration.end ); return;
} }
const defaultImport = declaration.specifiers.find( x => x.type === 'ImportDefaultSpecifier' || x.type === 'ImportSpecifier' && x.imported.name === 'default' );
const namespaceImport = declaration.specifiers.find( x => x.type === 'ImportNamespaceSpecifier' );
const namedImports = declaration.specifiers.filter( x => x.type === 'ImportSpecifier' && x.imported.name !== 'default' );
const name = ( defaultImport || namespaceImport ) ? ( defaultImport || namespaceImport ).local.name : `__import${i}`;
declaration.name = name; // hacky but makes life a bit easier later
const statements = namedImports.map( specifier => { const defaultImport = declaration.specifiers.find( x => x.type === 'ImportDefaultSpecifier' || x.type === 'ImportSpecifier' && x.imported.name === 'default' );
return `var ${specifier.local.name} = ${name}.${specifier.imported.name}`; const namespaceImport = declaration.specifiers.find( x => x.type === 'ImportNamespaceSpecifier' );
}); const namedImports = declaration.specifiers.filter( x => x.type === 'ImportSpecifier' && x.imported.name !== 'default' );
if ( defaultImport ) { const name = ( defaultImport || namespaceImport ) ? ( defaultImport || namespaceImport ).local.name : `__import${i}`;
statements.push( `${name} = ( ${name} && ${name}.__esModule ) ? ${name}['default'] : ${name};` ); declaration.name = name; // hacky but makes life a bit easier later
}
return statements.join( '\n' ); namedImports.forEach( specifier => {
}) builders.main.addLine( `var ${specifier.local.name} = ${name}.${specifier.imported.name}` );
.filter( Boolean ) });
.join( '\n' );
if ( parsed.js ) { if ( defaultImport ) {
if ( imports.length ) { builders.main.addLine( `${name} = ( ${name} && ${name}.__esModule ) ? ${name}['default'] : ${name};` );
topLevelStatements.push( importBlock );
} }
});
topLevelStatements.push( `[✂${parsed.js.content.start}-${parsed.js.content.end}✂]` ); if ( parsed.js ) {
builders.main.addBlock( `[✂${parsed.js.content.start}-${parsed.js.content.end}✂]` );
} }
if ( parsed.css && options.css !== false ) { if ( parsed.css && options.css !== false ) {
topLevelStatements.push( deindent` builders.main.addBlock( deindent`
let addedCss = false; let addedCss = false;
function addCss () { function addCss () {
var style = document.createElement( 'style' ); var style = document.createElement( 'style' );
@ -405,33 +400,30 @@ export default function generate ( parsed, source, options, names ) {
` ); ` );
} }
topLevelStatements.push( ...renderers.reverse() ); let i = renderers.length;
while ( i-- ) builders.main.addBlock( renderers[i] );
const constructorName = options.name || 'SvelteComponent'; const constructorName = options.name || 'SvelteComponent';
const initStatements = [];
if ( parsed.css && options.css !== false ) { if ( parsed.css && options.css !== false ) {
initStatements.push( `if ( !addedCss ) addCss();` ); builders.init.addLine( `if ( !addedCss ) addCss();` );
} }
if ( generator.hasComponents ) { if ( generator.hasComponents ) {
initStatements.push( deindent` builders.init.addLine( `this.__renderHooks = [];` );
this.__renderHooks = [];
` );
} }
if ( generator.hasComplexBindings ) { if ( generator.hasComplexBindings ) {
initStatements.push( deindent` builders.init.addBlock( deindent`
this.__bindings = []; this.__bindings = [];
var mainFragment = renderMainFragment( state, this ); var mainFragment = renderMainFragment( state, this );
if ( options.target ) this.mount( options.target ); if ( options.target ) this.mount( options.target );
while ( this.__bindings.length ) this.__bindings.pop()(); while ( this.__bindings.length ) this.__bindings.pop()();
` ); ` );
setStatements.push( `while ( this.__bindings.length ) this.__bindings.pop()();` ); builders.set.addLine( `while ( this.__bindings.length ) this.__bindings.pop()();` );
} else { } else {
initStatements.push( deindent` builders.init.addBlock( deindent`
var mainFragment = renderMainFragment( state, this ); var mainFragment = renderMainFragment( state, this );
if ( options.target ) this.mount( options.target ); if ( options.target ) this.mount( options.target );
` ); ` );
@ -445,12 +437,12 @@ export default function generate ( parsed, source, options, names ) {
} }
`; `;
initStatements.push( statement ); builders.init.addBlock( statement );
setStatements.push( statement ); builders.set.addBlock( statement );
} }
if ( templateProperties.onrender ) { if ( templateProperties.onrender ) {
initStatements.push( deindent` builders.init.addBlock( deindent`
if ( options.root ) { if ( options.root ) {
options.root.__renderHooks.push({ fn: template.onrender, context: this }); options.root.__renderHooks.push({ fn: template.onrender, context: this });
} else { } else {
@ -461,7 +453,7 @@ export default function generate ( parsed, source, options, names ) {
const initialState = templateProperties.data ? `Object.assign( template.data(), options.data )` : `options.data || {}`; const initialState = templateProperties.data ? `Object.assign( template.data(), options.data )` : `options.data || {}`;
topLevelStatements.push( deindent` builders.main.addBlock( deindent`
function ${constructorName} ( options ) { function ${constructorName} ( options ) {
options = options || {}; options = options || {};
@ -512,7 +504,7 @@ export default function generate ( parsed, source, options, names ) {
}; };
this.set = function set ( newState ) { this.set = function set ( newState ) {
${setStatements.join( '\n\n' )} ${builders.set}
}; };
this.mount = function mount ( target, anchor ) { this.mount = function mount ( target, anchor ) {
@ -562,15 +554,15 @@ export default function generate ( parsed, source, options, names ) {
this.root = options.root; this.root = options.root;
this.yield = options.yield; this.yield = options.yield;
${initStatements.join( '\n\n' )} ${builders.init}
} }
` ); ` );
if ( templateProperties.methods ) { if ( templateProperties.methods ) {
topLevelStatements.push( `${constructorName}.prototype = template.methods;` ); builders.main.addBlock( `${constructorName}.prototype = template.methods;` );
} }
const result = topLevelStatements.join( '\n\n' ); const result = builders.main.toString();
const pattern = /\[✂(\d+)-(\d+)$/; const pattern = /\[✂(\d+)-(\d+)$/;

Loading…
Cancel
Save