@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.
Files changed (56) hide show
  1. package/esm2022/src/compiler.mjs +1 -1
  2. package/esm2022/src/i18n/extractor_merger.mjs +1 -4
  3. package/esm2022/src/i18n/i18n_parser.mjs +1 -6
  4. package/esm2022/src/i18n/serializers/xliff.mjs +1 -3
  5. package/esm2022/src/i18n/serializers/xliff2.mjs +1 -3
  6. package/esm2022/src/i18n/serializers/xtb.mjs +1 -3
  7. package/esm2022/src/jit_compiler_facade.mjs +33 -7
  8. package/esm2022/src/ml_parser/ast.mjs +1 -17
  9. package/esm2022/src/ml_parser/html_whitespaces.mjs +2 -5
  10. package/esm2022/src/ml_parser/icu_ast_expander.mjs +2 -5
  11. package/esm2022/src/ml_parser/lexer.mjs +59 -49
  12. package/esm2022/src/ml_parser/parser.mjs +26 -71
  13. package/esm2022/src/ml_parser/tokens.mjs +1 -1
  14. package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
  15. package/esm2022/src/render3/partial/component.mjs +5 -2
  16. package/esm2022/src/render3/partial/directive.mjs +1 -1
  17. package/esm2022/src/render3/partial/factory.mjs +1 -1
  18. package/esm2022/src/render3/partial/injectable.mjs +1 -1
  19. package/esm2022/src/render3/partial/injector.mjs +1 -1
  20. package/esm2022/src/render3/partial/ng_module.mjs +1 -1
  21. package/esm2022/src/render3/partial/pipe.mjs +1 -1
  22. package/esm2022/src/render3/r3_ast.mjs +10 -3
  23. package/esm2022/src/render3/r3_class_metadata_compiler.mjs +14 -16
  24. package/esm2022/src/render3/r3_control_flow.mjs +116 -96
  25. package/esm2022/src/render3/r3_deferred_blocks.mjs +37 -39
  26. package/esm2022/src/render3/r3_deferred_triggers.mjs +30 -22
  27. package/esm2022/src/render3/r3_module_compiler.mjs +9 -11
  28. package/esm2022/src/render3/r3_template_transform.mjs +73 -31
  29. package/esm2022/src/render3/view/api.mjs +1 -1
  30. package/esm2022/src/render3/view/compiler.mjs +17 -6
  31. package/esm2022/src/render3/view/i18n/meta.mjs +1 -5
  32. package/esm2022/src/render3/view/t2_api.mjs +1 -1
  33. package/esm2022/src/render3/view/t2_binder.mjs +184 -79
  34. package/esm2022/src/render3/view/template.mjs +93 -83
  35. package/esm2022/src/template/pipeline/ir/src/enums.mjs +14 -7
  36. package/esm2022/src/template/pipeline/ir/src/expression.mjs +6 -4
  37. package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +30 -4
  38. package/esm2022/src/template/pipeline/src/compilation.mjs +6 -1
  39. package/esm2022/src/template/pipeline/src/conversion.mjs +7 -1
  40. package/esm2022/src/template/pipeline/src/emit.mjs +7 -1
  41. package/esm2022/src/template/pipeline/src/ingest.mjs +91 -42
  42. package/esm2022/src/template/pipeline/src/instruction.mjs +43 -27
  43. package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +22 -29
  44. package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +46 -0
  45. package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +33 -0
  46. package/esm2022/src/template/pipeline/src/phases/i18n_message_extraction.mjs +3 -2
  47. package/esm2022/src/template/pipeline/src/phases/no_listeners_on_templates.mjs +1 -5
  48. package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +39 -0
  49. package/esm2022/src/template/pipeline/src/phases/reify.mjs +23 -14
  50. package/esm2022/src/version.mjs +1 -1
  51. package/fesm2022/compiler.mjs +3425 -3072
  52. package/fesm2022/compiler.mjs.map +1 -1
  53. package/fesm2022/testing.mjs +1 -1
  54. package/index.d.ts +154 -143
  55. package/package.json +3 -3
  56. 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\s/;
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 = validateIfBlock(ast);
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 ast.blocks) {
32
- const children = html.visitAll(visitor, block.children);
33
- // `{:else}` block.
34
- if (block.name === 'else' && block.parameters.length === 0) {
35
- branches.push(new t.IfBlockBranch(null, children, null, block.sourceSpan, block.startSourceSpan));
36
- continue;
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
- const params = parseConditionalBlockParameters(block, errors, bindingParser);
39
- if (params !== null) {
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(primaryBlock, errors, bindingParser);
69
+ const params = parseForLoopParameters(ast, errors, bindingParser);
53
70
  let node = null;
54
71
  let empty = null;
55
- for (const block of secondaryBlocks) {
72
+ for (const block of connectedBlocks) {
56
73
  if (block.name === 'empty') {
57
74
  if (empty !== null) {
58
- errors.push(new ParseError(block.sourceSpan, 'For loop can only have one "empty" block'));
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, 'Empty block cannot have parameters'));
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, 'For loop must have a "track" expression'));
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, primaryBlock.children), empty, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);
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(primaryBlock.parameters[0], bindingParser);
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 block of secondaryBlocks) {
93
- const expression = block.name === 'case' ?
94
- parseBlockParameterToBinding(block.parameters[0], bindingParser) :
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, block.children), block.sourceSpan, block.startSourceSpan);
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, 'For loop does not have an expression'));
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. For loop expression must match the pattern "<identifier> of <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: null,
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
- result.context = result.context || {};
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, 'For loop can only have one "track" expression'));
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
- /** Checks that the shape of a `if` block is valid. Returns an array of errors. */
175
- function validateIfBlock(ast) {
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 < ast.blocks.length; i++) {
179
- const block = ast.blocks[i];
180
- // Conditional blocks only allow `if`, `else if` and `else` blocks.
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 "else" block'));
210
+ errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one @else block'));
191
211
  }
192
- else if (ast.blocks.length > 1 && i < ast.blocks.length - 1) {
193
- errors.push(new ParseError(block.sourceSpan, 'Else block must be last inside the conditional'));
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 (block.parameters.length > 0 && !ELSE_IF_PATTERN.test(block.parameters[0].expression)) {
200
- errors.push(new ParseError(block.sourceSpan, 'Else block cannot have parameters'));
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
- const hasPrimary = primaryBlock.children.length > 0 && primaryBlock.children.some(child => {
211
- // The main block might have empty text nodes if `preserveWhitespaces` is enabled.
212
- // Allow them since they might be used for code formatting.
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 block of secondaryBlocks) {
222
- if (block.name === 'case') {
223
- if (block.parameters.length !== 1) {
224
- errors.push(new ParseError(block.sourceSpan, 'Case block must have exactly one parameter'));
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
- else if (block.name === 'default') {
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(block.sourceSpan, 'Switch block can only have one "default" block'));
245
+ errors.push(new ParseError(node.sourceSpan, '@switch block can only have one @default block'));
230
246
  }
231
- else if (block.parameters.length > 0) {
232
- errors.push(new ParseError(block.sourceSpan, 'Default block cannot have parameters'));
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(block.sourceSpan, 'Switch block can only contain "case" and "default" blocks'));
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
- function parseBlockParameterToBinding(ast, bindingParser, part = 0) {
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 === 'number') {
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 isPrimaryIfBlock = block.name === 'if';
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 (!isPrimaryIfBlock) {
281
- errors.push(new ParseError(param.sourceSpan, '"as" expression is only allowed on the primary "if" block'));
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
- expressionAlias = aliasMatch[1].trim();
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"]}