Merge pull request #852 from sveltejs/gh-849

compile time error on slotted content inside if/each blocks. closes #849
pull/855/head
Rich Harris 8 years ago committed by GitHub
commit 6ad8c38663

@ -11,6 +11,7 @@ const meta = new Map([[':Window', validateWindow]]);
export default function validateHtml(validator: Validator, html: Node) { export default function validateHtml(validator: Validator, html: Node) {
const refs = new Map(); const refs = new Map();
const refCallees: Node[] = []; const refCallees: Node[] = [];
const stack: Node[] = [];
const elementStack: Node[] = []; const elementStack: Node[] = [];
function visit(node: Node) { function visit(node: Node) {
@ -21,7 +22,7 @@ export default function validateHtml(validator: Validator, html: Node) {
return meta.get(node.name)(validator, node, refs, refCallees); return meta.get(node.name)(validator, node, refs, refCallees);
} }
validateElement(validator, node, refs, refCallees, elementStack); validateElement(validator, node, refs, refCallees, stack, elementStack);
} else if (node.type === 'EachBlock') { } else if (node.type === 'EachBlock') {
if (validator.helpers.has(node.context)) { if (validator.helpers.has(node.context)) {
let c = node.expression.end; let c = node.expression.end;
@ -40,7 +41,9 @@ export default function validateHtml(validator: Validator, html: Node) {
if (node.children) { if (node.children) {
if (node.type === 'Element') elementStack.push(node); if (node.type === 'Element') elementStack.push(node);
stack.push(node);
node.children.forEach(visit); node.children.forEach(visit);
stack.pop();
if (node.type === 'Element') elementStack.pop(); if (node.type === 'Element') elementStack.pop();
} }

@ -10,6 +10,7 @@ export default function validateElement(
node: Node, node: Node,
refs: Map<string, Node[]>, refs: Map<string, Node[]>,
refCallees: Node[], refCallees: Node[],
stack: Node[],
elementStack: Node[] elementStack: Node[]
) { ) {
const isComponent = const isComponent =
@ -189,11 +190,23 @@ export default function validateElement(
} }
} }
if (attribute.name === 'slot' && !isComponent && isDynamic(attribute)) { if (attribute.name === 'slot' && !isComponent) {
validator.error( let i = stack.length;
`slot attribute cannot have a dynamic value`, while (i--) {
attribute.start const parent = stack[i];
); if (parent.type === 'Element' && validator.components.has(parent.name)) break;
if (parent.type === 'IfBlock' || parent.type === 'EachBlock') {
const message = `Cannot place slotted elements inside an ${parent.type === 'IfBlock' ? 'if' : 'each'}-block`;
validator.error(message, attribute.start);
}
}
if (isDynamic(attribute)) {
validator.error(
`slot attribute cannot have a dynamic value`,
attribute.start
);
}
} }
} }
}); });

@ -0,0 +1,8 @@
[{
"message": "Cannot place slotted elements inside an each-block",
"loc": {
"line": 3,
"column": 7
},
"pos": 43
}]

@ -0,0 +1,15 @@
<Nested>
{{#each things as thing}}
<div slot='foo'>{{thing}}</div>
{{/each}}
</Nested>
<script>
import Nested from './Nested.html';
export default {
components: {
Nested
}
};
</script>

@ -0,0 +1,8 @@
[{
"message": "Cannot place slotted elements inside an if-block",
"loc": {
"line": 3,
"column": 7
},
"pos": 31
}]

@ -0,0 +1,15 @@
<Nested>
{{#if thing}}
<div slot='foo'>{{thing}}</div>
{{/if}}
</Nested>
<script>
import Nested from './Nested.html';
export default {
components: {
Nested
}
};
</script>
Loading…
Cancel
Save