init store values in situ - fixes #2015

pull/2099/head
Richard Harris 6 years ago
parent 7cde8d06be
commit 534f6e5425

@ -750,7 +750,7 @@ export default class Component {
}); });
} }
rewrite_props() { rewrite_props(get_insert: (name: string) => string) {
const component = this; const component = this;
const { code, instance_scope, instance_scope_map: map, componentOptions } = this; const { code, instance_scope, instance_scope_map: map, componentOptions } = this;
let scope = instance_scope; let scope = instance_scope;
@ -771,73 +771,97 @@ export default class Component {
if (node.type === 'VariableDeclaration') { if (node.type === 'VariableDeclaration') {
if (node.kind === 'var' || scope === instance_scope) { if (node.kind === 'var' || scope === instance_scope) {
let has_exports = false; node.declarations.forEach((declarator, i) => {
let has_only_exports = true; const next = node.declarations[i + 1];
node.declarations.forEach(declarator => { if (declarator.id.type !== 'Identifier') {
extractNames(declarator.id).forEach(name => { const inserts = [];
const variable = component.var_lookup.get(name);
if (name === componentOptions.props_object) { extractNames(declarator.id).forEach(name => {
if (variable.export_name) { const variable = component.var_lookup.get(name);
component.error(declarator, {
code: 'exported-options-props',
message: `Cannot export props binding`
});
}
if (declarator.id.type !== 'Identifier') { if (variable.export_name || name === componentOptions.props_object) {
component.error(declarator, { component.error(declarator, {
code: 'destructured-prop', code: 'destructured-prop',
message: `props binding in destructured declaration is not yet supported` message: `Cannot declare props in destructured declaration`
}); });
} }
// can't use the @ trick here, because we're if (variable.subscribable) {
// manipulating the underlying magic string inserts.push(get_insert(name));
const exclude_internal_props = component.helper('exclude_internal_props'); }
});
const suffix = code.original[declarator.end] === ';'
? ` = ${exclude_internal_props}($$props)`
: ` = ${exclude_internal_props}($$props);`
if (declarator.id.end === declarator.end) { if (inserts.length > 0) {
code.appendLeft(declarator.end, suffix); if (next) {
code.overwrite(declarator.end, next.start, `; ${inserts.join('; ')}; ${node.kind} `);
} else { } else {
code.overwrite(declarator.id.end, declarator.end, suffix); code.appendLeft(declarator.end, `; ${inserts.join('; ')}`);
} }
} }
return;
}
const { name } = declarator.id;
const variable = component.var_lookup.get(name);
if (name === componentOptions.props_object) {
if (variable.export_name) { if (variable.export_name) {
has_exports = true; component.error(declarator, {
code: 'exported-options-props',
message: `Cannot export props binding`
});
} }
if (!variable.export_name || variable.subscribable) { // can't use the @ trick here, because we're
has_only_exports = false; // manipulating the underlying magic string
const exclude_internal_props = component.helper('exclude_internal_props');
const suffix = code.original[declarator.end] === ';'
? ` = ${exclude_internal_props}($$props)`
: ` = ${exclude_internal_props}($$props);`
if (declarator.id.end === declarator.end) {
code.appendLeft(declarator.end, suffix);
} else {
code.overwrite(declarator.id.end, declarator.end, suffix);
} }
}); }
});
if (variable.export_name) {
if (variable.subscribable) {
coalesced_declarations.push({
kind: node.kind,
declarators: [declarator],
insert: get_insert(name)
});
} else {
if (current_group && current_group.kind !== node.kind) {
current_group = null;
}
if (!current_group) {
current_group = { kind: node.kind, declarators: [], insert: null };
coalesced_declarations.push(current_group);
}
if (has_only_exports) { current_group.declarators.push(declarator);
if (current_group && current_group[current_group.length - 1].kind !== node.kind) { }
} else {
current_group = null; current_group = null;
}
// rewrite as a group, later if (variable.subscribable) {
if (!current_group) { let insert = get_insert(name);
current_group = [];
coalesced_declarations.push(current_group);
}
current_group.push(node); if (next) {
} else { code.overwrite(declarator.end, next.start, `; ${insert}; ${node.kind} `);
if (has_exports) { } else {
// rewrite in place code.appendLeft(declarator.end, `; ${insert}`);
throw new Error('TODO rewrite prop declaration in place'); }
}
} }
});
current_group = null;
}
} }
} else { } else {
if (node.type !== 'ExportNamedDeclaration') { if (node.type !== 'ExportNamedDeclaration') {
@ -858,31 +882,25 @@ export default class Component {
let combining = false; let combining = false;
group.forEach(node => { group.declarators.forEach(declarator => {
node.declarations.forEach(declarator => { const { id } = declarator;
const { id, init } = declarator;
if (id.type === 'Identifier') {
const value = init
? this.code.slice(id.start, init.end)
: this.code.slice(id.start, id.end);
if (combining) { if (combining) {
code.overwrite(c, id.start, ', '); code.overwrite(c, id.start, ', ');
} else { } else {
code.appendLeft(id.start, '{ '); code.appendLeft(id.start, '{ ');
combining = true; combining = true;
} }
} else {
throw new Error('TODO destructured declarations');
}
c = declarator.end; c = declarator.end;
});
}); });
if (combining) { if (combining) {
const suffix = code.original[c] === ';' ? ` } = $$props` : ` } = $$props;`; const insert = group.insert
? `; ${group.insert}`
: '';
const suffix = code.original[c] === ';' ? ` } = $$props${insert}` : ` } = $$props${insert};`;
code.appendLeft(c, suffix); code.appendLeft(c, suffix);
} }
}); });

@ -202,38 +202,6 @@ export default function dom(
component.has_reactive_assignments = true; component.has_reactive_assignments = true;
} }
else if (node.type === 'VariableDeclarator') {
const names = extractNames(node.id);
names.forEach(name => {
const owner = scope.findOwner(name);
if (owner && owner !== component.instance_scope) return;
const variable = component.var_lookup.get(name);
if (variable && variable.subscribable) {
const value = `$${name}`;
const subscribe = component.helper('subscribe');
const index = parent.declarations.indexOf(node);
const next = parent.declarations[index + 1];
let insert = `${subscribe}($$self, ${name}, $$value => { ${value} = $$value; $$invalidate('${value}', ${value}) })`;
if (component.compileOptions.dev) {
const validate_store = component.helper('validate_store');
insert = `${validate_store}(${name}, '${name}'); ${insert}`;
}
// initialise store value here
if (next) {
code.overwrite(node.end, next.start, `; ${insert}; ${parent.kind} `);
} else {
// final (or only) declarator
code.appendLeft(node.end, `; ${insert}`);
}
}
});
}
if (pending_assignments.size > 0) { if (pending_assignments.size > 0) {
if (node.type === 'ArrowFunctionExpression') { if (node.type === 'ArrowFunctionExpression') {
const insert = Array.from(pending_assignments).map(name => `$$invalidate('${name}', ${name})`).join(';'); const insert = Array.from(pending_assignments).map(name => `$$invalidate('${name}', ${name})`).join(';');
@ -272,7 +240,19 @@ export default function dom(
throw new Error(`TODO this should not happen!`); throw new Error(`TODO this should not happen!`);
} }
component.rewrite_props(); component.rewrite_props(name => {
const value = `$${name}`;
const subscribe = component.helper('subscribe');
let insert = `${subscribe}($$self, ${name}, $$value => { ${value} = $$value; $$invalidate('${value}', ${value}) })`;
if (component.compileOptions.dev) {
const validate_store = component.helper('validate_store');
insert = `${validate_store}(${name}, '${name}'); ${insert}`;
}
return insert;
});
} }
const args = ['$$self']; const args = ['$$self'];
@ -333,17 +313,34 @@ export default function dom(
addToSet(all_reactive_dependencies, d.dependencies); addToSet(all_reactive_dependencies, d.dependencies);
}); });
const user_code = component.javascript || ( let user_code;
!component.ast.instance && !component.ast.module && (filtered_props.length > 0 || component.componentOptions.props)
? [ if (component.javascript) {
component.componentOptions.props && `let ${component.componentOptions.props} = $$props;`, user_code = component.javascript;
filtered_props.length > 0 && `let { ${filtered_props.map(x => x.name).join(', ')} } = $$props;` } else {
].filter(Boolean).join('\n') if (!component.ast.instance && !component.ast.module && (filtered_props.length > 0 || component.componentOptions.props)) {
: null const statements = [];
);
if (component.componentOptions.props) statements.push(`let ${component.componentOptions.props} = $$props;`);
if (filtered_props.length > 0) statements.push(`let { ${filtered_props.map(x => x.name).join(', ')} } = $$props;`);
reactive_stores.forEach(({ name }) => {
if (component.compileOptions.dev) {
statements.push(`${component.compileOptions.dev && `@validate_store(${name.slice(1)}, '${name.slice(1)}');`}`);
}
statements.push(`@subscribe($$self, ${name.slice(1)}, $$value => { ${name} = $$value; $$invalidate('${name}', ${name}); });`);
});
user_code = statements.join('\n');
}
}
const reactive_store_subscriptions = reactive_stores const reactive_store_subscriptions = reactive_stores
.filter(store => component.var_lookup.get(store.name).hoistable) .filter(store => {
const variable = component.var_lookup.get(store.name.slice(1));
return variable.hoistable;
})
.map(({ name }) => deindent` .map(({ name }) => deindent`
${component.compileOptions.dev && `@validate_store(${name.slice(1)}, '${name.slice(1)}');`} ${component.compileOptions.dev && `@validate_store(${name.slice(1)}, '${name.slice(1)}');`}
@subscribe($$self, ${name.slice(1)}, $$value => { ${name} = $$value; $$invalidate('${name}', ${name}); }); @subscribe($$self, ${name.slice(1)}, $$value => { ${name} = $$value; $$invalidate('${name}', ${name}); });

@ -24,57 +24,16 @@ export default function ssr(
{ code: null, map: null } : { code: null, map: null } :
component.stylesheet.render(options.filename, true); component.stylesheet.render(options.filename, true);
// insert store values const reactive_stores = component.vars.filter(variable => variable.name[0] === '$');
if (component.ast.instance) { const reactive_store_values = reactive_stores
let scope = component.instance_scope; .filter(store => component.var_lookup.get(store.name).hoistable)
let map = component.instance_scope_map; .map(({ name }) => {
const assignment = `const ${name} = @get_store_value(${name.slice(1)});`;
walk(component.ast.instance.content, {
enter: (node, parent) => { return component.compileOptions.dev
if (map.has(node)) { ? `@validate_store(${name.slice(1)}, '${name.slice(1)}'); ${assignment}`
scope = map.get(node); : assignment;
}
},
leave(node, parent) {
if (map.has(node)) {
scope = scope.parent;
}
if (node.type === 'VariableDeclarator') {
const names = extractNames(node.id);
names.forEach(name => {
const owner = scope.findOwner(name);
if (owner && owner !== component.instance_scope) return;
const variable = component.var_lookup.get(name);
if (variable && variable.subscribable) {
const value = `$${name}`;
const get_store_value = component.helper('get_store_value');
const index = parent.declarations.indexOf(node);
const next = parent.declarations[index + 1];
let insert = `const ${value} = ${get_store_value}(${name});`;
if (component.compileOptions.dev) {
const validate_store = component.helper('validate_store');
insert = `${validate_store}(${name}, '${name}'); ${insert}`;
}
// initialise store value here
if (next) {
component.code.overwrite(node.end, next.start, `; ${insert}; ${parent.kind} `);
} else {
// final (or only) declarator
component.code.appendLeft(node.end, `; ${insert}`);
}
}
});
}
}
}); });
}
// TODO remove this, just use component.vars everywhere // TODO remove this, just use component.vars everywhere
const props = component.vars.filter(variable => !variable.module && variable.export_name && variable.export_name !== component.componentOptions.props_object); const props = component.vars.filter(variable => !variable.module && variable.export_name && variable.export_name !== component.componentOptions.props_object);
@ -82,26 +41,38 @@ export default function ssr(
let user_code; let user_code;
if (component.javascript) { if (component.javascript) {
component.rewrite_props(); component.rewrite_props(name => {
const value = `$${name}`;
const get_store_value = component.helper('get_store_value');
let insert = `const ${value} = ${get_store_value}(${name})`;
if (component.compileOptions.dev) {
const validate_store = component.helper('validate_store');
insert = `${validate_store}(${name}, '${name}'); ${insert}`;
}
return insert;
});
user_code = component.javascript; user_code = component.javascript;
} else if (!component.ast.instance && !component.ast.module && (props.length > 0 || component.componentOptions.props)) { } else if (!component.ast.instance && !component.ast.module && (props.length > 0 || component.componentOptions.props)) {
user_code = [ const statements = [];
component.componentOptions.props && `let ${component.componentOptions.props} = $$props;`,
props.length > 0 && `let { ${props.map(prop => prop.export_name).join(', ')} } = $$props;`
].filter(Boolean).join('\n');
}
const reactive_stores = component.vars.filter(variable => variable.name[0] === '$'); if (component.componentOptions.props) statements.push(`let ${component.componentOptions.props} = $$props;`);
const reactive_store_values = reactive_stores if (props.length > 0) statements.push(`let { ${props.map(x => x.name).join(', ')} } = $$props;`);
.filter(store => component.var_lookup.get(store.name).hoistable)
.map(({ name }) => {
const assignment = `const ${name} = @get_store_value(${name.slice(1)});`;
return component.compileOptions.dev reactive_stores.forEach(({ name }) => {
? `@validate_store(${name.slice(1)}, '${name.slice(1)}'); ${assignment}` if (component.compileOptions.dev) {
: assignment; statements.push(`${component.compileOptions.dev && `@validate_store(${name.slice(1)}, '${name.slice(1)}');`}`);
}
statements.push(`const ${name} = @get_store_value(${name.slice(1)});`);
}); });
user_code = statements.join('\n');
}
// TODO only do this for props with a default value // TODO only do this for props with a default value
const parent_bindings = component.javascript const parent_bindings = component.javascript
? props.map(prop => { ? props.map(prop => {

Loading…
Cancel
Save