Merge pull request #212 from sveltejs/helpers

helpers
pull/215/head
Rich Harris 9 years ago committed by GitHub
commit 18683349f9

@ -0,0 +1 @@
src/shared

@ -5,11 +5,13 @@ import namespaces from '../../utils/namespaces.js';
import processCss from '../shared/css/process.js'; import processCss from '../shared/css/process.js';
import visitors from './visitors/index.js'; import visitors from './visitors/index.js';
import Generator from '../Generator.js'; import Generator from '../Generator.js';
import * as shared from '../../shared/index.js';
class DomGenerator extends Generator { class DomGenerator extends Generator {
constructor ( parsed, source, names, visitors ) { constructor ( parsed, source, names, visitors ) {
super( parsed, source, names, visitors ); super( parsed, source, names, visitors );
this.renderers = []; this.renderers = [];
this.uses = {};
} }
addElement ( name, renderStatement, needsIdentifier = false ) { addElement ( name, renderStatement, needsIdentifier = false ) {
@ -21,15 +23,13 @@ class DomGenerator extends Generator {
this.createMountStatement( name ); this.createMountStatement( name );
} else { } else {
this.current.builders.init.addLine( this.uses.appendNode = true;
`${this.current.target}.appendChild( ${renderStatement} );` this.current.builders.init.addLine( `appendNode( ${renderStatement}, ${this.current.target} );` );
);
} }
if ( isToplevel ) { if ( isToplevel ) {
this.current.builders.detach.addLine( this.uses.detachNode = true;
`${name}.parentNode.removeChild( ${name} );` this.current.builders.detach.addLine( `detachNode( ${name} );` );
);
} }
} }
@ -54,19 +54,38 @@ class DomGenerator extends Generator {
if ( fragment.key ) properties.addBlock( `key: key,` ); if ( fragment.key ) properties.addBlock( `key: key,` );
if ( fragment.builders.mount.isEmpty() ) {
this.uses.noop = true;
properties.addBlock( `mount: noop,` );
} else {
properties.addBlock( deindent` properties.addBlock( deindent`
mount: function ( target, anchor ) { mount: function ( target, anchor ) {
${fragment.builders.mount} ${fragment.builders.mount}
}, },
` );
}
if ( fragment.builders.update.isEmpty() ) {
this.uses.noop = true;
properties.addBlock( `update: noop,` );
} else {
properties.addBlock( deindent`
update: function ( changed, ${fragment.params} ) { update: function ( changed, ${fragment.params} ) {
${fragment.builders.update} ${fragment.builders.update}
}, },
` );
}
if ( fragment.builders.teardown.isEmpty() ) {
this.uses.noop = true;
properties.addBlock( `teardown: noop,` );
} else {
properties.addBlock( deindent`
teardown: function ( detach ) { teardown: function ( detach ) {
${fragment.builders.teardown} ${fragment.builders.teardown}
} },
` ); ` );
}
this.renderers.push( deindent` this.renderers.push( deindent`
function ${fragment.name} ( ${fragment.params}, component${fragment.key ? `, key` : ''} ) { function ${fragment.name} ( ${fragment.params}, component${fragment.key ? `, key` : ''} ) {
@ -80,18 +99,18 @@ class DomGenerator extends Generator {
} }
createAnchor ( name, description = '' ) { createAnchor ( name, description = '' ) {
const renderStatement = `document.createComment( ${JSON.stringify( description )} )`; this.uses.createComment = true;
const renderStatement = `createComment( ${JSON.stringify( description )} )`;
this.addElement( name, renderStatement, true ); this.addElement( name, renderStatement, true );
} }
createMountStatement ( name ) { createMountStatement ( name ) {
if ( this.current.target === 'target' ) { if ( this.current.target === 'target' ) {
this.current.builders.mount.addLine( this.uses.insertNode = true;
`target.insertBefore( ${name}, anchor );` this.current.builders.mount.addLine( `insertNode( ${name}, target, anchor );` );
);
} else { } else {
this.current.builders.init.addLine( this.uses.appendNode = true;
`${this.current.target}.appendChild( ${name} );` ); this.current.builders.init.addLine( `appendNode( ${name}, ${this.current.target} );` );
} }
} }
@ -158,8 +177,8 @@ export default function dom ( parsed, source, options, names ) {
set: new CodeBuilder() set: new CodeBuilder()
}; };
builders.set.addLine( 'var oldState = state;' ); builders.set.addLine( 'var oldState = this._state;' );
builders.set.addLine( 'state = Object.assign( {}, oldState, newState );' ); builders.set.addLine( 'this._state = Object.assign( {}, oldState, newState );' );
if ( computations.length ) { if ( computations.length ) {
const builder = new CodeBuilder(); const builder = new CodeBuilder();
@ -178,13 +197,14 @@ export default function dom ( parsed, source, options, names ) {
} }
` ); ` );
builders.set.addLine( `applyComputations( state, newState, oldState )` ); builders.set.addLine( `applyComputations( this._state, newState, oldState )` );
} }
// TODO is the `if` necessary?
builders.set.addBlock( deindent` builders.set.addBlock( deindent`
dispatchObservers( observers.immediate, newState, oldState ); dispatchObservers( this, this._observers.pre, newState, oldState );
if ( mainFragment ) mainFragment.update( newState, state ); if ( this._fragment ) this._fragment.update( newState, this._state );
dispatchObservers( observers.deferred, newState, oldState ); dispatchObservers( this, this._observers.post, newState, oldState );
` ); ` );
if ( parsed.js ) { if ( parsed.js ) {
@ -192,12 +212,15 @@ export default function dom ( parsed, source, options, names ) {
} }
if ( parsed.css && options.css !== false ) { if ( parsed.css && options.css !== false ) {
generator.uses.appendNode = true;
generator.uses.createElement = true;
builders.main.addBlock( deindent` builders.main.addBlock( deindent`
let addedCss = false; let addedCss = false;
function addCss () { function addCss () {
var style = document.createElement( 'style' ); var style = createElement( 'style' );
style.textContent = ${JSON.stringify( processCss( parsed ) )}; style.textContent = ${JSON.stringify( processCss( parsed ) )};
document.head.appendChild( style ); appendNode( style, document.head );
addedCss = true; addedCss = true;
} }
@ -212,29 +235,29 @@ export default function dom ( parsed, source, options, names ) {
} }
if ( generator.hasComponents ) { if ( generator.hasComponents ) {
builders.init.addLine( `this.__renderHooks = [];` ); builders.init.addLine( `this._renderHooks = [];` );
} }
if ( generator.hasComplexBindings ) { if ( generator.hasComplexBindings ) {
builders.init.addBlock( deindent` builders.init.addBlock( deindent`
this.__bindings = []; this._bindings = [];
var mainFragment = renderMainFragment( state, this ); this._fragment = renderMainFragment( this._state, this );
if ( options.target ) this._mount( options.target ); if ( options.target ) this._fragment.mount( options.target, null );
while ( this.__bindings.length ) this.__bindings.pop()(); while ( this._bindings.length ) this._bindings.pop()();
` ); ` );
builders.set.addLine( `while ( this.__bindings.length ) this.__bindings.pop()();` ); builders.set.addLine( `while ( this._bindings.length ) this._bindings.pop()();` );
} else { } else {
builders.init.addBlock( deindent` builders.init.addBlock( deindent`
var mainFragment = renderMainFragment( state, this ); this._fragment = renderMainFragment( this._state, this );
if ( options.target ) this._mount( options.target ); if ( options.target ) this._fragment.mount( options.target, null );
` ); ` );
} }
if ( generator.hasComponents ) { if ( generator.hasComponents ) {
const statement = deindent` const statement = deindent`
while ( this.__renderHooks.length ) { while ( this._renderHooks.length ) {
var hook = this.__renderHooks.pop(); var hook = this._renderHooks.pop();
hook.fn.call( hook.context ); hook.fn.call( hook.context );
} }
`; `;
@ -245,8 +268,8 @@ export default function dom ( parsed, source, options, names ) {
if ( templateProperties.onrender ) { if ( templateProperties.onrender ) {
builders.init.addBlock( 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 {
template.onrender.call( this ); template.onrender.call( this );
} }
@ -258,111 +281,56 @@ export default function dom ( parsed, source, options, names ) {
builders.main.addBlock( deindent` builders.main.addBlock( deindent`
function ${name} ( options ) { function ${name} ( options ) {
options = options || {}; options = options || {};
${generator.usesRefs ? `\nthis.refs = {}` : ``}
this._state = ${initialState};${templateProperties.computed ? `\napplyComputations( this._state, this._state, {} );` : ``}
var component = this;${generator.usesRefs ? `\nthis.refs = {}` : ``} this._observers = {
var state = ${initialState};${templateProperties.computed ? `\napplyComputations( state, state, {} );` : ``} pre: Object.create( null ),
post: Object.create( null )
var observers = {
immediate: Object.create( null ),
deferred: Object.create( null )
}; };
var callbacks = Object.create( null ); this._handlers = Object.create( null );
function dispatchObservers ( group, newState, oldState ) {
for ( var key in group ) {
if ( !( key in newState ) ) continue;
var newValue = newState[ key ];
var oldValue = oldState[ key ];
if ( newValue === oldValue && typeof newValue !== 'object' ) continue;
var callbacks = group[ key ]; this._root = options._root;
if ( !callbacks ) continue; this._yield = options._yield;
for ( var i = 0; i < callbacks.length; i += 1 ) { ${builders.init}
var callback = callbacks[i];
if ( callback.__calling ) continue;
callback.__calling = true;
callback.call( component, newValue, oldValue );
callback.__calling = false;
}
}
}
this.fire = function fire ( eventName, data ) {
var handlers = eventName in callbacks && callbacks[ eventName ].slice();
if ( !handlers ) return;
for ( var i = 0; i < handlers.length; i += 1 ) {
handlers[i].call( this, data );
} }
}; ` );
this.get = function get ( key ) {
return key ? state[ key ] : state;
};
this.set = function set ( newState ) {
${builders.set}
};
this._mount = function mount ( target, anchor ) { if ( templateProperties.methods ) {
mainFragment.mount( target, anchor ); builders.main.addBlock( `${name}.prototype = template.methods;` );
} }
this.observe = function ( key, callback, options ) { builders.main.addBlock( deindent`
var group = ( options && options.defer ) ? observers.deferred : observers.immediate; ${name}.prototype.get = ${shared.get};
( group[ key ] || ( group[ key ] = [] ) ).push( callback ); ${name}.prototype.fire = ${shared.fire};
if ( !options || options.init !== false ) { ${name}.prototype.observe = ${shared.observe};
callback.__calling = true;
callback.call( component, state[ key ] );
callback.__calling = false;
}
return { ${name}.prototype.on = ${shared.on};
cancel: function () {
var index = group[ key ].indexOf( callback );
if ( ~index ) group[ key ].splice( index, 1 );
}
};
};
this.on = function on ( eventName, handler ) { ${name}.prototype.set = function set ( newState ) {
var handlers = callbacks[ eventName ] || ( callbacks[ eventName ] = [] ); ${builders.set}
handlers.push( handler );
return {
cancel: function () {
var index = handlers.indexOf( handler );
if ( ~index ) handlers.splice( index, 1 );
}
};
}; };
this.teardown = function teardown ( detach ) { ${name}.prototype.teardown = function teardown ( detach ) {
this.fire( 'teardown' );${templateProperties.onteardown ? `\ntemplate.onteardown.call( this );` : ``} this.fire( 'teardown' );${templateProperties.onteardown ? `\ntemplate.onteardown.call( this );` : ``}
mainFragment.teardown( detach !== false ); this._fragment.teardown( detach !== false );
mainFragment = null; this._fragment = null;
state = {}; this._state = {};
}; };
this.root = options.root;
this.yield = options.yield;
${builders.init}
}
` ); ` );
if ( templateProperties.methods ) { builders.main.addBlock( shared.dispatchObservers.toString() );
builders.main.addBlock( `${name}.prototype = template.methods;` );
} Object.keys( generator.uses ).forEach( key => {
const fn = shared[ key ]; // eslint-disable-line import/namespace
builders.main.addBlock( fn.toString() );
});
return generator.generate( builders.main.toString(), options, { name, format } ); return generator.generate( builders.main.toString(), options, { name, format } );
} }

@ -26,7 +26,7 @@ export default {
const componentInitProperties = [ const componentInitProperties = [
`target: ${!isToplevel ? generator.current.target: 'null'}`, `target: ${!isToplevel ? generator.current.target: 'null'}`,
'root: component.root || component' '_root: component._root || component'
]; ];
// Component has children, put them in a separate {{yield}} block // Component has children, put them in a separate {{yield}} block
@ -43,7 +43,7 @@ export default {
`${name}_yieldFragment.update( changed, root );` `${name}_yieldFragment.update( changed, root );`
); );
componentInitProperties.push(`yield: ${name}_yieldFragment`); componentInitProperties.push( `_yield: ${name}_yieldFragment`);
} }
const statements = []; const statements = [];
@ -83,7 +83,7 @@ export default {
` ); ` );
if ( isToplevel ) { if ( isToplevel ) {
generator.current.builders.mount.addLine( `${name}._mount( target, anchor );` ); generator.current.builders.mount.addLine( `${name}._fragment.mount( target, anchor );` );
} }
if ( local.dynamicAttributes.length ) { if ( local.dynamicAttributes.length ) {

@ -57,9 +57,19 @@ export default {
local.update.addBlock( updates ); local.update.addBlock( updates );
} }
let render = local.namespace ? let render;
`var ${name} = document.createElementNS( '${local.namespace}', '${node.name}' );` :
`var ${name} = document.createElement( '${node.name}' );`; if ( local.namespace ) {
if ( local.namespace === 'http://www.w3.org/2000/svg' ) {
generator.uses.createSvgElement = true;
render = `var ${name} = createSvgElement( '${node.name}' )`;
} else {
render = `var ${name} = document.createElementNS( '${local.namespace}', '${node.name}' );`;
}
} else {
generator.uses.createElement = true;
render = `var ${name} = createElement( '${node.name}' );`;
}
if ( generator.cssId && !generator.elementDepth ) { if ( generator.cssId && !generator.elementDepth ) {
render += `\n${name}.setAttribute( '${generator.cssId}', '' );`; render += `\n${name}.setAttribute( '${generator.cssId}', '' );`;
@ -67,7 +77,8 @@ export default {
local.init.addLineAtStart( render ); local.init.addLineAtStart( render );
if ( isToplevel ) { if ( isToplevel ) {
generator.current.builders.detach.addLine( `${name}.parentNode.removeChild( ${name} );` ); generator.uses.detachNode = true;
generator.current.builders.detach.addLine( `detachNode( ${name} );` );
} }
// special case bound <option> without a value attribute // special case bound <option> without a value attribute

@ -7,7 +7,8 @@ export default {
generator.addSourcemapLocations( node.expression ); generator.addSourcemapLocations( node.expression );
const { snippet } = generator.contextualise( node.expression ); const { snippet } = generator.contextualise( node.expression );
generator.addElement( name, `document.createTextNode( ${snippet} )`, true ); generator.uses.createText = true;
generator.addElement( name, `createText( ${snippet} )`, true );
generator.current.builders.update.addBlock( deindent` generator.current.builders.update.addBlock( deindent`
${name}.data = ${snippet}; ${name}.data = ${snippet};

@ -9,11 +9,13 @@ export default {
// we would have used comments here, but the `insertAdjacentHTML` api only // we would have used comments here, but the `insertAdjacentHTML` api only
// exists for `Element`s. // exists for `Element`s.
generator.uses.createElement = true;
const before = `${name}_before`; const before = `${name}_before`;
generator.addElement( before, `document.createElement( 'noscript' )`, true ); generator.addElement( before, `createElement( 'noscript' )`, true );
const after = `${name}_after`; const after = `${name}_after`;
generator.addElement( after, `document.createElement( 'noscript' )`, true ); generator.addElement( after, `createElement( 'noscript' )`, true );
const isToplevel = generator.current.localElementDepth === 0; const isToplevel = generator.current.localElementDepth === 0;

@ -5,6 +5,8 @@ export default {
} }
const name = generator.current.getUniqueName( `text` ); const name = generator.current.getUniqueName( `text` );
generator.addElement( name, `document.createTextNode( ${JSON.stringify( node.data )} )`, false ); generator.addElement( name, `createText( ${JSON.stringify( node.data )} )`, false );
generator.uses.createText = true;
} }
}; };

@ -4,11 +4,11 @@ export default {
generator.createAnchor( anchor, 'yield' ); generator.createAnchor( anchor, 'yield' );
generator.current.builders.mount.addLine( generator.current.builders.mount.addLine(
`component.yield && component.yield.mount( ${generator.current.target}, ${anchor} );` `component._yield && component._yield.mount( ${generator.current.target}, ${anchor} );`
); );
generator.current.builders.teardown.addLine( generator.current.builders.teardown.addLine(
`component.yield && component.yield.teardown( detach );` `component._yield && component._yield.teardown( detach );`
); );
} }
}; };

@ -87,7 +87,7 @@ export default function createBinding ( generator, node, attribute, current, loc
local.init.addBlock( deindent` local.init.addBlock( deindent`
var ${local.name}_updating = false; var ${local.name}_updating = false;
component.__bindings.push( function () { component._bindings.push( function () {
${local.name}.observe( '${attribute.name}', function ( value ) { ${local.name}.observe( '${attribute.name}', function ( value ) {
${local.name}_updating = true; ${local.name}_updating = true;
${setter} ${setter}
@ -128,6 +128,6 @@ export default function createBinding ( generator, node, attribute, current, loc
if ( node.name === 'select' ) { if ( node.name === 'select' ) {
generator.hasComplexBindings = true; generator.hasComplexBindings = true;
local.init.addLine( `component.__bindings.push( ${handler} )` ); local.init.addLine( `component._bindings.push( ${handler} )` );
} }
} }

@ -0,0 +1,27 @@
export function appendNode ( node, target ) {
target.appendChild( node );
}
export function insertNode ( node, target, anchor ) {
target.insertBefore( node, anchor );
}
export function detachNode ( node ) {
node.parentNode.removeChild( node );
}
export function createElement ( name ) {
return document.createElement( name );
}
export function createSvgElement ( name ) {
return document.createElementNS( 'http://www.w3.org/2000/svg', name );
}
export function createText ( data ) {
return document.createTextNode( data );
}
export function createComment ( data ) {
return document.createComment( data );
}

@ -0,0 +1,27 @@
export * from './dom.js';
export * from './methods.js';
export function noop () {}
export function dispatchObservers ( component, group, newState, oldState ) {
for ( var key in group ) {
if ( !( key in newState ) ) continue;
var newValue = newState[ key ];
var oldValue = oldState[ key ];
if ( newValue === oldValue && typeof newValue !== 'object' ) continue;
var callbacks = group[ key ];
if ( !callbacks ) continue;
for ( var i = 0; i < callbacks.length; i += 1 ) {
var callback = callbacks[i];
if ( callback.__calling ) continue;
callback.__calling = true;
callback.call( component, newValue, oldValue );
callback.__calling = false;
}
}
}

@ -0,0 +1,43 @@
export function get ( key ) {
return key ? this._state[ key ] : this._state;
}
export function fire ( eventName, data ) {
var handlers = eventName in this._handlers && this._handlers[ eventName ].slice();
if ( !handlers ) return;
for ( var i = 0; i < handlers.length; i += 1 ) {
handlers[i].call( this, data );
}
}
export function observe ( key, callback, options ) {
var group = ( options && options.defer ) ? this._observers.pre : this._observers.post;
( group[ key ] || ( group[ key ] = [] ) ).push( callback );
if ( !options || options.init !== false ) {
callback.__calling = true;
callback.call( this, this._state[ key ] );
callback.__calling = false;
}
return {
cancel: function () {
var index = group[ key ].indexOf( callback );
if ( ~index ) group[ key ].splice( index, 1 );
}
};
}
export function on ( eventName, handler ) {
var handlers = this._handlers[ eventName ] || ( this._handlers[ eventName ] = [] );
handlers.push( handler );
return {
cancel: function () {
var index = handlers.indexOf( handler );
if ( ~index ) handlers.splice( index, 1 );
}
};
}
Loading…
Cancel
Save