|
|
|
@ -1,5 +1,6 @@
|
|
|
|
|
import MagicString, { Bundle } from 'magic-string';
|
|
|
|
|
import { walk } from 'estree-walker';
|
|
|
|
|
import CodeBuilder from '../utils/CodeBuilder.js';
|
|
|
|
|
import deindent from '../utils/deindent.js';
|
|
|
|
|
import isReference from '../utils/isReference.js';
|
|
|
|
|
import counter from './utils/counter.js';
|
|
|
|
@ -18,31 +19,32 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
addElement ( name, renderStatement, needsIdentifier = false ) {
|
|
|
|
|
const isToplevel = generator.current.localElementDepth === 0;
|
|
|
|
|
if ( needsIdentifier || isToplevel ) {
|
|
|
|
|
generator.current.initStatements.push( deindent`
|
|
|
|
|
var ${name} = ${renderStatement};
|
|
|
|
|
` );
|
|
|
|
|
generator.current.builders.init.addLine(
|
|
|
|
|
`var ${name} = ${renderStatement};`
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
generator.createMountStatement( name );
|
|
|
|
|
} else {
|
|
|
|
|
generator.current.initStatements.push( deindent`
|
|
|
|
|
${generator.current.target}.appendChild( ${renderStatement} );
|
|
|
|
|
` );
|
|
|
|
|
generator.current.builders.init.addLine(
|
|
|
|
|
`${generator.current.target}.appendChild( ${renderStatement} );`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( isToplevel ) {
|
|
|
|
|
generator.current.detachStatements.push( deindent`
|
|
|
|
|
${name}.parentNode.removeChild( ${name} );
|
|
|
|
|
` );
|
|
|
|
|
generator.current.builders.detach.addLine(
|
|
|
|
|
`${name}.parentNode.removeChild( ${name} );`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
createMountStatement ( name ) {
|
|
|
|
|
if ( generator.current.target === 'target' ) {
|
|
|
|
|
generator.current.mountStatements.push( deindent`
|
|
|
|
|
target.insertBefore( ${name}, anchor );
|
|
|
|
|
` );
|
|
|
|
|
generator.current.builders.mount.addLine(
|
|
|
|
|
`target.insertBefore( ${name}, anchor );`
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
generator.current.initStatements.push( deindent`
|
|
|
|
|
${generator.current.target}.appendChild( ${name} );
|
|
|
|
|
` );
|
|
|
|
|
generator.current.builders.init.addLine(
|
|
|
|
|
`${generator.current.target}.appendChild( ${name} );` );
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
@ -58,13 +60,7 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
name,
|
|
|
|
|
target: 'target',
|
|
|
|
|
localElementDepth: 0,
|
|
|
|
|
|
|
|
|
|
initStatements: [],
|
|
|
|
|
mountStatements: [],
|
|
|
|
|
updateStatements: [],
|
|
|
|
|
detachStatements: [],
|
|
|
|
|
teardownStatements: [],
|
|
|
|
|
|
|
|
|
|
builders: generator.getBuilders(),
|
|
|
|
|
getUniqueName: generator.getUniqueNameMaker()
|
|
|
|
|
});
|
|
|
|
|
// walk the children here
|
|
|
|
@ -77,37 +73,36 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
|
|
|
|
|
addRenderer ( fragment ) {
|
|
|
|
|
if ( fragment.autofocus ) {
|
|
|
|
|
fragment.initStatements.push( `${fragment.autofocus}.focus();` );
|
|
|
|
|
fragment.builders.init.addLine( `${fragment.autofocus}.focus();` );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const detachStatements = fragment.detachStatements.join( '\n\n' );
|
|
|
|
|
const teardownStatements = fragment.teardownStatements.join( '\n\n' );
|
|
|
|
|
|
|
|
|
|
const detachBlock = deindent`
|
|
|
|
|
if ( detach ) {
|
|
|
|
|
${detachStatements}
|
|
|
|
|
}
|
|
|
|
|
`;
|
|
|
|
|
// minor hack – we need to ensure that any {{{triples}}} are detached
|
|
|
|
|
// first, so we append normal detach statements to detachRaw
|
|
|
|
|
fragment.builders.detachRaw.addBlock( fragment.builders.detach );
|
|
|
|
|
|
|
|
|
|
const teardownBlock = deindent`
|
|
|
|
|
${teardownStatements}${detachStatements ? `\n\n${detachBlock}` : ``}
|
|
|
|
|
`;
|
|
|
|
|
if ( !fragment.builders.detachRaw.isEmpty() ) {
|
|
|
|
|
fragment.builders.teardown.addBlock( deindent`
|
|
|
|
|
if ( detach ) {
|
|
|
|
|
${fragment.builders.detachRaw}
|
|
|
|
|
}
|
|
|
|
|
` );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
renderers.push( deindent`
|
|
|
|
|
function ${fragment.name} ( ${fragment.params}, component ) {
|
|
|
|
|
${fragment.initStatements.join( '\n\n' )}
|
|
|
|
|
${fragment.builders.init}
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
mount: function ( target, anchor ) {
|
|
|
|
|
${fragment.mountStatements.join( '\n\n' )}
|
|
|
|
|
${fragment.builders.mount}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
update: function ( changed, ${fragment.params} ) {
|
|
|
|
|
${fragment.updateStatements.join( '\n\n' )}
|
|
|
|
|
${fragment.builders.update}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
teardown: function ( detach ) {
|
|
|
|
|
${teardownBlock}
|
|
|
|
|
${fragment.builders.teardown}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
@ -174,6 +169,17 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
|
|
|
|
|
events: {},
|
|
|
|
|
|
|
|
|
|
getBuilders () {
|
|
|
|
|
return {
|
|
|
|
|
init: new CodeBuilder(),
|
|
|
|
|
mount: new CodeBuilder(),
|
|
|
|
|
update: new CodeBuilder(),
|
|
|
|
|
detach: new CodeBuilder(),
|
|
|
|
|
detachRaw: new CodeBuilder(),
|
|
|
|
|
teardown: new CodeBuilder()
|
|
|
|
|
};
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
getUniqueName: counter( names ),
|
|
|
|
|
|
|
|
|
|
getUniqueNameMaker () {
|
|
|
|
@ -293,12 +299,6 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
elementDepth: 0,
|
|
|
|
|
localElementDepth: 0,
|
|
|
|
|
|
|
|
|
|
initStatements: [],
|
|
|
|
|
mountStatements: [],
|
|
|
|
|
updateStatements: [],
|
|
|
|
|
detachStatements: [],
|
|
|
|
|
teardownStatements: [],
|
|
|
|
|
|
|
|
|
|
contexts: {},
|
|
|
|
|
indexes: {},
|
|
|
|
|
|
|
|
|
@ -306,6 +306,7 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
indexNames: {},
|
|
|
|
|
listNames: {},
|
|
|
|
|
|
|
|
|
|
builders: generator.getBuilders(),
|
|
|
|
|
getUniqueName: generator.getUniqueNameMaker()
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
@ -313,15 +314,17 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
|
|
|
|
|
generator.addRenderer( generator.pop() );
|
|
|
|
|
|
|
|
|
|
const topLevelStatements = [];
|
|
|
|
|
const builders = {
|
|
|
|
|
main: new CodeBuilder(),
|
|
|
|
|
init: new CodeBuilder(),
|
|
|
|
|
set: new CodeBuilder()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const setStatements = [ deindent`
|
|
|
|
|
var oldState = state;
|
|
|
|
|
state = Object.assign( {}, oldState, newState );
|
|
|
|
|
` ];
|
|
|
|
|
builders.set.addLine( 'var oldState = state;' );
|
|
|
|
|
builders.set.addLine( 'state = Object.assign( {}, oldState, newState );' );
|
|
|
|
|
|
|
|
|
|
if ( templateProperties.computed ) {
|
|
|
|
|
const statements = [];
|
|
|
|
|
const builder = new CodeBuilder();
|
|
|
|
|
const dependencies = new Map();
|
|
|
|
|
|
|
|
|
|
templateProperties.computed.properties.forEach( prop => {
|
|
|
|
@ -343,7 +346,7 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
const deps = dependencies.get( key );
|
|
|
|
|
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( ' || ' )} ) {
|
|
|
|
|
state.${key} = newState.${key} = template.computed.${key}( ${deps.map( dep => `state.${dep}` ).join( ', ' )} );
|
|
|
|
|
}
|
|
|
|
@ -352,57 +355,49 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
|
|
|
|
|
templateProperties.computed.properties.forEach( prop => visit( prop.key.name ) );
|
|
|
|
|
|
|
|
|
|
topLevelStatements.push( deindent`
|
|
|
|
|
builders.main.addBlock( deindent`
|
|
|
|
|
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 );
|
|
|
|
|
if ( mainFragment ) mainFragment.update( newState, state );
|
|
|
|
|
dispatchObservers( observers.deferred, newState, oldState );
|
|
|
|
|
` );
|
|
|
|
|
|
|
|
|
|
const importBlock = imports
|
|
|
|
|
.map( ( declaration, i ) => {
|
|
|
|
|
if ( format === 'es' ) {
|
|
|
|
|
return source.slice( declaration.start, declaration.end );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
imports.forEach( ( declaration, i ) => {
|
|
|
|
|
if ( format === 'es' ) {
|
|
|
|
|
builders.main.addLine( source.slice( declaration.start, declaration.end ) );
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const statements = namedImports.map( specifier => {
|
|
|
|
|
return `var ${specifier.local.name} = ${name}.${specifier.imported.name}`;
|
|
|
|
|
});
|
|
|
|
|
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' );
|
|
|
|
|
|
|
|
|
|
if ( defaultImport ) {
|
|
|
|
|
statements.push( `${name} = ( ${name} && ${name}.__esModule ) ? ${name}['default'] : ${name};` );
|
|
|
|
|
}
|
|
|
|
|
const name = ( defaultImport || namespaceImport ) ? ( defaultImport || namespaceImport ).local.name : `__import${i}`;
|
|
|
|
|
declaration.name = name; // hacky but makes life a bit easier later
|
|
|
|
|
|
|
|
|
|
return statements.join( '\n' );
|
|
|
|
|
})
|
|
|
|
|
.filter( Boolean )
|
|
|
|
|
.join( '\n' );
|
|
|
|
|
namedImports.forEach( specifier => {
|
|
|
|
|
builders.main.addLine( `var ${specifier.local.name} = ${name}.${specifier.imported.name}` );
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if ( parsed.js ) {
|
|
|
|
|
if ( imports.length ) {
|
|
|
|
|
topLevelStatements.push( importBlock );
|
|
|
|
|
if ( defaultImport ) {
|
|
|
|
|
builders.main.addLine( `${name} = ( ${name} && ${name}.__esModule ) ? ${name}['default'] : ${name};` );
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
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 ) {
|
|
|
|
|
topLevelStatements.push( deindent`
|
|
|
|
|
builders.main.addBlock( deindent`
|
|
|
|
|
let addedCss = false;
|
|
|
|
|
function addCss () {
|
|
|
|
|
var style = document.createElement( 'style' );
|
|
|
|
@ -414,33 +409,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 initStatements = [];
|
|
|
|
|
|
|
|
|
|
if ( parsed.css && options.css !== false ) {
|
|
|
|
|
initStatements.push( `if ( !addedCss ) addCss();` );
|
|
|
|
|
builders.init.addLine( `if ( !addedCss ) addCss();` );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( generator.hasComponents ) {
|
|
|
|
|
initStatements.push( deindent`
|
|
|
|
|
this.__renderHooks = [];
|
|
|
|
|
` );
|
|
|
|
|
builders.init.addLine( `this.__renderHooks = [];` );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( generator.hasComplexBindings ) {
|
|
|
|
|
initStatements.push( deindent`
|
|
|
|
|
builders.init.addBlock( deindent`
|
|
|
|
|
this.__bindings = [];
|
|
|
|
|
var mainFragment = renderMainFragment( state, this );
|
|
|
|
|
if ( options.target ) this._mount( options.target );
|
|
|
|
|
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 {
|
|
|
|
|
initStatements.push( deindent`
|
|
|
|
|
builders.init.addBlock( deindent`
|
|
|
|
|
var mainFragment = renderMainFragment( state, this );
|
|
|
|
|
if ( options.target ) this._mount( options.target );
|
|
|
|
|
` );
|
|
|
|
@ -454,12 +446,12 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
}
|
|
|
|
|
`;
|
|
|
|
|
|
|
|
|
|
initStatements.push( statement );
|
|
|
|
|
setStatements.push( statement );
|
|
|
|
|
builders.init.addBlock( statement );
|
|
|
|
|
builders.set.addBlock( statement );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( templateProperties.onrender ) {
|
|
|
|
|
initStatements.push( deindent`
|
|
|
|
|
builders.init.addBlock( deindent`
|
|
|
|
|
if ( options.root ) {
|
|
|
|
|
options.root.__renderHooks.push({ fn: template.onrender, context: this });
|
|
|
|
|
} else {
|
|
|
|
@ -470,7 +462,7 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
|
|
|
|
|
const initialState = templateProperties.data ? `Object.assign( template.data(), options.data )` : `options.data || {}`;
|
|
|
|
|
|
|
|
|
|
topLevelStatements.push( deindent`
|
|
|
|
|
builders.main.addBlock( deindent`
|
|
|
|
|
function ${constructorName} ( options ) {
|
|
|
|
|
options = options || {};
|
|
|
|
|
|
|
|
|
@ -521,7 +513,7 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.set = function set ( newState ) {
|
|
|
|
|
${setStatements.join( '\n\n' )}
|
|
|
|
|
${builders.set}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this._mount = function mount ( target, anchor ) {
|
|
|
|
@ -571,15 +563,15 @@ export default function generate ( parsed, source, options, names ) {
|
|
|
|
|
this.root = options.root;
|
|
|
|
|
this.yield = options.yield;
|
|
|
|
|
|
|
|
|
|
${initStatements.join( '\n\n' )}
|
|
|
|
|
${builders.init}
|
|
|
|
|
}
|
|
|
|
|
` );
|
|
|
|
|
|
|
|
|
|
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+)$/;
|
|
|
|
|
|
|
|
|
|