Skip to content

Commit

Permalink
feat: add 'each .. of ..' syntax to support all iterable objects (inc…
Browse files Browse the repository at this point in the history
…luding Arrays) (#3179)
  • Loading branch information
maxrumsey authored and ForbesLindesay committed Sep 16, 2019
1 parent fa8d287 commit 8aed002
Show file tree
Hide file tree
Showing 5 changed files with 65 additions and 0 deletions.
10 changes: 10 additions & 0 deletions packages/pug-code-gen/index.js
Expand Up @@ -767,6 +767,16 @@ Compiler.prototype = {
this.buf.push(' }\n}).call(this);\n');
},

visitEachOf: function(each){
this.buf.push(''
+ '// iterate ' + each.obj + '\n'
+ 'for (const ' + each.val + ' of ' + each.obj + ') {\n')

this.visit(each.block, each);

this.buf.push('}\n');
},

/**
* Visit `attrs`.
*
Expand Down
34 changes: 34 additions & 0 deletions packages/pug-lexer/index.js
Expand Up @@ -971,6 +971,39 @@ Lexer.prototype = {
}
},

/**
* EachOf.
*/

eachOf: function() {
var captures;
if (captures = /^(?:each|for) (.*) of *([^\n]+)/.exec(this.input)) {
this.consume(captures[0].length);
var tok = this.tok('eachOf', captures[1]);
tok.value = captures[1];
this.incrementColumn(captures[0].length - captures[2].length);
this.assertExpression(captures[2])
tok.code = captures[2];
this.incrementColumn(captures[2].length);
this.tokens.push(this.tokEnd(tok));

if (!(/^[a-zA-Z_$][\w$]*$/.test(tok.value.trim()) || /^\[ *[a-zA-Z_$][\w$]* *\, *[a-zA-Z_$][\w$]* *\]$/.test(tok.value.trim()))) {
this.error(
'MALFORMED_EACH_OF_LVAL',
'The value variable for each must either be a valid identifier (e.g. `item`) or a pair of identifiers in square brackets (e.g. `[key, value]`).'
);
}

return true;
}
if (captures = /^- *(?:each|for) +([a-zA-Z_$][\w$]*)(?: *, *([a-zA-Z_$][\w$]*))? +of +([^\n]+)/.exec(this.input)) {
this.error(
'MALFORMED_EACH',
'Pug each and for should not be prefixed with a dash ("-"). They are pug keywords and not part of JavaScript.'
);
}
},

/**
* Code.
*/
Expand Down Expand Up @@ -1485,6 +1518,7 @@ Lexer.prototype = {
|| this.callLexerFunction('mixin')
|| this.callLexerFunction('call')
|| this.callLexerFunction('conditional')
|| this.callLexerFunction('eachOf')
|| this.callLexerFunction('each')
|| this.callLexerFunction('while')
|| this.callLexerFunction('tag')
Expand Down
1 change: 1 addition & 0 deletions packages/pug-lexer/test/check-lexer-functions.test.js
Expand Up @@ -23,6 +23,7 @@ var lexerFunctions = {
doctype: true,
dot: true,
each: true,
eachOf: true,
eos: true,
endInterpolation: true,
extends: true,
Expand Down
15 changes: 15 additions & 0 deletions packages/pug-parser/index.js
Expand Up @@ -234,6 +234,8 @@ Parser.prototype = {
return this.parseDot();
case 'each':
return this.parseEach();
case 'eachOf':
return this.parseEachOf();
case 'code':
return this.parseCode();
case 'blockcode':
Expand Down Expand Up @@ -761,6 +763,19 @@ loop:
return node;
},

parseEachOf: function(){
var tok = this.expect('eachOf');
var node = {
type: 'EachOf',
obj: tok.code,
val: tok.val,
block: this.block(),
line: tok.loc.start.line,
column: tok.loc.start.column,
filename: this.filename
};
return node;
},
/**
* 'extends' name
*/
Expand Down
5 changes: 5 additions & 0 deletions packages/pug-walk/index.js
Expand Up @@ -56,6 +56,11 @@ function walkAST(ast, before, after, options) {
ast.alternate = walkAST(ast.alternate, before, after, options);
}
break;
case 'EachOf':
if (ast.block) {
ast.block = walkAST(ast.block, before, after, options);
}
break;
case 'Conditional':
if (ast.consequent) {
ast.consequent = walkAST(ast.consequent, before, after, options);
Expand Down

0 comments on commit 8aed002

Please sign in to comment.