From c0a8e630e36bfa5eb8b3b726f0bfa3d133a285c2 Mon Sep 17 00:00:00 2001 From: Rich Harris Date: Mon, 31 Dec 2018 09:43:20 -0500 Subject: [PATCH] Fix media bindings, simplify others --- src/compile/nodes/Binding.ts | 5 - src/compile/render-dom/Block.ts | 2 +- src/compile/render-dom/wrappers/EachBlock.ts | 57 +++--- .../render-dom/wrappers/Element/Binding.ts | 182 +++++++++--------- .../render-dom/wrappers/Element/index.ts | 107 +++------- .../wrappers/InlineComponent/index.ts | 4 +- test/js/samples/input-files/expected.js | 11 +- test/js/samples/media-bindings/expected.js | 22 ++- 8 files changed, 169 insertions(+), 221 deletions(-) diff --git a/src/compile/nodes/Binding.ts b/src/compile/nodes/Binding.ts index 081f736778..3d26d5f34c 100644 --- a/src/compile/nodes/Binding.ts +++ b/src/compile/nodes/Binding.ts @@ -7,7 +7,6 @@ export default class Binding extends Node { name: string; expression: Expression; isContextual: boolean; - usesContext: boolean; obj: string; prop: string; @@ -34,13 +33,9 @@ export default class Binding extends Node { prop = `[✂${this.expression.node.property.start}-${this.expression.node.property.end}✂]`; if (!this.expression.node.computed) prop = `'${prop}'`; obj = `[✂${this.expression.node.object.start}-${this.expression.node.object.end}✂]`; - - this.usesContext = true; } else { obj = 'ctx'; prop = `'${name}'`; - - this.usesContext = scope.names.has(name); } this.obj = obj; diff --git a/src/compile/render-dom/Block.ts b/src/compile/render-dom/Block.ts index 573a9863d8..5a192b55a3 100644 --- a/src/compile/render-dom/Block.ts +++ b/src/compile/render-dom/Block.ts @@ -29,7 +29,7 @@ export default class Block { dependencies: Set; - bindings: Map { object: string, property: string, snippet: string }>; + bindings: Map; contextOwners: Map; builders: { diff --git a/src/compile/render-dom/wrappers/EachBlock.ts b/src/compile/render-dom/wrappers/EachBlock.ts index 439f99fd09..603920070d 100644 --- a/src/compile/render-dom/wrappers/EachBlock.ts +++ b/src/compile/render-dom/wrappers/EachBlock.ts @@ -97,15 +97,33 @@ export default class EachBlockWrapper extends Wrapper { this.indexName = this.node.index || renderer.component.getUniqueName(`${this.node.context}_index`); + // hack the sourcemap, so that if data is missing the bug + // is easy to find + let c = this.node.start + 2; + while (renderer.component.source[c] !== 'e') c += 1; + renderer.component.code.overwrite(c, c + 4, 'length'); + const length = `[✂${c}-${c+4}✂]`; + + this.vars = { + create_each_block: this.block.name, + each_block_value: renderer.component.getUniqueName(`${this.var}_value`), + get_each_context: renderer.component.getUniqueName(`get_${this.var}_context`), + iterations: block.getUniqueName(`${this.var}_blocks`), + length: `[✂${c}-${c+4}✂]`, + + // filled out later + anchor: null, + mountOrIntro: null + }; + node.contexts.forEach(prop => { this.block.contextOwners.set(prop.key.name, this); - // TODO this doesn't feel great - this.block.bindings.set(prop.key.name, () => ({ + this.block.bindings.set(prop.key.name, { object: this.vars.each_block_value, property: this.indexName, snippet: `${this.vars.each_block_value}[${this.indexName}]${prop.tail}` - })); + }); }); if (this.node.index) { @@ -147,30 +165,15 @@ export default class EachBlockWrapper extends Wrapper { const { renderer } = this; const { component } = renderer; - // hack the sourcemap, so that if data is missing the bug - // is easy to find - let c = this.node.start + 2; - while (component.source[c] !== 'e') c += 1; - component.code.overwrite(c, c + 4, 'length'); - const length = `[✂${c}-${c+4}✂]`; - const needsAnchor = this.next ? !this.next.isDomNode() : !parentNode || !this.parent.isDomNode(); - this.vars = { - anchor: needsAnchor - ? block.getUniqueName(`${this.var}_anchor`) - : (this.next && this.next.var) || 'null', - create_each_block: this.block.name, - each_block_value: renderer.component.getUniqueName(`${this.var}_value`), - get_each_context: renderer.component.getUniqueName(`get_${this.var}_context`), - iterations: block.getUniqueName(`${this.var}_blocks`), - length: `[✂${c}-${c+4}✂]`, - mountOrIntro: (this.block.hasIntroMethod || this.block.hasOutroMethod) - ? 'i' - : 'm' - }; + this.vars.anchor = needsAnchor + ? block.getUniqueName(`${this.var}_anchor`) + : (this.next && this.next.var) || 'null'; + + this.vars.mountOrIntro = (this.block.hasIntroMethod || this.block.hasOutroMethod) ? 'i' : 'm'; this.contextProps = this.node.contexts.map(prop => `child_ctx.${prop.key.name} = list[i]${prop.tail};`); @@ -212,7 +215,7 @@ export default class EachBlockWrapper extends Wrapper { // TODO neaten this up... will end up with an empty line in the block block.builders.init.addBlock(deindent` - if (!${this.vars.each_block_value}.${length}) { + if (!${this.vars.each_block_value}.${this.vars.length}) { ${each_block_else} = ${this.else.block.name}(#component, ctx); ${each_block_else}.c(); } @@ -228,9 +231,9 @@ export default class EachBlockWrapper extends Wrapper { if (this.else.block.hasUpdateMethod) { block.builders.update.addBlock(deindent` - if (!${this.vars.each_block_value}.${length} && ${each_block_else}) { + if (!${this.vars.each_block_value}.${this.vars.length} && ${each_block_else}) { ${each_block_else}.p(changed, ctx); - } else if (!${this.vars.each_block_value}.${length}) { + } else if (!${this.vars.each_block_value}.${this.vars.length}) { ${each_block_else} = ${this.else.block.name}(#component, ctx); ${each_block_else}.c(); ${each_block_else}.${mountOrIntro}(${initialMountNode}, ${this.vars.anchor}); @@ -241,7 +244,7 @@ export default class EachBlockWrapper extends Wrapper { `); } else { block.builders.update.addBlock(deindent` - if (${this.vars.each_block_value}.${length}) { + if (${this.vars.each_block_value}.${this.vars.length}) { if (${each_block_else}) { ${each_block_else}.d(1); ${each_block_else} = null; diff --git a/src/compile/render-dom/wrappers/Element/Binding.ts b/src/compile/render-dom/wrappers/Element/Binding.ts index 6b2f37cfb1..672fb21ab7 100644 --- a/src/compile/render-dom/wrappers/Element/Binding.ts +++ b/src/compile/render-dom/wrappers/Element/Binding.ts @@ -21,11 +21,15 @@ export default class BindingWrapper { parent: ElementWrapper; object: string; - handler: any; // TODO - updateDom: string; + handler: { + usesContext: boolean; + mutation: string; + contextual_dependencies: Set + }; + snippet: string; initialUpdate: string; + isReadOnly: boolean; needsLock: boolean; - updateCondition: string; constructor(block: Block, node: Binding, parent: ElementWrapper) { this.node = node; @@ -51,45 +55,34 @@ export default class BindingWrapper { eachBlock.hasBinding = true; } - } - isReadOnlyMediaAttribute() { - return readOnlyMediaAttributes.has(this.node.name); - } - - munge(block: Block) { - const { parent } = this; - const { renderer } = parent; + this.object = getObject(this.node.expression.node).name; - const needsLock = ( - parent.node.name !== 'input' || - !/radio|checkbox|range|color/.test(parent.node.getStaticAttributeValue('type')) - ); + // TODO unfortunate code is necessary because we need to use `ctx` + // inside the fragment, but not inside the