@angular/compiler 17.0.0-next.4 → 17.0.0-next.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/src/compiler.mjs +1 -1
- package/esm2022/src/i18n/extractor_merger.mjs +1 -4
- package/esm2022/src/i18n/i18n_parser.mjs +1 -6
- package/esm2022/src/i18n/serializers/xliff.mjs +1 -3
- package/esm2022/src/i18n/serializers/xliff2.mjs +1 -3
- package/esm2022/src/i18n/serializers/xtb.mjs +1 -3
- package/esm2022/src/jit_compiler_facade.mjs +33 -7
- package/esm2022/src/ml_parser/ast.mjs +1 -17
- package/esm2022/src/ml_parser/html_whitespaces.mjs +2 -5
- package/esm2022/src/ml_parser/icu_ast_expander.mjs +2 -5
- package/esm2022/src/ml_parser/lexer.mjs +59 -49
- package/esm2022/src/ml_parser/parser.mjs +26 -71
- package/esm2022/src/ml_parser/tokens.mjs +1 -1
- package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
- package/esm2022/src/render3/partial/component.mjs +5 -2
- package/esm2022/src/render3/partial/directive.mjs +1 -1
- package/esm2022/src/render3/partial/factory.mjs +1 -1
- package/esm2022/src/render3/partial/injectable.mjs +1 -1
- package/esm2022/src/render3/partial/injector.mjs +1 -1
- package/esm2022/src/render3/partial/ng_module.mjs +1 -1
- package/esm2022/src/render3/partial/pipe.mjs +1 -1
- package/esm2022/src/render3/r3_ast.mjs +10 -3
- package/esm2022/src/render3/r3_class_metadata_compiler.mjs +14 -16
- package/esm2022/src/render3/r3_control_flow.mjs +116 -96
- package/esm2022/src/render3/r3_deferred_blocks.mjs +37 -39
- package/esm2022/src/render3/r3_deferred_triggers.mjs +30 -22
- package/esm2022/src/render3/r3_module_compiler.mjs +9 -11
- package/esm2022/src/render3/r3_template_transform.mjs +73 -31
- package/esm2022/src/render3/view/api.mjs +1 -1
- package/esm2022/src/render3/view/compiler.mjs +17 -6
- package/esm2022/src/render3/view/i18n/meta.mjs +1 -5
- package/esm2022/src/render3/view/t2_api.mjs +1 -1
- package/esm2022/src/render3/view/t2_binder.mjs +184 -79
- package/esm2022/src/render3/view/template.mjs +93 -83
- package/esm2022/src/template/pipeline/ir/src/enums.mjs +14 -7
- package/esm2022/src/template/pipeline/ir/src/expression.mjs +6 -4
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +30 -4
- package/esm2022/src/template/pipeline/src/compilation.mjs +6 -1
- package/esm2022/src/template/pipeline/src/conversion.mjs +7 -1
- package/esm2022/src/template/pipeline/src/emit.mjs +7 -1
- package/esm2022/src/template/pipeline/src/ingest.mjs +91 -42
- package/esm2022/src/template/pipeline/src/instruction.mjs +43 -27
- package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +22 -29
- package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +46 -0
- package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +33 -0
- package/esm2022/src/template/pipeline/src/phases/i18n_message_extraction.mjs +3 -2
- package/esm2022/src/template/pipeline/src/phases/no_listeners_on_templates.mjs +1 -5
- package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +39 -0
- package/esm2022/src/template/pipeline/src/phases/reify.mjs +23 -14
- package/esm2022/src/version.mjs +1 -1
- package/fesm2022/compiler.mjs +3425 -3072
- package/fesm2022/compiler.mjs.map +1 -1
- package/fesm2022/testing.mjs +1 -1
- package/index.d.ts +154 -143
- package/package.json +3 -3
- package/testing/index.d.ts +1 -1
|
@@ -15,29 +15,47 @@ const FOR_LOOP_TRACK_PATTERN = /^track\s+(.*)/;
|
|
|
15
15
|
/** Pattern for the `as` expression in a conditional block. */
|
|
16
16
|
const CONDITIONAL_ALIAS_PATTERN = /^as\s+(.*)/;
|
|
17
17
|
/** Pattern used to identify an `else if` block. */
|
|
18
|
-
const ELSE_IF_PATTERN = /^if
|
|
18
|
+
const ELSE_IF_PATTERN = /^else[^\S\r\n]+if/;
|
|
19
19
|
/** Pattern used to identify a `let` parameter. */
|
|
20
20
|
const FOR_LOOP_LET_PATTERN = /^let\s+(.*)/;
|
|
21
21
|
/** Names of variables that are allowed to be used in the `let` expression of a `for` loop. */
|
|
22
22
|
const ALLOWED_FOR_LOOP_LET_VARIABLES = new Set(['$index', '$first', '$last', '$even', '$odd', '$count']);
|
|
23
|
+
/**
|
|
24
|
+
* Predicate function that determines if a block with
|
|
25
|
+
* a specific name cam be connected to a `for` block.
|
|
26
|
+
*/
|
|
27
|
+
export function isConnectedForLoopBlock(name) {
|
|
28
|
+
return name === 'empty';
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Predicate function that determines if a block with
|
|
32
|
+
* a specific name cam be connected to an `if` block.
|
|
33
|
+
*/
|
|
34
|
+
export function isConnectedIfLoopBlock(name) {
|
|
35
|
+
return name === 'else' || ELSE_IF_PATTERN.test(name);
|
|
36
|
+
}
|
|
23
37
|
/** Creates an `if` loop block from an HTML AST node. */
|
|
24
|
-
export function createIfBlock(ast, visitor, bindingParser) {
|
|
25
|
-
const errors =
|
|
38
|
+
export function createIfBlock(ast, connectedBlocks, visitor, bindingParser) {
|
|
39
|
+
const errors = validateIfConnectedBlocks(connectedBlocks);
|
|
26
40
|
const branches = [];
|
|
27
41
|
if (errors.length > 0) {
|
|
28
42
|
return { node: null, errors };
|
|
29
43
|
}
|
|
44
|
+
const mainBlockParams = parseConditionalBlockParameters(ast, errors, bindingParser);
|
|
45
|
+
if (mainBlockParams !== null) {
|
|
46
|
+
branches.push(new t.IfBlockBranch(mainBlockParams.expression, html.visitAll(visitor, ast.children, ast.children), mainBlockParams.expressionAlias, ast.sourceSpan, ast.startSourceSpan));
|
|
47
|
+
}
|
|
30
48
|
// Assumes that the structure is valid since we validated it above.
|
|
31
|
-
for (const block of
|
|
32
|
-
const children = html.visitAll(visitor, block.children);
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
49
|
+
for (const block of connectedBlocks) {
|
|
50
|
+
const children = html.visitAll(visitor, block.children, block.children);
|
|
51
|
+
if (ELSE_IF_PATTERN.test(block.name)) {
|
|
52
|
+
const params = parseConditionalBlockParameters(block, errors, bindingParser);
|
|
53
|
+
if (params !== null) {
|
|
54
|
+
branches.push(new t.IfBlockBranch(params.expression, children, params.expressionAlias, block.sourceSpan, block.startSourceSpan));
|
|
55
|
+
}
|
|
37
56
|
}
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
branches.push(new t.IfBlockBranch(params.expression, children, params.expressionAlias, block.sourceSpan, block.startSourceSpan));
|
|
57
|
+
else if (block.name === 'else') {
|
|
58
|
+
branches.push(new t.IfBlockBranch(null, children, null, block.sourceSpan, block.startSourceSpan));
|
|
41
59
|
}
|
|
42
60
|
}
|
|
43
61
|
return {
|
|
@@ -46,54 +64,55 @@ export function createIfBlock(ast, visitor, bindingParser) {
|
|
|
46
64
|
};
|
|
47
65
|
}
|
|
48
66
|
/** Creates a `for` loop block from an HTML AST node. */
|
|
49
|
-
export function createForLoop(ast, visitor, bindingParser) {
|
|
50
|
-
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
67
|
+
export function createForLoop(ast, connectedBlocks, visitor, bindingParser) {
|
|
51
68
|
const errors = [];
|
|
52
|
-
const params = parseForLoopParameters(
|
|
69
|
+
const params = parseForLoopParameters(ast, errors, bindingParser);
|
|
53
70
|
let node = null;
|
|
54
71
|
let empty = null;
|
|
55
|
-
for (const block of
|
|
72
|
+
for (const block of connectedBlocks) {
|
|
56
73
|
if (block.name === 'empty') {
|
|
57
74
|
if (empty !== null) {
|
|
58
|
-
errors.push(new ParseError(block.sourceSpan, '
|
|
75
|
+
errors.push(new ParseError(block.sourceSpan, '@for loop can only have one @empty block'));
|
|
59
76
|
}
|
|
60
77
|
else if (block.parameters.length > 0) {
|
|
61
|
-
errors.push(new ParseError(block.sourceSpan, '
|
|
78
|
+
errors.push(new ParseError(block.sourceSpan, '@empty block cannot have parameters'));
|
|
62
79
|
}
|
|
63
80
|
else {
|
|
64
|
-
empty = new t.ForLoopBlockEmpty(html.visitAll(visitor, block.children), block.sourceSpan, block.startSourceSpan);
|
|
81
|
+
empty = new t.ForLoopBlockEmpty(html.visitAll(visitor, block.children, block.children), block.sourceSpan, block.startSourceSpan);
|
|
65
82
|
}
|
|
66
83
|
}
|
|
67
84
|
else {
|
|
68
|
-
errors.push(new ParseError(block.sourceSpan, `Unrecognized loop block "${block.name}"`));
|
|
85
|
+
errors.push(new ParseError(block.sourceSpan, `Unrecognized @for loop block "${block.name}"`));
|
|
69
86
|
}
|
|
70
87
|
}
|
|
71
88
|
if (params !== null) {
|
|
72
89
|
if (params.trackBy === null) {
|
|
73
|
-
errors.push(new ParseError(ast.sourceSpan, '
|
|
90
|
+
errors.push(new ParseError(ast.sourceSpan, '@for loop must have a "track" expression'));
|
|
74
91
|
}
|
|
75
92
|
else {
|
|
76
|
-
node = new t.ForLoopBlock(params.itemName, params.expression, params.trackBy, params.context, html.visitAll(visitor,
|
|
93
|
+
node = new t.ForLoopBlock(params.itemName, params.expression, params.trackBy, params.context, html.visitAll(visitor, ast.children, ast.children), empty, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);
|
|
77
94
|
}
|
|
78
95
|
}
|
|
79
96
|
return { node, errors };
|
|
80
97
|
}
|
|
81
98
|
/** Creates a switch block from an HTML AST node. */
|
|
82
99
|
export function createSwitchBlock(ast, visitor, bindingParser) {
|
|
83
|
-
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
84
100
|
const errors = validateSwitchBlock(ast);
|
|
85
101
|
if (errors.length > 0) {
|
|
86
102
|
return { node: null, errors };
|
|
87
103
|
}
|
|
88
|
-
const primaryExpression = parseBlockParameterToBinding(
|
|
104
|
+
const primaryExpression = parseBlockParameterToBinding(ast.parameters[0], bindingParser);
|
|
89
105
|
const cases = [];
|
|
90
106
|
let defaultCase = null;
|
|
91
107
|
// Here we assume that all the blocks are valid given that we validated them above.
|
|
92
|
-
for (const
|
|
93
|
-
|
|
94
|
-
|
|
108
|
+
for (const node of ast.children) {
|
|
109
|
+
if (!(node instanceof html.Block)) {
|
|
110
|
+
continue;
|
|
111
|
+
}
|
|
112
|
+
const expression = node.name === 'case' ?
|
|
113
|
+
parseBlockParameterToBinding(node.parameters[0], bindingParser) :
|
|
95
114
|
null;
|
|
96
|
-
const ast = new t.SwitchBlockCase(expression, html.visitAll(visitor,
|
|
115
|
+
const ast = new t.SwitchBlockCase(expression, html.visitAll(visitor, node.children, node.children), node.sourceSpan, node.startSourceSpan);
|
|
97
116
|
if (expression === null) {
|
|
98
117
|
defaultCase = ast;
|
|
99
118
|
}
|
|
@@ -113,52 +132,58 @@ export function createSwitchBlock(ast, visitor, bindingParser) {
|
|
|
113
132
|
/** Parses the parameters of a `for` loop block. */
|
|
114
133
|
function parseForLoopParameters(block, errors, bindingParser) {
|
|
115
134
|
if (block.parameters.length === 0) {
|
|
116
|
-
errors.push(new ParseError(block.sourceSpan, '
|
|
135
|
+
errors.push(new ParseError(block.sourceSpan, '@for loop does not have an expression'));
|
|
117
136
|
return null;
|
|
118
137
|
}
|
|
119
138
|
const [expressionParam, ...secondaryParams] = block.parameters;
|
|
120
139
|
const match = stripOptionalParentheses(expressionParam, errors)?.match(FOR_LOOP_EXPRESSION_PATTERN);
|
|
121
140
|
if (!match || match[2].trim().length === 0) {
|
|
122
|
-
errors.push(new ParseError(expressionParam.sourceSpan, 'Cannot parse expression.
|
|
141
|
+
errors.push(new ParseError(expressionParam.sourceSpan, 'Cannot parse expression. @for loop expression must match the pattern "<identifier> of <expression>"'));
|
|
123
142
|
return null;
|
|
124
143
|
}
|
|
125
144
|
const [, itemName, rawExpression] = match;
|
|
126
145
|
const result = {
|
|
127
|
-
itemName,
|
|
146
|
+
itemName: new t.Variable(itemName, '$implicit', expressionParam.sourceSpan, expressionParam.sourceSpan),
|
|
128
147
|
trackBy: null,
|
|
129
148
|
expression: parseBlockParameterToBinding(expressionParam, bindingParser, rawExpression),
|
|
130
|
-
context:
|
|
149
|
+
context: {},
|
|
131
150
|
};
|
|
132
151
|
for (const param of secondaryParams) {
|
|
133
152
|
const letMatch = param.expression.match(FOR_LOOP_LET_PATTERN);
|
|
134
153
|
if (letMatch !== null) {
|
|
135
|
-
|
|
136
|
-
parseLetParameter(param.sourceSpan, letMatch[1], result.context, errors);
|
|
154
|
+
parseLetParameter(param.sourceSpan, letMatch[1], param.sourceSpan, result.context, errors);
|
|
137
155
|
continue;
|
|
138
156
|
}
|
|
139
157
|
const trackMatch = param.expression.match(FOR_LOOP_TRACK_PATTERN);
|
|
140
158
|
if (trackMatch !== null) {
|
|
141
159
|
if (result.trackBy !== null) {
|
|
142
|
-
errors.push(new ParseError(param.sourceSpan, '
|
|
160
|
+
errors.push(new ParseError(param.sourceSpan, '@for loop can only have one "track" expression'));
|
|
143
161
|
}
|
|
144
162
|
else {
|
|
145
163
|
result.trackBy = parseBlockParameterToBinding(param, bindingParser, trackMatch[1]);
|
|
146
164
|
}
|
|
147
165
|
continue;
|
|
148
166
|
}
|
|
149
|
-
errors.push(new ParseError(param.sourceSpan, `Unrecognized loop paramater "${param.expression}"`));
|
|
167
|
+
errors.push(new ParseError(param.sourceSpan, `Unrecognized @for loop paramater "${param.expression}"`));
|
|
168
|
+
}
|
|
169
|
+
// Fill out any variables that haven't been defined explicitly.
|
|
170
|
+
for (const variableName of ALLOWED_FOR_LOOP_LET_VARIABLES) {
|
|
171
|
+
if (!result.context.hasOwnProperty(variableName)) {
|
|
172
|
+
result.context[variableName] =
|
|
173
|
+
new t.Variable(variableName, variableName, block.startSourceSpan, block.startSourceSpan);
|
|
174
|
+
}
|
|
150
175
|
}
|
|
151
176
|
return result;
|
|
152
177
|
}
|
|
153
178
|
/** Parses the `let` parameter of a `for` loop block. */
|
|
154
|
-
function parseLetParameter(sourceSpan, expression, context, errors) {
|
|
179
|
+
function parseLetParameter(sourceSpan, expression, span, context, errors) {
|
|
155
180
|
const parts = expression.split(',');
|
|
156
181
|
for (const part of parts) {
|
|
157
182
|
const expressionParts = part.split('=');
|
|
158
183
|
const name = expressionParts.length === 2 ? expressionParts[0].trim() : '';
|
|
159
|
-
const variableName = expressionParts.length === 2 ? expressionParts[1].trim() : '';
|
|
184
|
+
const variableName = (expressionParts.length === 2 ? expressionParts[1].trim() : '');
|
|
160
185
|
if (name.length === 0 || variableName.length === 0) {
|
|
161
|
-
errors.push(new ParseError(sourceSpan, `Invalid for loop "let" parameter. Parameter should match the pattern "<name> = <variable name>"`));
|
|
186
|
+
errors.push(new ParseError(sourceSpan, `Invalid @for loop "let" parameter. Parameter should match the pattern "<name> = <variable name>"`));
|
|
162
187
|
}
|
|
163
188
|
else if (!ALLOWED_FOR_LOOP_LET_VARIABLES.has(variableName)) {
|
|
164
189
|
errors.push(new ParseError(sourceSpan, `Unknown "let" parameter variable "${variableName}". The allowed variables are: ${Array.from(ALLOWED_FOR_LOOP_LET_VARIABLES).join(', ')}`));
|
|
@@ -167,86 +192,79 @@ function parseLetParameter(sourceSpan, expression, context, errors) {
|
|
|
167
192
|
errors.push(new ParseError(sourceSpan, `Duplicate "let" parameter variable "${variableName}"`));
|
|
168
193
|
}
|
|
169
194
|
else {
|
|
170
|
-
context[variableName] = name;
|
|
195
|
+
context[variableName] = new t.Variable(name, variableName, span, span);
|
|
171
196
|
}
|
|
172
197
|
}
|
|
173
198
|
}
|
|
174
|
-
/**
|
|
175
|
-
|
|
199
|
+
/**
|
|
200
|
+
* Checks that the shape of the blocks connected to an
|
|
201
|
+
* `@if` block is correct. Returns an array of errors.
|
|
202
|
+
*/
|
|
203
|
+
function validateIfConnectedBlocks(connectedBlocks) {
|
|
176
204
|
const errors = [];
|
|
177
205
|
let hasElse = false;
|
|
178
|
-
for (let i = 0; i <
|
|
179
|
-
const block =
|
|
180
|
-
|
|
181
|
-
if ((block.name !== 'if' || i > 0) && block.name !== 'else') {
|
|
182
|
-
errors.push(new ParseError(block.sourceSpan, `Unrecognized conditional block "${block.name}"`));
|
|
183
|
-
continue;
|
|
184
|
-
}
|
|
185
|
-
if (block.name === 'if') {
|
|
186
|
-
continue;
|
|
187
|
-
}
|
|
188
|
-
if (block.parameters.length === 0) {
|
|
206
|
+
for (let i = 0; i < connectedBlocks.length; i++) {
|
|
207
|
+
const block = connectedBlocks[i];
|
|
208
|
+
if (block.name === 'else') {
|
|
189
209
|
if (hasElse) {
|
|
190
|
-
errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one
|
|
210
|
+
errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one @else block'));
|
|
191
211
|
}
|
|
192
|
-
else if (
|
|
193
|
-
errors.push(new ParseError(block.sourceSpan, '
|
|
212
|
+
else if (connectedBlocks.length > 1 && i < connectedBlocks.length - 1) {
|
|
213
|
+
errors.push(new ParseError(block.sourceSpan, '@else block must be last inside the conditional'));
|
|
214
|
+
}
|
|
215
|
+
else if (block.parameters.length > 0) {
|
|
216
|
+
errors.push(new ParseError(block.sourceSpan, '@else block cannot have parameters'));
|
|
194
217
|
}
|
|
195
218
|
hasElse = true;
|
|
196
|
-
// `else if` is an edge case, because it has a space after the block name
|
|
197
|
-
// which means that the `if` is captured as a part of the parameters.
|
|
198
219
|
}
|
|
199
|
-
else if (
|
|
200
|
-
errors.push(new ParseError(block.sourceSpan,
|
|
220
|
+
else if (!ELSE_IF_PATTERN.test(block.name)) {
|
|
221
|
+
errors.push(new ParseError(block.sourceSpan, `Unrecognized conditional block @${block.name}`));
|
|
201
222
|
}
|
|
202
223
|
}
|
|
203
224
|
return errors;
|
|
204
225
|
}
|
|
205
226
|
/** Checks that the shape of a `switch` block is valid. Returns an array of errors. */
|
|
206
227
|
function validateSwitchBlock(ast) {
|
|
207
|
-
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
208
228
|
const errors = [];
|
|
209
229
|
let hasDefault = false;
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
return !(child instanceof html.Text) || child.value.trim().length > 0;
|
|
214
|
-
});
|
|
215
|
-
if (hasPrimary) {
|
|
216
|
-
errors.push(new ParseError(primaryBlock.sourceSpan, 'Switch block can only contain "case" and "default" blocks'));
|
|
217
|
-
}
|
|
218
|
-
if (primaryBlock.parameters.length !== 1) {
|
|
219
|
-
errors.push(new ParseError(primaryBlock.sourceSpan, 'Switch block must have exactly one parameter'));
|
|
230
|
+
if (ast.parameters.length !== 1) {
|
|
231
|
+
errors.push(new ParseError(ast.sourceSpan, '@switch block must have exactly one parameter'));
|
|
232
|
+
return errors;
|
|
220
233
|
}
|
|
221
|
-
for (const
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
}
|
|
234
|
+
for (const node of ast.children) {
|
|
235
|
+
// Skip over empty text nodes inside the switch block since they can be used for formatting.
|
|
236
|
+
if (node instanceof html.Text && node.value.trim().length === 0) {
|
|
237
|
+
continue;
|
|
226
238
|
}
|
|
227
|
-
|
|
239
|
+
if (!(node instanceof html.Block) || (node.name !== 'case' && node.name !== 'default')) {
|
|
240
|
+
errors.push(new ParseError(node.sourceSpan, '@switch block can only contain @case and @default blocks'));
|
|
241
|
+
continue;
|
|
242
|
+
}
|
|
243
|
+
if (node.name === 'default') {
|
|
228
244
|
if (hasDefault) {
|
|
229
|
-
errors.push(new ParseError(
|
|
245
|
+
errors.push(new ParseError(node.sourceSpan, '@switch block can only have one @default block'));
|
|
230
246
|
}
|
|
231
|
-
else if (
|
|
232
|
-
errors.push(new ParseError(
|
|
247
|
+
else if (node.parameters.length > 0) {
|
|
248
|
+
errors.push(new ParseError(node.sourceSpan, '@default block cannot have parameters'));
|
|
233
249
|
}
|
|
234
250
|
hasDefault = true;
|
|
235
251
|
}
|
|
236
|
-
else {
|
|
237
|
-
errors.push(new ParseError(
|
|
252
|
+
else if (node.name === 'case' && node.parameters.length !== 1) {
|
|
253
|
+
errors.push(new ParseError(node.sourceSpan, '@case block must have exactly one parameter'));
|
|
238
254
|
}
|
|
239
255
|
}
|
|
240
256
|
return errors;
|
|
241
257
|
}
|
|
242
|
-
|
|
258
|
+
/**
|
|
259
|
+
* Parses a block parameter into a binding AST.
|
|
260
|
+
* @param ast Block parameter that should be parsed.
|
|
261
|
+
* @param bindingParser Parser that the expression should be parsed with.
|
|
262
|
+
* @param part Specific part of the expression that should be parsed.
|
|
263
|
+
*/
|
|
264
|
+
function parseBlockParameterToBinding(ast, bindingParser, part) {
|
|
243
265
|
let start;
|
|
244
266
|
let end;
|
|
245
|
-
if (typeof part === '
|
|
246
|
-
start = part;
|
|
247
|
-
end = ast.expression.length;
|
|
248
|
-
}
|
|
249
|
-
else {
|
|
267
|
+
if (typeof part === 'string') {
|
|
250
268
|
// Note: `lastIndexOf` here should be enough to know the start index of the expression,
|
|
251
269
|
// because we know that it'll be at the end of the param. Ideally we could use the `d`
|
|
252
270
|
// flag when matching via regex and get the index from `match.indices`, but it's unclear
|
|
@@ -255,6 +273,10 @@ function parseBlockParameterToBinding(ast, bindingParser, part = 0) {
|
|
|
255
273
|
start = Math.max(0, ast.expression.lastIndexOf(part));
|
|
256
274
|
end = start + part.length;
|
|
257
275
|
}
|
|
276
|
+
else {
|
|
277
|
+
start = 0;
|
|
278
|
+
end = ast.expression.length;
|
|
279
|
+
}
|
|
258
280
|
return bindingParser.parseBinding(ast.expression.slice(start, end), false, ast.sourceSpan, ast.sourceSpan.start.offset + start);
|
|
259
281
|
}
|
|
260
282
|
/** Parses the parameter of a conditional block (`if` or `else if`). */
|
|
@@ -263,10 +285,7 @@ function parseConditionalBlockParameters(block, errors, bindingParser) {
|
|
|
263
285
|
errors.push(new ParseError(block.sourceSpan, 'Conditional block does not have an expression'));
|
|
264
286
|
return null;
|
|
265
287
|
}
|
|
266
|
-
const
|
|
267
|
-
const expression =
|
|
268
|
-
// Expressions for `{:else if}` blocks start at 2 to skip the `if` from the expression.
|
|
269
|
-
parseBlockParameterToBinding(block.parameters[0], bindingParser, isPrimaryIfBlock ? 0 : 2);
|
|
288
|
+
const expression = parseBlockParameterToBinding(block.parameters[0], bindingParser);
|
|
270
289
|
let expressionAlias = null;
|
|
271
290
|
// Start from 1 since we processed the first parameter already.
|
|
272
291
|
for (let i = 1; i < block.parameters.length; i++) {
|
|
@@ -277,14 +296,15 @@ function parseConditionalBlockParameters(block, errors, bindingParser) {
|
|
|
277
296
|
if (aliasMatch === null) {
|
|
278
297
|
errors.push(new ParseError(param.sourceSpan, `Unrecognized conditional paramater "${param.expression}"`));
|
|
279
298
|
}
|
|
280
|
-
else if (
|
|
281
|
-
errors.push(new ParseError(param.sourceSpan, '"as" expression is only allowed on the primary
|
|
299
|
+
else if (block.name !== 'if') {
|
|
300
|
+
errors.push(new ParseError(param.sourceSpan, '"as" expression is only allowed on the primary @if block'));
|
|
282
301
|
}
|
|
283
302
|
else if (expressionAlias !== null) {
|
|
284
303
|
errors.push(new ParseError(param.sourceSpan, 'Conditional can only have one "as" expression'));
|
|
285
304
|
}
|
|
286
305
|
else {
|
|
287
|
-
|
|
306
|
+
const name = aliasMatch[1].trim();
|
|
307
|
+
expressionAlias = new t.Variable(name, name, param.sourceSpan, param.sourceSpan);
|
|
288
308
|
}
|
|
289
309
|
}
|
|
290
310
|
return { expression, expressionAlias };
|
|
@@ -334,4 +354,4 @@ function stripOptionalParentheses(param, errors) {
|
|
|
334
354
|
}
|
|
335
355
|
return expression.slice(start, end);
|
|
336
356
|
}
|
|
337
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_control_flow.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_control_flow.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {ASTWithSource} from '../expression_parser/ast';\nimport * as html from '../ml_parser/ast';\nimport {ParseError, ParseSourceSpan} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\n\n/** Pattern for the expression in a for loop block. */\nconst FOR_LOOP_EXPRESSION_PATTERN = /^\\s*([0-9A-Za-z_$]*)\\s+of\\s+(.*)/;\n\n/** Pattern for the tracking expression in a for loop block. */\nconst FOR_LOOP_TRACK_PATTERN = /^track\\s+(.*)/;\n\n/** Pattern for the `as` expression in a conditional block. */\nconst CONDITIONAL_ALIAS_PATTERN = /^as\\s+(.*)/;\n\n/** Pattern used to identify an `else if` block. */\nconst ELSE_IF_PATTERN = /^if\\s/;\n\n/** Pattern used to identify a `let` parameter. */\nconst FOR_LOOP_LET_PATTERN = /^let\\s+(.*)/;\n\n/** Names of variables that are allowed to be used in the `let` expression of a `for` loop. */\nconst ALLOWED_FOR_LOOP_LET_VARIABLES =\n    new Set(['$index', '$first', '$last', '$even', '$odd', '$count']);\n\n/** Creates an `if` loop block from an HTML AST node. */\nexport function createIfBlock(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.IfBlock|null, errors: ParseError[]} {\n  const errors: ParseError[] = validateIfBlock(ast);\n  const branches: t.IfBlockBranch[] = [];\n\n  if (errors.length > 0) {\n    return {node: null, errors};\n  }\n\n  // Assumes that the structure is valid since we validated it above.\n  for (const block of ast.blocks) {\n    const children = html.visitAll(visitor, block.children);\n\n    // `{:else}` block.\n    if (block.name === 'else' && block.parameters.length === 0) {\n      branches.push(\n          new t.IfBlockBranch(null, children, null, block.sourceSpan, block.startSourceSpan));\n      continue;\n    }\n\n    const params = parseConditionalBlockParameters(block, errors, bindingParser);\n\n    if (params !== null) {\n      branches.push(new t.IfBlockBranch(\n          params.expression, children, params.expressionAlias, block.sourceSpan,\n          block.startSourceSpan));\n    }\n  }\n\n  return {\n    node: new t.IfBlock(branches, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors,\n  };\n}\n\n/** Creates a `for` loop block from an HTML AST node. */\nexport function createForLoop(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.ForLoopBlock|null, errors: ParseError[]} {\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const errors: ParseError[] = [];\n  const params = parseForLoopParameters(primaryBlock, errors, bindingParser);\n  let node: t.ForLoopBlock|null = null;\n  let empty: t.ForLoopBlockEmpty|null = null;\n\n  for (const block of secondaryBlocks) {\n    if (block.name === 'empty') {\n      if (empty !== null) {\n        errors.push(new ParseError(block.sourceSpan, 'For loop can only have one \"empty\" block'));\n      } else if (block.parameters.length > 0) {\n        errors.push(new ParseError(block.sourceSpan, 'Empty block cannot have parameters'));\n      } else {\n        empty = new t.ForLoopBlockEmpty(\n            html.visitAll(visitor, block.children), block.sourceSpan, block.startSourceSpan);\n      }\n    } else {\n      errors.push(new ParseError(block.sourceSpan, `Unrecognized loop block \"${block.name}\"`));\n    }\n  }\n\n  if (params !== null) {\n    if (params.trackBy === null) {\n      errors.push(new ParseError(ast.sourceSpan, 'For loop must have a \"track\" expression'));\n    } else {\n      node = new t.ForLoopBlock(\n          params.itemName, params.expression, params.trackBy, params.context,\n          html.visitAll(visitor, primaryBlock.children), empty, ast.sourceSpan, ast.startSourceSpan,\n          ast.endSourceSpan);\n    }\n  }\n\n  return {node, errors};\n}\n\n/** Creates a switch block from an HTML AST node. */\nexport function createSwitchBlock(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.SwitchBlock|null, errors: ParseError[]} {\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const errors = validateSwitchBlock(ast);\n\n  if (errors.length > 0) {\n    return {node: null, errors};\n  }\n\n  const primaryExpression = parseBlockParameterToBinding(primaryBlock.parameters[0], bindingParser);\n  const cases: t.SwitchBlockCase[] = [];\n  let defaultCase: t.SwitchBlockCase|null = null;\n\n  // Here we assume that all the blocks are valid given that we validated them above.\n  for (const block of secondaryBlocks) {\n    const expression = block.name === 'case' ?\n        parseBlockParameterToBinding(block.parameters[0], bindingParser) :\n        null;\n    const ast = new t.SwitchBlockCase(\n        expression, html.visitAll(visitor, block.children), block.sourceSpan,\n        block.startSourceSpan);\n\n    if (expression === null) {\n      defaultCase = ast;\n    } else {\n      cases.push(ast);\n    }\n  }\n\n  // Ensure that the default case is last in the array.\n  if (defaultCase !== null) {\n    cases.push(defaultCase);\n  }\n\n  return {\n    node: new t.SwitchBlock(\n        primaryExpression, cases, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors\n  };\n}\n\n/** Parses the parameters of a `for` loop block. */\nfunction parseForLoopParameters(\n    block: html.Block, errors: ParseError[], bindingParser: BindingParser) {\n  if (block.parameters.length === 0) {\n    errors.push(new ParseError(block.sourceSpan, 'For loop does not have an expression'));\n    return null;\n  }\n\n  const [expressionParam, ...secondaryParams] = block.parameters;\n  const match =\n      stripOptionalParentheses(expressionParam, errors)?.match(FOR_LOOP_EXPRESSION_PATTERN);\n\n  if (!match || match[2].trim().length === 0) {\n    errors.push(new ParseError(\n        expressionParam.sourceSpan,\n        'Cannot parse expression. For loop expression must match the pattern \"<identifier> of <expression>\"'));\n    return null;\n  }\n\n  const [, itemName, rawExpression] = match;\n  const result = {\n    itemName,\n    trackBy: null as ASTWithSource | null,\n    expression: parseBlockParameterToBinding(expressionParam, bindingParser, rawExpression),\n    context: null as t.ForLoopBlockContext | null,\n  };\n\n  for (const param of secondaryParams) {\n    const letMatch = param.expression.match(FOR_LOOP_LET_PATTERN);\n\n    if (letMatch !== null) {\n      result.context = result.context || {};\n      parseLetParameter(param.sourceSpan, letMatch[1], result.context, errors);\n      continue;\n    }\n\n    const trackMatch = param.expression.match(FOR_LOOP_TRACK_PATTERN);\n\n    if (trackMatch !== null) {\n      if (result.trackBy !== null) {\n        errors.push(\n            new ParseError(param.sourceSpan, 'For loop can only have one \"track\" expression'));\n      } else {\n        result.trackBy = parseBlockParameterToBinding(param, bindingParser, trackMatch[1]);\n      }\n      continue;\n    }\n\n    errors.push(\n        new ParseError(param.sourceSpan, `Unrecognized loop paramater \"${param.expression}\"`));\n  }\n\n  return result;\n}\n\n/** Parses the `let` parameter of a `for` loop block. */\nfunction parseLetParameter(\n    sourceSpan: ParseSourceSpan, expression: string, context: t.ForLoopBlockContext,\n    errors: ParseError[]): void {\n  const parts = expression.split(',');\n\n  for (const part of parts) {\n    const expressionParts = part.split('=');\n    const name = expressionParts.length === 2 ? expressionParts[0].trim() : '';\n    const variableName = expressionParts.length === 2 ? expressionParts[1].trim() : '';\n\n    if (name.length === 0 || variableName.length === 0) {\n      errors.push(new ParseError(\n          sourceSpan,\n          `Invalid for loop \"let\" parameter. Parameter should match the pattern \"<name> = <variable name>\"`));\n    } else if (!ALLOWED_FOR_LOOP_LET_VARIABLES.has(variableName)) {\n      errors.push(new ParseError(\n          sourceSpan,\n          `Unknown \"let\" parameter variable \"${variableName}\". The allowed variables are: ${\n              Array.from(ALLOWED_FOR_LOOP_LET_VARIABLES).join(', ')}`));\n    } else if (context.hasOwnProperty(variableName)) {\n      errors.push(\n          new ParseError(sourceSpan, `Duplicate \"let\" parameter variable \"${variableName}\"`));\n    } else {\n      context[variableName as keyof t.ForLoopBlockContext] = name;\n    }\n  }\n}\n\n/** Checks that the shape of a `if` block is valid. Returns an array of errors. */\nfunction validateIfBlock(ast: html.BlockGroup): ParseError[] {\n  const errors: ParseError[] = [];\n  let hasElse = false;\n\n  for (let i = 0; i < ast.blocks.length; i++) {\n    const block = ast.blocks[i];\n\n    // Conditional blocks only allow `if`, `else if` and `else` blocks.\n    if ((block.name !== 'if' || i > 0) && block.name !== 'else') {\n      errors.push(\n          new ParseError(block.sourceSpan, `Unrecognized conditional block \"${block.name}\"`));\n      continue;\n    }\n\n    if (block.name === 'if') {\n      continue;\n    }\n\n    if (block.parameters.length === 0) {\n      if (hasElse) {\n        errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one \"else\" block'));\n      } else if (ast.blocks.length > 1 && i < ast.blocks.length - 1) {\n        errors.push(\n            new ParseError(block.sourceSpan, 'Else block must be last inside the conditional'));\n      }\n      hasElse = true;\n\n      // `else if` is an edge case, because it has a space after the block name\n      // which means that the `if` is captured as a part of the parameters.\n    } else if (\n        block.parameters.length > 0 && !ELSE_IF_PATTERN.test(block.parameters[0].expression)) {\n      errors.push(new ParseError(block.sourceSpan, 'Else block cannot have parameters'));\n    }\n  }\n\n  return errors;\n}\n\n/** Checks that the shape of a `switch` block is valid. Returns an array of errors. */\nfunction validateSwitchBlock(ast: html.BlockGroup): ParseError[] {\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const errors: ParseError[] = [];\n  let hasDefault = false;\n  const hasPrimary = primaryBlock.children.length > 0 && primaryBlock.children.some(child => {\n    // The main block might have empty text nodes if `preserveWhitespaces` is enabled.\n    // Allow them since they might be used for code formatting.\n    return !(child instanceof html.Text) || child.value.trim().length > 0;\n  });\n\n  if (hasPrimary) {\n    errors.push(new ParseError(\n        primaryBlock.sourceSpan, 'Switch block can only contain \"case\" and \"default\" blocks'));\n  }\n\n  if (primaryBlock.parameters.length !== 1) {\n    errors.push(\n        new ParseError(primaryBlock.sourceSpan, 'Switch block must have exactly one parameter'));\n  }\n\n  for (const block of secondaryBlocks) {\n    if (block.name === 'case') {\n      if (block.parameters.length !== 1) {\n        errors.push(new ParseError(block.sourceSpan, 'Case block must have exactly one parameter'));\n      }\n    } else if (block.name === 'default') {\n      if (hasDefault) {\n        errors.push(\n            new ParseError(block.sourceSpan, 'Switch block can only have one \"default\" block'));\n      } else if (block.parameters.length > 0) {\n        errors.push(new ParseError(block.sourceSpan, 'Default block cannot have parameters'));\n      }\n      hasDefault = true;\n    } else {\n      errors.push(new ParseError(\n          block.sourceSpan, 'Switch block can only contain \"case\" and \"default\" blocks'));\n    }\n  }\n\n  return errors;\n}\n\n/**\n * Parses a block parameter into a binding AST.\n * @param ast Block parameter that should be parsed.\n * @param bindingParser Parser that the expression should be parsed with.\n * @param start Index from which to start the parsing. Defaults to 0.\n */\nfunction parseBlockParameterToBinding(\n    ast: html.BlockParameter, bindingParser: BindingParser, start?: number): ASTWithSource;\n\n/**\n * Parses a block parameter into a binding AST.\n * @param ast Block parameter that should be parsed.\n * @param bindingParser Parser that the expression should be parsed with.\n * @param part Specific part of the expression that should be parsed.\n */\nfunction parseBlockParameterToBinding(\n    ast: html.BlockParameter, bindingParser: BindingParser, part: string): ASTWithSource;\n\nfunction parseBlockParameterToBinding(\n    ast: html.BlockParameter, bindingParser: BindingParser,\n    part: string|number = 0): ASTWithSource {\n  let start: number;\n  let end: number;\n\n  if (typeof part === 'number') {\n    start = part;\n    end = ast.expression.length;\n  } else {\n    // Note: `lastIndexOf` here should be enough to know the start index of the expression,\n    // because we know that it'll be at the end of the param. Ideally we could use the `d`\n    // flag when matching via regex and get the index from `match.indices`, but it's unclear\n    // if we can use it yet since it's a relatively new feature. See:\n    // https://github.com/tc39/proposal-regexp-match-indices\n    start = Math.max(0, ast.expression.lastIndexOf(part));\n    end = start + part.length;\n  }\n\n  return bindingParser.parseBinding(\n      ast.expression.slice(start, end), false, ast.sourceSpan, ast.sourceSpan.start.offset + start);\n}\n\n/** Parses the parameter of a conditional block (`if` or `else if`). */\nfunction parseConditionalBlockParameters(\n    block: html.Block, errors: ParseError[], bindingParser: BindingParser) {\n  if (block.parameters.length === 0) {\n    errors.push(new ParseError(block.sourceSpan, 'Conditional block does not have an expression'));\n    return null;\n  }\n\n  const isPrimaryIfBlock = block.name === 'if';\n  const expression =\n      // Expressions for `{:else if}` blocks start at 2 to skip the `if` from the expression.\n      parseBlockParameterToBinding(block.parameters[0], bindingParser, isPrimaryIfBlock ? 0 : 2);\n  let expressionAlias: string|null = null;\n\n  // Start from 1 since we processed the first parameter already.\n  for (let i = 1; i < block.parameters.length; i++) {\n    const param = block.parameters[i];\n    const aliasMatch = param.expression.match(CONDITIONAL_ALIAS_PATTERN);\n\n    // For now conditionals can only have an `as` parameter.\n    // We may want to rework this later if we add more.\n    if (aliasMatch === null) {\n      errors.push(new ParseError(\n          param.sourceSpan, `Unrecognized conditional paramater \"${param.expression}\"`));\n    } else if (!isPrimaryIfBlock) {\n      errors.push(new ParseError(\n          param.sourceSpan, '\"as\" expression is only allowed on the primary \"if\" block'));\n    } else if (expressionAlias !== null) {\n      errors.push(\n          new ParseError(param.sourceSpan, 'Conditional can only have one \"as\" expression'));\n    } else {\n      expressionAlias = aliasMatch[1].trim();\n    }\n  }\n\n  return {expression, expressionAlias};\n}\n\n/** Strips optional parentheses around from a control from expression parameter. */\nfunction stripOptionalParentheses(param: html.BlockParameter, errors: ParseError[]): string|null {\n  const expression = param.expression;\n  const spaceRegex = /^\\s$/;\n  let openParens = 0;\n  let start = 0;\n  let end = expression.length - 1;\n\n  for (let i = 0; i < expression.length; i++) {\n    const char = expression[i];\n\n    if (char === '(') {\n      start = i + 1;\n      openParens++;\n    } else if (spaceRegex.test(char)) {\n      continue;\n    } else {\n      break;\n    }\n  }\n\n  if (openParens === 0) {\n    return expression;\n  }\n\n  for (let i = expression.length - 1; i > -1; i--) {\n    const char = expression[i];\n\n    if (char === ')') {\n      end = i;\n      openParens--;\n      if (openParens === 0) {\n        break;\n      }\n    } else if (spaceRegex.test(char)) {\n      continue;\n    } else {\n      break;\n    }\n  }\n\n  if (openParens !== 0) {\n    errors.push(new ParseError(param.sourceSpan, 'Unclosed parentheses in expression'));\n    return null;\n  }\n\n  return expression.slice(start, end);\n}\n"]}
|
|
357
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_control_flow.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_control_flow.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {ASTWithSource} from '../expression_parser/ast';\nimport * as html from '../ml_parser/ast';\nimport {ParseError, ParseSourceSpan} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\n\n/** Pattern for the expression in a for loop block. */\nconst FOR_LOOP_EXPRESSION_PATTERN = /^\\s*([0-9A-Za-z_$]*)\\s+of\\s+(.*)/;\n\n/** Pattern for the tracking expression in a for loop block. */\nconst FOR_LOOP_TRACK_PATTERN = /^track\\s+(.*)/;\n\n/** Pattern for the `as` expression in a conditional block. */\nconst CONDITIONAL_ALIAS_PATTERN = /^as\\s+(.*)/;\n\n/** Pattern used to identify an `else if` block. */\nconst ELSE_IF_PATTERN = /^else[^\\S\\r\\n]+if/;\n\n/** Pattern used to identify a `let` parameter. */\nconst FOR_LOOP_LET_PATTERN = /^let\\s+(.*)/;\n\n/** Names of variables that are allowed to be used in the `let` expression of a `for` loop. */\nconst ALLOWED_FOR_LOOP_LET_VARIABLES =\n    new Set<keyof t.ForLoopBlockContext>(['$index', '$first', '$last', '$even', '$odd', '$count']);\n\n/**\n * Predicate function that determines if a block with\n * a specific name cam be connected to a `for` block.\n */\nexport function isConnectedForLoopBlock(name: string): boolean {\n  return name === 'empty';\n}\n\n/**\n * Predicate function that determines if a block with\n * a specific name cam be connected to an `if` block.\n */\nexport function isConnectedIfLoopBlock(name: string): boolean {\n  return name === 'else' || ELSE_IF_PATTERN.test(name);\n}\n\n/** Creates an `if` loop block from an HTML AST node. */\nexport function createIfBlock(\n    ast: html.Block, connectedBlocks: html.Block[], visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.IfBlock|null, errors: ParseError[]} {\n  const errors: ParseError[] = validateIfConnectedBlocks(connectedBlocks);\n  const branches: t.IfBlockBranch[] = [];\n\n  if (errors.length > 0) {\n    return {node: null, errors};\n  }\n\n  const mainBlockParams = parseConditionalBlockParameters(ast, errors, bindingParser);\n\n  if (mainBlockParams !== null) {\n    branches.push(new t.IfBlockBranch(\n        mainBlockParams.expression, html.visitAll(visitor, ast.children, ast.children),\n        mainBlockParams.expressionAlias, ast.sourceSpan, ast.startSourceSpan));\n  }\n\n  // Assumes that the structure is valid since we validated it above.\n  for (const block of connectedBlocks) {\n    const children = html.visitAll(visitor, block.children, block.children);\n\n    if (ELSE_IF_PATTERN.test(block.name)) {\n      const params = parseConditionalBlockParameters(block, errors, bindingParser);\n\n      if (params !== null) {\n        branches.push(new t.IfBlockBranch(\n            params.expression, children, params.expressionAlias, block.sourceSpan,\n            block.startSourceSpan));\n      }\n    } else if (block.name === 'else') {\n      branches.push(\n          new t.IfBlockBranch(null, children, null, block.sourceSpan, block.startSourceSpan));\n    }\n  }\n\n  return {\n    node: new t.IfBlock(branches, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors,\n  };\n}\n\n/** Creates a `for` loop block from an HTML AST node. */\nexport function createForLoop(\n    ast: html.Block, connectedBlocks: html.Block[], visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.ForLoopBlock|null, errors: ParseError[]} {\n  const errors: ParseError[] = [];\n  const params = parseForLoopParameters(ast, errors, bindingParser);\n  let node: t.ForLoopBlock|null = null;\n  let empty: t.ForLoopBlockEmpty|null = null;\n\n  for (const block of connectedBlocks) {\n    if (block.name === 'empty') {\n      if (empty !== null) {\n        errors.push(new ParseError(block.sourceSpan, '@for loop can only have one @empty block'));\n      } else if (block.parameters.length > 0) {\n        errors.push(new ParseError(block.sourceSpan, '@empty block cannot have parameters'));\n      } else {\n        empty = new t.ForLoopBlockEmpty(\n            html.visitAll(visitor, block.children, block.children), block.sourceSpan,\n            block.startSourceSpan);\n      }\n    } else {\n      errors.push(new ParseError(block.sourceSpan, `Unrecognized @for loop block \"${block.name}\"`));\n    }\n  }\n\n  if (params !== null) {\n    if (params.trackBy === null) {\n      errors.push(new ParseError(ast.sourceSpan, '@for loop must have a \"track\" expression'));\n    } else {\n      node = new t.ForLoopBlock(\n          params.itemName, params.expression, params.trackBy, params.context,\n          html.visitAll(visitor, ast.children, ast.children), empty, ast.sourceSpan,\n          ast.startSourceSpan, ast.endSourceSpan);\n    }\n  }\n\n  return {node, errors};\n}\n\n/** Creates a switch block from an HTML AST node. */\nexport function createSwitchBlock(\n    ast: html.Block, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.SwitchBlock|null, errors: ParseError[]} {\n  const errors = validateSwitchBlock(ast);\n\n  if (errors.length > 0) {\n    return {node: null, errors};\n  }\n\n  const primaryExpression = parseBlockParameterToBinding(ast.parameters[0], bindingParser);\n  const cases: t.SwitchBlockCase[] = [];\n  let defaultCase: t.SwitchBlockCase|null = null;\n\n  // Here we assume that all the blocks are valid given that we validated them above.\n  for (const node of ast.children) {\n    if (!(node instanceof html.Block)) {\n      continue;\n    }\n\n    const expression = node.name === 'case' ?\n        parseBlockParameterToBinding(node.parameters[0], bindingParser) :\n        null;\n    const ast = new t.SwitchBlockCase(\n        expression, html.visitAll(visitor, node.children, node.children), node.sourceSpan,\n        node.startSourceSpan);\n\n    if (expression === null) {\n      defaultCase = ast;\n    } else {\n      cases.push(ast);\n    }\n  }\n\n  // Ensure that the default case is last in the array.\n  if (defaultCase !== null) {\n    cases.push(defaultCase);\n  }\n\n  return {\n    node: new t.SwitchBlock(\n        primaryExpression, cases, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors\n  };\n}\n\n/** Parses the parameters of a `for` loop block. */\nfunction parseForLoopParameters(\n    block: html.Block, errors: ParseError[], bindingParser: BindingParser) {\n  if (block.parameters.length === 0) {\n    errors.push(new ParseError(block.sourceSpan, '@for loop does not have an expression'));\n    return null;\n  }\n\n  const [expressionParam, ...secondaryParams] = block.parameters;\n  const match =\n      stripOptionalParentheses(expressionParam, errors)?.match(FOR_LOOP_EXPRESSION_PATTERN);\n\n  if (!match || match[2].trim().length === 0) {\n    errors.push(new ParseError(\n        expressionParam.sourceSpan,\n        'Cannot parse expression. @for loop expression must match the pattern \"<identifier> of <expression>\"'));\n    return null;\n  }\n\n  const [, itemName, rawExpression] = match;\n  const result = {\n    itemName: new t.Variable(\n        itemName, '$implicit', expressionParam.sourceSpan, expressionParam.sourceSpan),\n    trackBy: null as ASTWithSource | null,\n    expression: parseBlockParameterToBinding(expressionParam, bindingParser, rawExpression),\n    context: {} as t.ForLoopBlockContext,\n  };\n\n  for (const param of secondaryParams) {\n    const letMatch = param.expression.match(FOR_LOOP_LET_PATTERN);\n\n    if (letMatch !== null) {\n      parseLetParameter(param.sourceSpan, letMatch[1], param.sourceSpan, result.context, errors);\n      continue;\n    }\n\n    const trackMatch = param.expression.match(FOR_LOOP_TRACK_PATTERN);\n\n    if (trackMatch !== null) {\n      if (result.trackBy !== null) {\n        errors.push(\n            new ParseError(param.sourceSpan, '@for loop can only have one \"track\" expression'));\n      } else {\n        result.trackBy = parseBlockParameterToBinding(param, bindingParser, trackMatch[1]);\n      }\n      continue;\n    }\n\n    errors.push(\n        new ParseError(param.sourceSpan, `Unrecognized @for loop paramater \"${param.expression}\"`));\n  }\n\n  // Fill out any variables that haven't been defined explicitly.\n  for (const variableName of ALLOWED_FOR_LOOP_LET_VARIABLES) {\n    if (!result.context.hasOwnProperty(variableName)) {\n      result.context[variableName] =\n          new t.Variable(variableName, variableName, block.startSourceSpan, block.startSourceSpan);\n    }\n  }\n\n  return result;\n}\n\n/** Parses the `let` parameter of a `for` loop block. */\nfunction parseLetParameter(\n    sourceSpan: ParseSourceSpan, expression: string, span: ParseSourceSpan,\n    context: t.ForLoopBlockContext, errors: ParseError[]): void {\n  const parts = expression.split(',');\n\n  for (const part of parts) {\n    const expressionParts = part.split('=');\n    const name = expressionParts.length === 2 ? expressionParts[0].trim() : '';\n    const variableName = (expressionParts.length === 2 ? expressionParts[1].trim() : '') as\n        keyof t.ForLoopBlockContext;\n\n    if (name.length === 0 || variableName.length === 0) {\n      errors.push(new ParseError(\n          sourceSpan,\n          `Invalid @for loop \"let\" parameter. Parameter should match the pattern \"<name> = <variable name>\"`));\n    } else if (!ALLOWED_FOR_LOOP_LET_VARIABLES.has(variableName)) {\n      errors.push(new ParseError(\n          sourceSpan,\n          `Unknown \"let\" parameter variable \"${variableName}\". The allowed variables are: ${\n              Array.from(ALLOWED_FOR_LOOP_LET_VARIABLES).join(', ')}`));\n    } else if (context.hasOwnProperty(variableName)) {\n      errors.push(\n          new ParseError(sourceSpan, `Duplicate \"let\" parameter variable \"${variableName}\"`));\n    } else {\n      context[variableName] = new t.Variable(name, variableName, span, span);\n    }\n  }\n}\n\n/**\n * Checks that the shape of the blocks connected to an\n * `@if` block is correct. Returns an array of errors.\n */\nfunction validateIfConnectedBlocks(connectedBlocks: html.Block[]): ParseError[] {\n  const errors: ParseError[] = [];\n  let hasElse = false;\n\n  for (let i = 0; i < connectedBlocks.length; i++) {\n    const block = connectedBlocks[i];\n\n    if (block.name === 'else') {\n      if (hasElse) {\n        errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one @else block'));\n      } else if (connectedBlocks.length > 1 && i < connectedBlocks.length - 1) {\n        errors.push(\n            new ParseError(block.sourceSpan, '@else block must be last inside the conditional'));\n      } else if (block.parameters.length > 0) {\n        errors.push(new ParseError(block.sourceSpan, '@else block cannot have parameters'));\n      }\n      hasElse = true;\n    } else if (!ELSE_IF_PATTERN.test(block.name)) {\n      errors.push(\n          new ParseError(block.sourceSpan, `Unrecognized conditional block @${block.name}`));\n    }\n  }\n\n  return errors;\n}\n\n/** Checks that the shape of a `switch` block is valid. Returns an array of errors. */\nfunction validateSwitchBlock(ast: html.Block): ParseError[] {\n  const errors: ParseError[] = [];\n  let hasDefault = false;\n\n  if (ast.parameters.length !== 1) {\n    errors.push(new ParseError(ast.sourceSpan, '@switch block must have exactly one parameter'));\n    return errors;\n  }\n\n  for (const node of ast.children) {\n    // Skip over empty text nodes inside the switch block since they can be used for formatting.\n    if (node instanceof html.Text && node.value.trim().length === 0) {\n      continue;\n    }\n\n    if (!(node instanceof html.Block) || (node.name !== 'case' && node.name !== 'default')) {\n      errors.push(new ParseError(\n          node.sourceSpan, '@switch block can only contain @case and @default blocks'));\n      continue;\n    }\n\n    if (node.name === 'default') {\n      if (hasDefault) {\n        errors.push(\n            new ParseError(node.sourceSpan, '@switch block can only have one @default block'));\n      } else if (node.parameters.length > 0) {\n        errors.push(new ParseError(node.sourceSpan, '@default block cannot have parameters'));\n      }\n      hasDefault = true;\n    } else if (node.name === 'case' && node.parameters.length !== 1) {\n      errors.push(new ParseError(node.sourceSpan, '@case block must have exactly one parameter'));\n    }\n  }\n\n  return errors;\n}\n\n/**\n * Parses a block parameter into a binding AST.\n * @param ast Block parameter that should be parsed.\n * @param bindingParser Parser that the expression should be parsed with.\n * @param part Specific part of the expression that should be parsed.\n */\nfunction parseBlockParameterToBinding(\n    ast: html.BlockParameter, bindingParser: BindingParser, part?: string): ASTWithSource {\n  let start: number;\n  let end: number;\n\n  if (typeof part === 'string') {\n    // Note: `lastIndexOf` here should be enough to know the start index of the expression,\n    // because we know that it'll be at the end of the param. Ideally we could use the `d`\n    // flag when matching via regex and get the index from `match.indices`, but it's unclear\n    // if we can use it yet since it's a relatively new feature. See:\n    // https://github.com/tc39/proposal-regexp-match-indices\n    start = Math.max(0, ast.expression.lastIndexOf(part));\n    end = start + part.length;\n  } else {\n    start = 0;\n    end = ast.expression.length;\n  }\n\n  return bindingParser.parseBinding(\n      ast.expression.slice(start, end), false, ast.sourceSpan, ast.sourceSpan.start.offset + start);\n}\n\n/** Parses the parameter of a conditional block (`if` or `else if`). */\nfunction parseConditionalBlockParameters(\n    block: html.Block, errors: ParseError[], bindingParser: BindingParser) {\n  if (block.parameters.length === 0) {\n    errors.push(new ParseError(block.sourceSpan, 'Conditional block does not have an expression'));\n    return null;\n  }\n\n  const expression = parseBlockParameterToBinding(block.parameters[0], bindingParser);\n  let expressionAlias: t.Variable|null = null;\n\n  // Start from 1 since we processed the first parameter already.\n  for (let i = 1; i < block.parameters.length; i++) {\n    const param = block.parameters[i];\n    const aliasMatch = param.expression.match(CONDITIONAL_ALIAS_PATTERN);\n\n    // For now conditionals can only have an `as` parameter.\n    // We may want to rework this later if we add more.\n    if (aliasMatch === null) {\n      errors.push(new ParseError(\n          param.sourceSpan, `Unrecognized conditional paramater \"${param.expression}\"`));\n    } else if (block.name !== 'if') {\n      errors.push(new ParseError(\n          param.sourceSpan, '\"as\" expression is only allowed on the primary @if block'));\n    } else if (expressionAlias !== null) {\n      errors.push(\n          new ParseError(param.sourceSpan, 'Conditional can only have one \"as\" expression'));\n    } else {\n      const name = aliasMatch[1].trim();\n      expressionAlias = new t.Variable(name, name, param.sourceSpan, param.sourceSpan);\n    }\n  }\n\n  return {expression, expressionAlias};\n}\n\n/** Strips optional parentheses around from a control from expression parameter. */\nfunction stripOptionalParentheses(param: html.BlockParameter, errors: ParseError[]): string|null {\n  const expression = param.expression;\n  const spaceRegex = /^\\s$/;\n  let openParens = 0;\n  let start = 0;\n  let end = expression.length - 1;\n\n  for (let i = 0; i < expression.length; i++) {\n    const char = expression[i];\n\n    if (char === '(') {\n      start = i + 1;\n      openParens++;\n    } else if (spaceRegex.test(char)) {\n      continue;\n    } else {\n      break;\n    }\n  }\n\n  if (openParens === 0) {\n    return expression;\n  }\n\n  for (let i = expression.length - 1; i > -1; i--) {\n    const char = expression[i];\n\n    if (char === ')') {\n      end = i;\n      openParens--;\n      if (openParens === 0) {\n        break;\n      }\n    } else if (spaceRegex.test(char)) {\n      continue;\n    } else {\n      break;\n    }\n  }\n\n  if (openParens !== 0) {\n    errors.push(new ParseError(param.sourceSpan, 'Unclosed parentheses in expression'));\n    return null;\n  }\n\n  return expression.slice(start, end);\n}\n"]}
|