Support invalid JS attributes when passing data to components and on binding attributes, too

pull/1555/head
Fernando Jorge Mota 7 years ago
parent c641ce423a
commit 9a0af96c44

@ -5,7 +5,7 @@ import stringifyProps from '../../utils/stringifyProps';
import CodeBuilder from '../../utils/CodeBuilder'; import CodeBuilder from '../../utils/CodeBuilder';
import getTailSnippet from '../../utils/getTailSnippet'; import getTailSnippet from '../../utils/getTailSnippet';
import getObject from '../../utils/getObject'; import getObject from '../../utils/getObject';
import { quoteNameIfNecessary } from '../../utils/quoteIfNecessary'; import { quoteNameIfNecessary, quotePropIfNecessary } from '../../utils/quoteIfNecessary';
import { escape, escapeTemplate, stringify } from '../../utils/stringify'; import { escape, escapeTemplate, stringify } from '../../utils/stringify';
import Node from './shared/Node'; import Node from './shared/Node';
import Block from '../dom/Block'; import Block from '../dom/Block';
@ -219,7 +219,7 @@ export default class Component extends Node {
updates.push(deindent` updates.push(deindent`
if (${[...attribute.dependencies] if (${[...attribute.dependencies]
.map(dependency => `changed.${dependency}`) .map(dependency => `changed.${dependency}`)
.join(' || ')}) ${name_changes}.${attribute.name} = ${attribute.getValue()}; .join(' || ')}) ${name_changes}${quotePropIfNecessary(attribute.name)} = ${attribute.getValue()};
`); `);
} }
}); });
@ -250,10 +250,10 @@ export default class Component extends Node {
const lhs = binding.value.node.type === 'MemberExpression' const lhs = binding.value.node.type === 'MemberExpression'
? binding.value.snippet ? binding.value.snippet
: `${head}${tail} = childState.${binding.name}`; : `${head}${tail} = childState${quotePropIfNecessary(binding.name)}`;
setFromChild = deindent` setFromChild = deindent`
${lhs} = childState.${binding.name}; ${lhs} = childState${quotePropIfNecessary(binding.name)};
${[...binding.value.dependencies] ${[...binding.value.dependencies]
.map((name: string) => { .map((name: string) => {
@ -264,7 +264,7 @@ export default class Component extends Node {
if (isStoreProp) hasStoreBindings = true; if (isStoreProp) hasStoreBindings = true;
else hasLocalBindings = true; else hasLocalBindings = true;
return `${newState}.${prop} = ctx.${name};`; return `${newState}${quotePropIfNecessary(prop)} = ctx${quotePropIfNecessary(name)};`;
})} })}
`; `;
} }
@ -279,32 +279,32 @@ export default class Component extends Node {
if (binding.value.node.type === 'MemberExpression') { if (binding.value.node.type === 'MemberExpression') {
setFromChild = deindent` setFromChild = deindent`
${binding.value.snippet} = childState.${binding.name}; ${binding.value.snippet} = childState${quotePropIfNecessary(binding.name)};
${newState}.${prop} = ctx.${key}; ${newState}${quotePropIfNecessary(prop)} = ctx${quotePropIfNecessary(key)};
`; `;
} }
else { else {
setFromChild = `${newState}.${prop} = childState.${binding.name};`; setFromChild = `${newState}${quotePropIfNecessary(prop)} = childState${quotePropIfNecessary(binding.name)};`;
} }
} }
statements.push(deindent` statements.push(deindent`
if (${binding.value.snippet} !== void 0) { if (${binding.value.snippet} !== void 0) {
${name_initial_data}.${binding.name} = ${binding.value.snippet}; ${name_initial_data}${quotePropIfNecessary(binding.name)} = ${binding.value.snippet};
${name_updating}.${binding.name} = true; ${name_updating}${quotePropIfNecessary(binding.name)} = true;
}` }`
); );
builder.addConditional( builder.addConditional(
`!${name_updating}.${binding.name} && changed.${binding.name}`, `!${name_updating}${quotePropIfNecessary(binding.name)} && changed${quotePropIfNecessary(binding.name)}`,
setFromChild setFromChild
); );
updates.push(deindent` updates.push(deindent`
if (!${name_updating}.${binding.name} && ${[...binding.value.dependencies].map((dependency: string) => `changed.${dependency}`).join(' || ')}) { if (!${name_updating}${quotePropIfNecessary(binding.name)} && ${[...binding.value.dependencies].map((dependency: string) => `changed.${dependency}`).join(' || ')}) {
${name_changes}.${binding.name} = ${binding.value.snippet}; ${name_changes}${quotePropIfNecessary(binding.name)} = ${binding.value.snippet};
${name_updating}.${binding.name} = ${binding.value.snippet} !== void 0; ${name_updating}${quotePropIfNecessary(binding.name)} = ${binding.value.snippet} !== void 0;
} }
`); `);
}); });
@ -329,7 +329,7 @@ export default class Component extends Node {
beforecreate = deindent` beforecreate = deindent`
#component.root._beforecreate.push(() => { #component.root._beforecreate.push(() => {
${name}._bind({ ${this.bindings.map(b => `${b.name}: 1`).join(', ')} }, ${name}.get()); ${name}._bind({ ${this.bindings.map(b => `${quoteNameIfNecessary(b.name)}: 1`).join(', ')} }, ${name}.get());
}); });
`; `;
} }
@ -519,7 +519,7 @@ export default class Component extends Node {
? getTailSnippet(binding.value.node) ? getTailSnippet(binding.value.node)
: ''; : '';
return `${binding.name}: ctx.${name}${tail}`; return `${quoteNameIfNecessary(binding.name)}: ctx${quotePropIfNecessary(name)}${tail}`;
}); });
function getAttributeValue(attribute) { function getAttributeValue(attribute) {
@ -547,14 +547,14 @@ export default class Component extends Node {
if (attribute.isSpread) { if (attribute.isSpread) {
return attribute.expression.snippet; return attribute.expression.snippet;
} else { } else {
return `{ ${attribute.name}: ${getAttributeValue(attribute)} }`; return `{ ${quoteNameIfNecessary(attribute.name)}: ${getAttributeValue(attribute)} }`;
} }
}) })
.concat(bindingProps.map(p => `{ ${p} }`)) .concat(bindingProps.map(p => `{ ${p} }`))
.join(', ') .join(', ')
})` })`
: `{ ${this.attributes : `{ ${this.attributes
.map(attribute => `${attribute.name}: ${getAttributeValue(attribute)}`) .map(attribute => `${quoteNameIfNecessary(attribute.name)}: ${getAttributeValue(attribute)}`)
.concat(bindingProps) .concat(bindingProps)
.join(', ')} }`; .join(', ')} }`;
@ -585,7 +585,7 @@ export default class Component extends Node {
if (${conditions.reverse().join('&&')}) { if (${conditions.reverse().join('&&')}) {
tmp = ${expression}.data(); tmp = ${expression}.data();
if ('${name}' in tmp) { if ('${name}' in tmp) {
ctx.${binding.name} = tmp.${name}; ctx${quotePropIfNecessary(binding.name)} = tmp.${name};
settled = false; settled = false;
} }
} }

@ -0,0 +1,14 @@
<button on:click='set({ "x-count": state["x-count"] + 1 })'>+1</button>
<script>
export default {
data: () => ({
"x-count": 0
}),
computed: {
state(arg) {
return arg;
}
}
};
</script>

@ -0,0 +1,29 @@
export default {
'skip-ssr': true, // TODO delete this line, once binding works
html: `
<button>+1</button>
<p>count: 0</p>
`,
test ( assert, component, target, window ) {
const click = new window.MouseEvent( 'click' );
const button = target.querySelector( 'button' );
button.dispatchEvent( click );
assert.equal( component.get().x, 1 );
assert.htmlEqual( target.innerHTML, `
<button>+1</button>
<p>count: 1</p>
` );
button.dispatchEvent( click );
assert.equal( component.get().x, 2 );
assert.htmlEqual( target.innerHTML, `
<button>+1</button>
<p>count: 2</p>
` );
}
};

@ -0,0 +1,12 @@
<Counter bind:x-count='x'/>
<p>count: {x}</p>
<script>
import Counter from './Counter.html';
export default {
components: {
Counter
}
};
</script>

@ -0,0 +1,10 @@
<p>{state["b-c"]}</p>
<script>
export default {
computed: {
state(states) {
return states;
}
}
};
</script>

@ -0,0 +1,3 @@
export default {
html: '<div><p>i am a widget</p></div>'
};

@ -0,0 +1,11 @@
<div>
<Widget b-c="i am a widget"/>
</div>
<script>
import Widget from './Widget.html';
export default {
components: { Widget }
};
</script>
Loading…
Cancel
Save