You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
svelte/compiler/parse/state/mustache.js

120 lines
2.8 KiB

8 years ago
import readExpression from '../read/expression.js';
import { whitespace } from '../patterns.js';
import { trimStart, trimEnd } from '../utils/trim.js';
8 years ago
const validIdentifier = /[a-zA-Z_$][a-zA-Z0-9_$]*/;
8 years ago
export default function mustache ( parser ) {
const start = parser.index;
parser.index += 2;
parser.allowWhitespace();
// {{/if}} or {{/each}}
if ( parser.eat( '/' ) ) {
const block = parser.current();
8 years ago
let expected;
if ( block.type === 'IfBlock' ) {
8 years ago
expected = 'if';
} else if ( block.type === 'EachBlock' ) {
8 years ago
expected = 'each';
} else {
parser.error( `Unexpected block closing tag` );
}
parser.eat( expected, true );
parser.allowWhitespace();
parser.eat( '}}', true );
// strip leading/trailing whitespace as necessary
if ( !block.children.length ) parser.error( `Empty block`, block.start );
const firstChild = block.children[0];
const lastChild = block.children[ block.children.length - 1 ];
const charBefore = parser.template[ block.start - 1 ];
const charAfter = parser.template[ parser.index ];
if ( firstChild.type === 'Text' && !charBefore || whitespace.test( charBefore ) ) {
firstChild.data = trimStart( firstChild.data );
if ( !firstChild.data ) block.children.shift();
}
if ( lastChild.type === 'Text' && !charAfter || whitespace.test( charAfter ) ) {
lastChild.data = trimEnd( lastChild.data );
if ( !lastChild.data ) block.children.pop();
}
block.end = parser.index;
8 years ago
parser.stack.pop();
}
// TODO {{else}} and {{elseif expression}}
// {{#if foo}} or {{#each foo}}
else if ( parser.eat( '#' ) ) {
let type;
if ( parser.eat( 'if' ) ) {
type = 'IfBlock';
} else if ( parser.eat( 'each' ) ) {
type = 'EachBlock';
} else {
parser.error( `Expected if or each` );
}
parser.requireWhitespace();
const expression = readExpression( parser );
const block = {
start,
end: null,
type,
expression,
children: []
};
parser.allowWhitespace();
// {{#each}} blocks must declare a context {{#each list as item}}
if ( type === 'EachBlock' ) {
parser.eat( 'as', true );
parser.requireWhitespace();
block.context = parser.read( validIdentifier ); // TODO check it's not a keyword
if ( !block.context ) parser.error( `Expected name` );
parser.allowWhitespace();
if ( parser.eat( ',' ) ) {
parser.allowWhitespace();
block.index = parser.read( validIdentifier );
if ( !block.index ) parser.error( `Expected name` );
parser.allowWhitespace();
}
}
parser.eat( '}}', true );
8 years ago
parser.current().children.push( block );
parser.stack.push( block );
}
else {
const expression = readExpression( parser );
parser.allowWhitespace();
parser.eat( '}}', true );
parser.current().children.push({
start,
end: parser.index,
type: 'MustacheTag',
expression
});
}
return null;
}