@angular/compiler 16.2.0 → 17.0.0-next.0
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 +3 -3
- package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
- package/esm2022/src/render3/partial/component.mjs +4 -1
- 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 +113 -9
- package/esm2022/src/render3/r3_class_metadata_compiler.mjs +55 -1
- package/esm2022/src/render3/r3_control_flow.mjs +289 -0
- package/esm2022/src/render3/r3_deferred_blocks.mjs +16 -9
- package/esm2022/src/render3/r3_deferred_triggers.mjs +33 -19
- package/esm2022/src/render3/r3_identifiers.mjs +17 -1
- package/esm2022/src/render3/r3_template_transform.mjs +28 -7
- package/esm2022/src/render3/view/api.mjs +1 -1
- package/esm2022/src/render3/view/compiler.mjs +12 -2
- package/esm2022/src/render3/view/t2_binder.mjs +66 -5
- package/esm2022/src/render3/view/template.mjs +130 -54
- package/esm2022/src/template/pipeline/ir/index.mjs +2 -3
- package/esm2022/src/template/pipeline/ir/src/enums.mjs +41 -3
- package/esm2022/src/template/pipeline/ir/src/expression.mjs +11 -1
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +16 -4
- package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +5 -3
- package/esm2022/src/template/pipeline/src/emit.mjs +3 -1
- package/esm2022/src/template/pipeline/src/ingest.mjs +10 -10
- package/esm2022/src/template/pipeline/src/phases/attribute_extraction.mjs +43 -75
- package/esm2022/src/template/pipeline/src/phases/binding_specialization.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +90 -13
- package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +38 -0
- package/esm2022/src/version.mjs +1 -1
- package/fesm2022/compiler.mjs +1110 -425
- package/fesm2022/compiler.mjs.map +1 -1
- package/fesm2022/testing.mjs +1 -1
- package/index.d.ts +141 -5
- package/package.json +2 -2
- package/testing/index.d.ts +1 -1
- package/esm2022/src/template/pipeline/ir/src/element.mjs +0 -108
|
@@ -20,4 +20,58 @@ export function compileClassMetadata(metadata) {
|
|
|
20
20
|
const iife = o.fn([], [devOnlyGuardedExpression(fnCall).toStmt()]);
|
|
21
21
|
return iife.callFn([]);
|
|
22
22
|
}
|
|
23
|
-
|
|
23
|
+
/**
|
|
24
|
+
* Wraps the `setClassMetadata` function with extra logic that dynamically
|
|
25
|
+
* loads dependencies from `{#defer}` blocks.
|
|
26
|
+
*
|
|
27
|
+
* Generates a call like this:
|
|
28
|
+
* ```
|
|
29
|
+
* setClassMetadataAsync(type, () => {
|
|
30
|
+
* return [
|
|
31
|
+
* import('./cmp-a').then(m => m.CmpA);
|
|
32
|
+
* import('./cmp-b').then(m => m.CmpB);
|
|
33
|
+
* ];
|
|
34
|
+
* }, (CmpA, CmpB) => {
|
|
35
|
+
* setClassMetadata(type, decorators, ctorParameters, propParameters);
|
|
36
|
+
* });
|
|
37
|
+
* ```
|
|
38
|
+
*
|
|
39
|
+
* Similar to the `setClassMetadata` call, it's wrapped into the `ngDevMode`
|
|
40
|
+
* check to tree-shake away this code in production mode.
|
|
41
|
+
*/
|
|
42
|
+
export function compileComponentClassMetadata(metadata, deferrableTypes) {
|
|
43
|
+
if (deferrableTypes.size === 0) {
|
|
44
|
+
// If there are no deferrable symbols - just generate a regular `setClassMetadata` call.
|
|
45
|
+
return compileClassMetadata(metadata);
|
|
46
|
+
}
|
|
47
|
+
const dynamicImports = [];
|
|
48
|
+
const importedSymbols = [];
|
|
49
|
+
for (const [symbolName, importPath] of deferrableTypes) {
|
|
50
|
+
// e.g. `function(m) { return m.CmpA; }`
|
|
51
|
+
const innerFn = o.fn([new o.FnParam('m', o.DYNAMIC_TYPE)], [new o.ReturnStatement(o.variable('m').prop(symbolName))]);
|
|
52
|
+
// e.g. `import('./cmp-a').then(...)`
|
|
53
|
+
const importExpr = (new o.DynamicImportExpr(importPath)).prop('then').callFn([innerFn]);
|
|
54
|
+
dynamicImports.push(importExpr);
|
|
55
|
+
importedSymbols.push(new o.FnParam(symbolName, o.DYNAMIC_TYPE));
|
|
56
|
+
}
|
|
57
|
+
// e.g. `function() { return [ ... ]; }`
|
|
58
|
+
const dependencyLoadingFn = o.fn([], [new o.ReturnStatement(o.literalArr(dynamicImports))]);
|
|
59
|
+
// e.g. `setClassMetadata(...)`
|
|
60
|
+
const setClassMetadataCall = o.importExpr(R3.setClassMetadata).callFn([
|
|
61
|
+
metadata.type,
|
|
62
|
+
metadata.decorators,
|
|
63
|
+
metadata.ctorParameters ?? o.literal(null),
|
|
64
|
+
metadata.propDecorators ?? o.literal(null),
|
|
65
|
+
]);
|
|
66
|
+
// e.g. `function(CmpA) { setClassMetadata(...); }`
|
|
67
|
+
const setClassMetaWrapper = o.fn(importedSymbols, [setClassMetadataCall.toStmt()]);
|
|
68
|
+
// Final `setClassMetadataAsync()` call with all arguments
|
|
69
|
+
const setClassMetaAsync = o.importExpr(R3.setClassMetadataAsync).callFn([
|
|
70
|
+
metadata.type, dependencyLoadingFn, setClassMetaWrapper
|
|
71
|
+
]);
|
|
72
|
+
// Generate an ngDevMode guarded call to `setClassMetadataAsync` with
|
|
73
|
+
// the class identifier and its metadata, so that this call can be tree-shaken.
|
|
74
|
+
const iife = o.fn([], [devOnlyGuardedExpression(setClassMetaAsync).toStmt()]);
|
|
75
|
+
return iife.callFn([]);
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_class_metadata_compiler.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_class_metadata_compiler.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 */\nimport * as o from '../output/output_ast';\n\nimport {Identifiers as R3} from './r3_identifiers';\nimport {devOnlyGuardedExpression} from './util';\n\nexport type CompileClassMetadataFn = (metadata: R3ClassMetadata) => o.Expression;\n\n/**\n * Metadata of a class which captures the original Angular decorators of a class. The original\n * decorators are preserved in the generated code to allow TestBed APIs to recompile the class\n * using the original decorator with a set of overrides applied.\n */\nexport interface R3ClassMetadata {\n  /**\n   * The class type for which the metadata is captured.\n   */\n  type: o.Expression;\n\n  /**\n   * An expression representing the Angular decorators that were applied on the class.\n   */\n  decorators: o.Expression;\n\n  /**\n   * An expression representing the Angular decorators applied to constructor parameters, or `null`\n   * if there is no constructor.\n   */\n  ctorParameters: o.Expression|null;\n\n  /**\n   * An expression representing the Angular decorators that were applied on the properties of the\n   * class, or `null` if no properties have decorators.\n   */\n  propDecorators: o.Expression|null;\n}\n\nexport function compileClassMetadata(metadata: R3ClassMetadata): o.Expression {\n  // Generate an ngDevMode guarded call to setClassMetadata with the class identifier and its\n  // metadata.\n  const fnCall = o.importExpr(R3.setClassMetadata).callFn([\n    metadata.type,\n    metadata.decorators,\n    metadata.ctorParameters ?? o.literal(null),\n    metadata.propDecorators ?? o.literal(null),\n  ]);\n  const iife = o.fn([], [devOnlyGuardedExpression(fnCall).toStmt()]);\n  return iife.callFn([]);\n}\n\n/**\n * Wraps the `setClassMetadata` function with extra logic that dynamically\n * loads dependencies from `{#defer}` blocks.\n *\n * Generates a call like this:\n * ```\n * setClassMetadataAsync(type, () => {\n *   return [\n *     import('./cmp-a').then(m => m.CmpA);\n *     import('./cmp-b').then(m => m.CmpB);\n *   ];\n * }, (CmpA, CmpB) => {\n *   setClassMetadata(type, decorators, ctorParameters, propParameters);\n * });\n * ```\n *\n * Similar to the `setClassMetadata` call, it's wrapped into the `ngDevMode`\n * check to tree-shake away this code in production mode.\n */\nexport function compileComponentClassMetadata(\n    metadata: R3ClassMetadata, deferrableTypes: Map<string, string>): o.Expression {\n  if (deferrableTypes.size === 0) {\n    // If there are no deferrable symbols - just generate a regular `setClassMetadata` call.\n    return compileClassMetadata(metadata);\n  }\n\n  const dynamicImports: o.Expression[] = [];\n  const importedSymbols: o.FnParam[] = [];\n  for (const [symbolName, importPath] of deferrableTypes) {\n    // e.g. `function(m) { return m.CmpA; }`\n    const innerFn = o.fn(\n        [new o.FnParam('m', o.DYNAMIC_TYPE)],\n        [new o.ReturnStatement(o.variable('m').prop(symbolName))]);\n\n    // e.g. `import('./cmp-a').then(...)`\n    const importExpr = (new o.DynamicImportExpr(importPath)).prop('then').callFn([innerFn]);\n\n    dynamicImports.push(importExpr);\n    importedSymbols.push(new o.FnParam(symbolName, o.DYNAMIC_TYPE));\n  }\n\n  // e.g. `function() { return [ ... ]; }`\n  const dependencyLoadingFn = o.fn([], [new o.ReturnStatement(o.literalArr(dynamicImports))]);\n\n  // e.g. `setClassMetadata(...)`\n  const setClassMetadataCall = o.importExpr(R3.setClassMetadata).callFn([\n    metadata.type,\n    metadata.decorators,\n    metadata.ctorParameters ?? o.literal(null),\n    metadata.propDecorators ?? o.literal(null),\n  ]);\n\n  // e.g. `function(CmpA) { setClassMetadata(...); }`\n  const setClassMetaWrapper = o.fn(importedSymbols, [setClassMetadataCall.toStmt()]);\n\n  // Final `setClassMetadataAsync()` call with all arguments\n  const setClassMetaAsync = o.importExpr(R3.setClassMetadataAsync).callFn([\n    metadata.type, dependencyLoadingFn, setClassMetaWrapper\n  ]);\n\n  // Generate an ngDevMode guarded call to `setClassMetadataAsync` with\n  // the class identifier and its metadata, so that this call can be tree-shaken.\n  const iife = o.fn([], [devOnlyGuardedExpression(setClassMetaAsync).toStmt()]);\n  return iife.callFn([]);\n}\n"]}
|
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
import * as html from '../ml_parser/ast';
|
|
9
|
+
import { ParseError } from '../parse_util';
|
|
10
|
+
import * as t from './r3_ast';
|
|
11
|
+
/** Pattern for the expression in a for loop block. */
|
|
12
|
+
const FOR_LOOP_EXPRESSION_PATTERN = /^\s*([0-9A-Za-z_$]*)\s+of\s+(.*)/;
|
|
13
|
+
/** Pattern for the tracking expression in a for loop block. */
|
|
14
|
+
const FOR_LOOP_TRACK_PATTERN = /^track\s+(.*)/;
|
|
15
|
+
/** Pattern for the `as` expression in a conditional block. */
|
|
16
|
+
const CONDITIONAL_ALIAS_PATTERN = /^as\s+(.*)/;
|
|
17
|
+
/** Pattern used to identify an `else if` block. */
|
|
18
|
+
const ELSE_IF_PATTERN = /^if\s/;
|
|
19
|
+
/** Creates an `if` loop block from an HTML AST node. */
|
|
20
|
+
export function createIfBlock(ast, visitor, bindingParser) {
|
|
21
|
+
const errors = validateIfBlock(ast);
|
|
22
|
+
const branches = [];
|
|
23
|
+
if (errors.length > 0) {
|
|
24
|
+
return { node: null, errors };
|
|
25
|
+
}
|
|
26
|
+
// Assumes that the structure is valid since we validated it above.
|
|
27
|
+
for (const block of ast.blocks) {
|
|
28
|
+
const children = html.visitAll(visitor, block.children);
|
|
29
|
+
// `{:else}` block.
|
|
30
|
+
if (block.name === 'else' && block.parameters.length === 0) {
|
|
31
|
+
branches.push(new t.IfBlockBranch(null, children, null, block.sourceSpan, block.startSourceSpan));
|
|
32
|
+
continue;
|
|
33
|
+
}
|
|
34
|
+
// Expressions for `{:else if}` blocks start at 2 to skip the `if` from the expression.
|
|
35
|
+
const expressionStart = block.name === 'if' ? 0 : 2;
|
|
36
|
+
const params = parseConditionalBlockParameters(block, errors, bindingParser, expressionStart);
|
|
37
|
+
if (params !== null) {
|
|
38
|
+
branches.push(new t.IfBlockBranch(params.expression, children, params.expressionAlias, block.sourceSpan, block.startSourceSpan));
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
return {
|
|
42
|
+
node: new t.IfBlock(branches, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),
|
|
43
|
+
errors,
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
/** Creates a `for` loop block from an HTML AST node. */
|
|
47
|
+
export function createForLoop(ast, visitor, bindingParser) {
|
|
48
|
+
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
49
|
+
const errors = [];
|
|
50
|
+
const params = parseForLoopParameters(primaryBlock, errors, bindingParser);
|
|
51
|
+
let node = null;
|
|
52
|
+
let empty = null;
|
|
53
|
+
for (const block of secondaryBlocks) {
|
|
54
|
+
if (block.name === 'empty') {
|
|
55
|
+
if (empty !== null) {
|
|
56
|
+
errors.push(new ParseError(block.sourceSpan, 'For loop can only have one "empty" block'));
|
|
57
|
+
}
|
|
58
|
+
else if (block.parameters.length > 0) {
|
|
59
|
+
errors.push(new ParseError(block.sourceSpan, 'Empty block cannot have parameters'));
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
empty = new t.ForLoopBlockEmpty(html.visitAll(visitor, block.children), block.sourceSpan, block.startSourceSpan);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
errors.push(new ParseError(block.sourceSpan, `Unrecognized loop block "${block.name}"`));
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
if (params !== null) {
|
|
70
|
+
if (params.trackBy === null) {
|
|
71
|
+
errors.push(new ParseError(ast.sourceSpan, 'For loop must have a "track" expression'));
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
node = new t.ForLoopBlock(params.itemName, params.expression, params.trackBy, html.visitAll(visitor, primaryBlock.children), empty, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
return { node, errors };
|
|
78
|
+
}
|
|
79
|
+
/** Creates a switch block from an HTML AST node. */
|
|
80
|
+
export function createSwitchBlock(ast, visitor, bindingParser) {
|
|
81
|
+
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
82
|
+
const errors = validateSwitchBlock(ast);
|
|
83
|
+
if (errors.length > 0) {
|
|
84
|
+
return { node: null, errors };
|
|
85
|
+
}
|
|
86
|
+
const primaryExpression = parseBlockParameterToBinding(primaryBlock.parameters[0], bindingParser);
|
|
87
|
+
const cases = [];
|
|
88
|
+
let defaultCase = null;
|
|
89
|
+
// Here we assume that all the blocks are valid given that we validated them above.
|
|
90
|
+
for (const block of secondaryBlocks) {
|
|
91
|
+
const expression = block.name === 'case' ?
|
|
92
|
+
parseBlockParameterToBinding(block.parameters[0], bindingParser) :
|
|
93
|
+
null;
|
|
94
|
+
const ast = new t.SwitchBlockCase(expression, html.visitAll(visitor, block.children), block.sourceSpan, block.startSourceSpan);
|
|
95
|
+
if (expression === null) {
|
|
96
|
+
defaultCase = ast;
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
cases.push(ast);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
// Ensure that the default case is last in the array.
|
|
103
|
+
if (defaultCase !== null) {
|
|
104
|
+
cases.push(defaultCase);
|
|
105
|
+
}
|
|
106
|
+
return {
|
|
107
|
+
node: new t.SwitchBlock(primaryExpression, cases, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),
|
|
108
|
+
errors
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
/** Parses the parameters of a `for` loop block. */
|
|
112
|
+
function parseForLoopParameters(block, errors, bindingParser) {
|
|
113
|
+
if (block.parameters.length === 0) {
|
|
114
|
+
errors.push(new ParseError(block.sourceSpan, 'For loop does not have an expression'));
|
|
115
|
+
return null;
|
|
116
|
+
}
|
|
117
|
+
const [expressionParam, ...secondaryParams] = block.parameters;
|
|
118
|
+
const match = stripOptionalParentheses(expressionParam, errors)?.match(FOR_LOOP_EXPRESSION_PATTERN);
|
|
119
|
+
if (!match || match[2].trim().length === 0) {
|
|
120
|
+
errors.push(new ParseError(expressionParam.sourceSpan, 'Cannot parse expression. For loop expression must match the pattern "<identifier> of <expression>"'));
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
123
|
+
const [, itemName, rawExpression] = match;
|
|
124
|
+
const result = {
|
|
125
|
+
itemName,
|
|
126
|
+
trackBy: null,
|
|
127
|
+
expression: bindingParser.parseBinding(rawExpression, false, expressionParam.sourceSpan,
|
|
128
|
+
// Note: `lastIndexOf` here should be enough to know the start index of the expression,
|
|
129
|
+
// because we know that it'll be the last matching group. Ideally we could use the `d`
|
|
130
|
+
// flag on the regex and get the index from `match.indices`, but it's unclear if we can
|
|
131
|
+
// use it yet since it's a relatively new feature. See:
|
|
132
|
+
// https://github.com/tc39/proposal-regexp-match-indices
|
|
133
|
+
Math.max(0, expressionParam.expression.lastIndexOf(rawExpression)))
|
|
134
|
+
};
|
|
135
|
+
for (const param of secondaryParams) {
|
|
136
|
+
const trackMatch = param.expression.match(FOR_LOOP_TRACK_PATTERN);
|
|
137
|
+
// For now loops can only have a `track` parameter.
|
|
138
|
+
// We may want to rework this later if we add more.
|
|
139
|
+
if (trackMatch === null) {
|
|
140
|
+
errors.push(new ParseError(param.sourceSpan, `Unrecognized loop paramater "${param.expression}"`));
|
|
141
|
+
}
|
|
142
|
+
else if (result.trackBy !== null) {
|
|
143
|
+
errors.push(new ParseError(param.sourceSpan, 'For loop can only have one "track" expression'));
|
|
144
|
+
}
|
|
145
|
+
else {
|
|
146
|
+
result.trackBy = trackMatch[1].trim();
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
return result;
|
|
150
|
+
}
|
|
151
|
+
/** Checks that the shape of a `if` block is valid. Returns an array of errors. */
|
|
152
|
+
function validateIfBlock(ast) {
|
|
153
|
+
const errors = [];
|
|
154
|
+
let hasElse = false;
|
|
155
|
+
for (let i = 0; i < ast.blocks.length; i++) {
|
|
156
|
+
const block = ast.blocks[i];
|
|
157
|
+
// Conditional blocks only allow `if`, `else if` and `else` blocks.
|
|
158
|
+
if ((block.name !== 'if' || i > 0) && block.name !== 'else') {
|
|
159
|
+
errors.push(new ParseError(block.sourceSpan, `Unrecognized conditional block "${block.name}"`));
|
|
160
|
+
continue;
|
|
161
|
+
}
|
|
162
|
+
if (block.name === 'if') {
|
|
163
|
+
continue;
|
|
164
|
+
}
|
|
165
|
+
if (block.parameters.length === 0) {
|
|
166
|
+
if (hasElse) {
|
|
167
|
+
errors.push(new ParseError(block.sourceSpan, 'Conditional can only have one "else" block'));
|
|
168
|
+
}
|
|
169
|
+
else if (ast.blocks.length > 1 && i < ast.blocks.length - 1) {
|
|
170
|
+
errors.push(new ParseError(block.sourceSpan, 'Else block must be last inside the conditional'));
|
|
171
|
+
}
|
|
172
|
+
hasElse = true;
|
|
173
|
+
// `else if` is an edge case, because it has a space after the block name
|
|
174
|
+
// which means that the `if` is captured as a part of the parameters.
|
|
175
|
+
}
|
|
176
|
+
else if (block.parameters.length > 0 && !ELSE_IF_PATTERN.test(block.parameters[0].expression)) {
|
|
177
|
+
errors.push(new ParseError(block.sourceSpan, 'Else block cannot have parameters'));
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
return errors;
|
|
181
|
+
}
|
|
182
|
+
/** Checks that the shape of a `switch` block is valid. Returns an array of errors. */
|
|
183
|
+
function validateSwitchBlock(ast) {
|
|
184
|
+
const [primaryBlock, ...secondaryBlocks] = ast.blocks;
|
|
185
|
+
const errors = [];
|
|
186
|
+
let hasDefault = false;
|
|
187
|
+
if (primaryBlock.children.length > 0) {
|
|
188
|
+
errors.push(new ParseError(primaryBlock.sourceSpan, 'Switch block can only contain "case" and "default" blocks'));
|
|
189
|
+
}
|
|
190
|
+
if (primaryBlock.parameters.length !== 1) {
|
|
191
|
+
errors.push(new ParseError(primaryBlock.sourceSpan, 'Switch block must have exactly one parameter'));
|
|
192
|
+
}
|
|
193
|
+
for (const block of secondaryBlocks) {
|
|
194
|
+
if (block.name === 'case') {
|
|
195
|
+
if (block.parameters.length !== 1) {
|
|
196
|
+
errors.push(new ParseError(block.sourceSpan, 'Case block must have exactly one parameter'));
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
else if (block.name === 'default') {
|
|
200
|
+
if (hasDefault) {
|
|
201
|
+
errors.push(new ParseError(block.sourceSpan, 'Switch block can only have one "default" block'));
|
|
202
|
+
}
|
|
203
|
+
else if (block.parameters.length > 0) {
|
|
204
|
+
errors.push(new ParseError(block.sourceSpan, 'Default block cannot have parameters'));
|
|
205
|
+
}
|
|
206
|
+
hasDefault = true;
|
|
207
|
+
}
|
|
208
|
+
else {
|
|
209
|
+
errors.push(new ParseError(block.sourceSpan, 'Switch block can only contain "case" and "default" blocks'));
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
return errors;
|
|
213
|
+
}
|
|
214
|
+
/** Parses a block parameter into a binding AST. */
|
|
215
|
+
function parseBlockParameterToBinding(ast, bindingParser, start = 0) {
|
|
216
|
+
return bindingParser.parseBinding(ast.expression.slice(start), false, ast.sourceSpan, ast.sourceSpan.start.offset + start);
|
|
217
|
+
}
|
|
218
|
+
/** Parses the parameter of a conditional block (`if` or `else if`). */
|
|
219
|
+
function parseConditionalBlockParameters(block, errors, bindingParser, primaryExpressionStart) {
|
|
220
|
+
if (block.parameters.length === 0) {
|
|
221
|
+
errors.push(new ParseError(block.sourceSpan, 'Conditional block does not have an expression'));
|
|
222
|
+
return null;
|
|
223
|
+
}
|
|
224
|
+
const expression = parseBlockParameterToBinding(block.parameters[0], bindingParser, primaryExpressionStart);
|
|
225
|
+
let expressionAlias = null;
|
|
226
|
+
// Start from 1 since we processed the first parameter already.
|
|
227
|
+
for (let i = 1; i < block.parameters.length; i++) {
|
|
228
|
+
const param = block.parameters[i];
|
|
229
|
+
const aliasMatch = param.expression.match(CONDITIONAL_ALIAS_PATTERN);
|
|
230
|
+
// For now conditionals can only have an `as` parameter.
|
|
231
|
+
// We may want to rework this later if we add more.
|
|
232
|
+
if (aliasMatch === null) {
|
|
233
|
+
errors.push(new ParseError(param.sourceSpan, `Unrecognized conditional paramater "${param.expression}"`));
|
|
234
|
+
}
|
|
235
|
+
else if (expressionAlias !== null) {
|
|
236
|
+
errors.push(new ParseError(param.sourceSpan, 'Conditional can only have one "as" expression'));
|
|
237
|
+
}
|
|
238
|
+
else {
|
|
239
|
+
expressionAlias = aliasMatch[1].trim();
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
return { expression, expressionAlias };
|
|
243
|
+
}
|
|
244
|
+
/** Strips optional parentheses around from a control from expression parameter. */
|
|
245
|
+
function stripOptionalParentheses(param, errors) {
|
|
246
|
+
const expression = param.expression;
|
|
247
|
+
const spaceRegex = /^\s$/;
|
|
248
|
+
let openParens = 0;
|
|
249
|
+
let start = 0;
|
|
250
|
+
let end = expression.length - 1;
|
|
251
|
+
for (let i = 0; i < expression.length; i++) {
|
|
252
|
+
const char = expression[i];
|
|
253
|
+
if (char === '(') {
|
|
254
|
+
start = i + 1;
|
|
255
|
+
openParens++;
|
|
256
|
+
}
|
|
257
|
+
else if (spaceRegex.test(char)) {
|
|
258
|
+
continue;
|
|
259
|
+
}
|
|
260
|
+
else {
|
|
261
|
+
break;
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
if (openParens === 0) {
|
|
265
|
+
return expression;
|
|
266
|
+
}
|
|
267
|
+
for (let i = expression.length - 1; i > -1; i--) {
|
|
268
|
+
const char = expression[i];
|
|
269
|
+
if (char === ')') {
|
|
270
|
+
end = i;
|
|
271
|
+
openParens--;
|
|
272
|
+
if (openParens === 0) {
|
|
273
|
+
break;
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
else if (spaceRegex.test(char)) {
|
|
277
|
+
continue;
|
|
278
|
+
}
|
|
279
|
+
else {
|
|
280
|
+
break;
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
if (openParens !== 0) {
|
|
284
|
+
errors.push(new ParseError(param.sourceSpan, 'Unclosed parentheses in expression'));
|
|
285
|
+
return null;
|
|
286
|
+
}
|
|
287
|
+
return expression.slice(start, end);
|
|
288
|
+
}
|
|
289
|
+
//# 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} 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/** 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    // Expressions for `{:else if}` blocks start at 2 to skip the `if` from the expression.\n    const expressionStart = block.name === 'if' ? 0 : 2;\n    const params = parseConditionalBlockParameters(block, errors, bindingParser, expressionStart);\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,\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 string | null,\n    expression: bindingParser.parseBinding(\n        rawExpression, false, expressionParam.sourceSpan,\n        // Note: `lastIndexOf` here should be enough to know the start index of the expression,\n        // because we know that it'll be the last matching group. Ideally we could use the `d`\n        // flag on the regex and get the index from `match.indices`, but it's unclear if we can\n        // use it yet since it's a relatively new feature. See:\n        // https://github.com/tc39/proposal-regexp-match-indices\n        Math.max(0, expressionParam.expression.lastIndexOf(rawExpression)))\n  };\n\n  for (const param of secondaryParams) {\n    const trackMatch = param.expression.match(FOR_LOOP_TRACK_PATTERN);\n\n    // For now loops can only have a `track` parameter.\n    // We may want to rework this later if we add more.\n    if (trackMatch === null) {\n      errors.push(\n          new ParseError(param.sourceSpan, `Unrecognized loop paramater \"${param.expression}\"`));\n    } else 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 = trackMatch[1].trim();\n    }\n  }\n\n  return result;\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\n  if (primaryBlock.children.length > 0) {\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/** Parses a block parameter into a binding AST. */\nfunction parseBlockParameterToBinding(\n    ast: html.BlockParameter, bindingParser: BindingParser, start = 0): ASTWithSource {\n  return bindingParser.parseBinding(\n      ast.expression.slice(start), 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    primaryExpressionStart: number) {\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 =\n      parseBlockParameterToBinding(block.parameters[0], bindingParser, primaryExpressionStart);\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 (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"]}
|
|
@@ -85,6 +85,9 @@ function parsePlaceholderBlock(ast, visitor) {
|
|
|
85
85
|
let minimumTime = null;
|
|
86
86
|
for (const param of ast.parameters) {
|
|
87
87
|
if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {
|
|
88
|
+
if (minimumTime != null) {
|
|
89
|
+
throw new Error(`Placeholder block can only have one "minimum" parameter`);
|
|
90
|
+
}
|
|
88
91
|
const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
|
|
89
92
|
if (parsedTime === null) {
|
|
90
93
|
throw new Error(`Could not parse time value of parameter "minimum"`);
|
|
@@ -102,6 +105,9 @@ function parseLoadingBlock(ast, visitor) {
|
|
|
102
105
|
let minimumTime = null;
|
|
103
106
|
for (const param of ast.parameters) {
|
|
104
107
|
if (AFTER_PARAMETER_PATTERN.test(param.expression)) {
|
|
108
|
+
if (afterTime != null) {
|
|
109
|
+
throw new Error(`Loading block can only have one "after" parameter`);
|
|
110
|
+
}
|
|
105
111
|
const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
|
|
106
112
|
if (parsedTime === null) {
|
|
107
113
|
throw new Error(`Could not parse time value of parameter "after"`);
|
|
@@ -109,6 +115,9 @@ function parseLoadingBlock(ast, visitor) {
|
|
|
109
115
|
afterTime = parsedTime;
|
|
110
116
|
}
|
|
111
117
|
else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {
|
|
118
|
+
if (minimumTime != null) {
|
|
119
|
+
throw new Error(`Loading block can only have one "minimum" parameter`);
|
|
120
|
+
}
|
|
112
121
|
const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
|
|
113
122
|
if (parsedTime === null) {
|
|
114
123
|
throw new Error(`Could not parse time value of parameter "minimum"`);
|
|
@@ -128,24 +137,22 @@ function parseErrorBlock(ast, visitor) {
|
|
|
128
137
|
return new t.DeferredBlockError(html.visitAll(visitor, ast.children), ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);
|
|
129
138
|
}
|
|
130
139
|
function parsePrimaryTriggers(params, bindingParser, errors) {
|
|
131
|
-
const triggers =
|
|
132
|
-
const prefetchTriggers =
|
|
140
|
+
const triggers = {};
|
|
141
|
+
const prefetchTriggers = {};
|
|
133
142
|
for (const param of params) {
|
|
134
143
|
// The lexer ignores the leading spaces so we can assume
|
|
135
144
|
// that the expression starts with a keyword.
|
|
136
145
|
if (WHEN_PARAMETER_PATTERN.test(param.expression)) {
|
|
137
|
-
|
|
138
|
-
result !== null && triggers.push(result);
|
|
146
|
+
parseWhenTrigger(param, bindingParser, triggers, errors);
|
|
139
147
|
}
|
|
140
148
|
else if (ON_PARAMETER_PATTERN.test(param.expression)) {
|
|
141
|
-
|
|
149
|
+
parseOnTrigger(param, triggers, errors);
|
|
142
150
|
}
|
|
143
151
|
else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {
|
|
144
|
-
|
|
145
|
-
result !== null && prefetchTriggers.push(result);
|
|
152
|
+
parseWhenTrigger(param, bindingParser, prefetchTriggers, errors);
|
|
146
153
|
}
|
|
147
154
|
else if (PREFETCH_ON_PATTERN.test(param.expression)) {
|
|
148
|
-
|
|
155
|
+
parseOnTrigger(param, prefetchTriggers, errors);
|
|
149
156
|
}
|
|
150
157
|
else {
|
|
151
158
|
errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));
|
|
@@ -153,4 +160,4 @@ function parsePrimaryTriggers(params, bindingParser, errors) {
|
|
|
153
160
|
}
|
|
154
161
|
return { triggers, prefetchTriggers };
|
|
155
162
|
}
|
|
156
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_deferred_blocks.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_deferred_blocks.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,IAAI,MAAM,kBAAkB,CAAC;AACzC,OAAO,EAAC,UAAU,EAAC,MAAM,eAAe,CAAC;AAGzC,OAAO,KAAK,CAAC,MAAM,UAAU,CAAC;AAC9B,OAAO,EAAC,yBAAyB,EAAE,iBAAiB,EAAE,cAAc,EAAE,gBAAgB,EAAC,MAAM,wBAAwB,CAAC;AAEtH,qDAAqD;AACrD,MAAM,qBAAqB,GAAG,oBAAoB,CAAC;AAEnD,mDAAmD;AACnD,MAAM,mBAAmB,GAAG,kBAAkB,CAAC;AAE/C,4DAA4D;AAC5D,MAAM,yBAAyB,GAAG,YAAY,CAAC;AAE/C,0DAA0D;AAC1D,MAAM,uBAAuB,GAAG,UAAU,CAAC;AAE3C,yDAAyD;AACzD,MAAM,sBAAsB,GAAG,SAAS,CAAC;AAEzC,uDAAuD;AACvD,MAAM,oBAAoB,GAAG,OAAO,CAAC;AAErC,mDAAmD;AACnD,MAAM,CAAN,IAAY,0BAIX;AAJD,WAAY,0BAA0B;IACpC,yDAA2B,CAAA;IAC3B,iDAAmB,CAAA;IACnB,6CAAe,CAAA;AACjB,CAAC,EAJW,0BAA0B,KAA1B,0BAA0B,QAIrC;AAED,sDAAsD;AACtD,MAAM,UAAU,mBAAmB,CAC/B,GAAoB,EAAE,OAAqB,EAC3C,aAA4B;IAC9B,MAAM,MAAM,GAAiB,EAAE,CAAC;IAChC,MAAM,CAAC,YAAY,EAAE,GAAG,eAAe,CAAC,GAAG,GAAG,CAAC,MAAM,CAAC;IACtD,MAAM,EAAC,QAAQ,EAAE,gBAAgB,EAAC,GAC9B,oBAAoB,CAAC,YAAY,CAAC,UAAU,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;IACzE,MAAM,EAAC,WAAW,EAAE,OAAO,EAAE,KAAK,EAAC,GAAG,oBAAoB,CAAC,eAAe,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;IAE7F,OAAO;QACL,IAAI,EAAE,IAAI,CAAC,CAAC,aAAa,CACrB,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,YAAY,CAAC,QAAQ,CAAC,EAAE,QAAQ,EAAE,gBAAgB,EAAE,WAAW,EACtF,OAAO,EAAE,KAAK,EAAE,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,eAAe,EAAE,GAAG,CAAC,aAAa,CAAC;QAC3E,MAAM;KACP,CAAC;AACJ,CAAC;AAED,SAAS,oBAAoB,CAAC,MAAoB,EAAE,MAAoB,EAAE,OAAqB;IAC7F,IAAI,WAAW,GAAoC,IAAI,CAAC;IACxD,IAAI,OAAO,GAAgC,IAAI,CAAC;IAChD,IAAI,KAAK,GAA8B,IAAI,CAAC;IAE5C,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE;QAC1B,IAAI;YACF,QAAQ,KAAK,CAAC,IAAI,EAAE;gBAClB,KAAK,0BAA0B,CAAC,WAAW;oBACzC,IAAI,WAAW,KAAK,IAAI,EAAE;wBACxB,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CACtB,KAAK,CAAC,eAAe,EACrB,oCACI,0BAA0B,CAAC,WAAW,SAAS,CAAC,CAAC,CAAC;qBAC3D;yBAAM;wBACL,WAAW,GAAG,qBAAqB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;qBACrD;oBACD,MAAM;gBAER,KAAK,0BAA0B,CAAC,OAAO;oBACrC,IAAI,OAAO,KAAK,IAAI,EAAE;wBACpB,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CACtB,KAAK,CAAC,eAAe,EACrB,oCAAoC,0BAA0B,CAAC,OAAO,SAAS,CAAC,CAAC,CAAC;qBACvF;yBAAM;wBACL,OAAO,GAAG,iBAAiB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;qBAC7C;oBACD,MAAM;gBAER,KAAK,0BAA0B,CAAC,KAAK;oBACnC,IAAI,KAAK,KAAK,IAAI,EAAE;wBAClB,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CACtB,KAAK,CAAC,eAAe,EACrB,oCAAoC,0BAA0B,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC;qBACrF;yBAAM;wBACL,KAAK,GAAG,eAAe,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;qBACzC;oBACD,MAAM;gBAER;oBACE,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,KAAK,CAAC,eAAe,EAAE,uBAAuB,KAAK,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC;oBACzF,MAAM;aACT;SACF;QAAC,OAAO,CAAC,EAAE;YACV,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,KAAK,CAAC,eAAe,EAAG,CAAW,CAAC,OAAO,CAAC,CAAC,CAAC;SAC1E;KACF;IAED,OAAO,EAAC,WAAW,EAAE,OAAO,EAAE,KAAK,EAAC,CAAC;AACvC,CAAC;AAED,SAAS,qBAAqB,CAAC,GAAe,EAAE,OAAqB;IACnE,IAAI,WAAW,GAAgB,IAAI,CAAC;IAEpC,KAAK,MAAM,KAAK,IAAI,GAAG,CAAC,UAAU,EAAE;QAClC,IAAI,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YACpD,MAAM,UAAU,GACZ,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,yBAAyB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YAE3F,IAAI,UAAU,KAAK,IAAI,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;aACtE;YAED,WAAW,GAAG,UAAU,CAAC;SAC1B;aAAM;YACL,MAAM,IAAI,KAAK,CAAC,8BACZ,0BAA0B,CAAC,WAAW,aAAa,KAAK,CAAC,UAAU,GAAG,CAAC,CAAC;SAC7E;KACF;IAED,OAAO,IAAI,CAAC,CAAC,wBAAwB,CACjC,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,EAAE,WAAW,EAAE,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,eAAe,EACtF,GAAG,CAAC,aAAa,CAAC,CAAC;AACzB,CAAC;AAED,SAAS,iBAAiB,CAAC,GAAe,EAAE,OAAqB;IAC/D,IAAI,SAAS,GAAgB,IAAI,CAAC;IAClC,IAAI,WAAW,GAAgB,IAAI,CAAC;IAEpC,KAAK,MAAM,KAAK,IAAI,GAAG,CAAC,UAAU,EAAE;QAClC,IAAI,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YAClD,MAAM,UAAU,GACZ,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,yBAAyB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YAE3F,IAAI,UAAU,KAAK,IAAI,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;aACpE;YAED,SAAS,GAAG,UAAU,CAAC;SACxB;aAAM,IAAI,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YAC3D,MAAM,UAAU,GACZ,iBAAiB,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,yBAAyB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YAE3F,IAAI,UAAU,KAAK,IAAI,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;aACtE;YAED,WAAW,GAAG,UAAU,CAAC;SAC1B;aAAM;YACL,MAAM,IAAI,KAAK,CAAC,8BAA8B,0BAA0B,CAAC,OAAO,aAC5E,KAAK,CAAC,UAAU,GAAG,CAAC,CAAC;SAC1B;KACF;IAED,OAAO,IAAI,CAAC,CAAC,oBAAoB,CAC7B,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,EAAE,SAAS,EAAE,WAAW,EAAE,GAAG,CAAC,UAAU,EAC5E,GAAG,CAAC,eAAe,EAAE,GAAG,CAAC,aAAa,CAAC,CAAC;AAC9C,CAAC;AAGD,SAAS,eAAe,CAAC,GAAe,EAAE,OAAqB;IAC7D,IAAI,GAAG,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE;QAC7B,MAAM,IAAI,KAAK,CAAC,IAAI,0BAA0B,CAAC,KAAK,gCAAgC,CAAC,CAAC;KACvF;IAED,OAAO,IAAI,CAAC,CAAC,kBAAkB,CAC3B,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,eAAe,EAAE,GAAG,CAAC,aAAa,CAAC,CAAC;AACpG,CAAC;AAED,SAAS,oBAAoB,CACzB,MAA6B,EAAE,aAA4B,EAAE,MAAoB;IACnF,MAAM,QAAQ,GAAwB,EAAE,CAAC;IACzC,MAAM,gBAAgB,GAAwB,EAAE,CAAC;IAEjD,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE;QAC1B,wDAAwD;QACxD,6CAA6C;QAC7C,IAAI,sBAAsB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YACjD,MAAM,MAAM,GAAG,gBAAgB,CAAC,KAAK,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;YAC9D,MAAM,KAAK,IAAI,IAAI,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;SAC1C;aAAM,IAAI,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YACtD,QAAQ,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC;SACjD;aAAM,IAAI,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YACvD,MAAM,MAAM,GAAG,gBAAgB,CAAC,KAAK,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;YAC9D,MAAM,KAAK,IAAI,IAAI,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;SAClD;aAAM,IAAI,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YACrD,gBAAgB,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC;SACzD;aAAM;YACL,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,KAAK,CAAC,UAAU,EAAE,sBAAsB,CAAC,CAAC,CAAC;SACvE;KACF;IAED,OAAO,EAAC,QAAQ,EAAE,gBAAgB,EAAC,CAAC;AACtC,CAAC","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 * as html from '../ml_parser/ast';\nimport {ParseError} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\nimport {getTriggerParametersStart, parseDeferredTime, parseOnTrigger, parseWhenTrigger} from './r3_deferred_triggers';\n\n/** Pattern to identify a `prefetch when` trigger. */\nconst PREFETCH_WHEN_PATTERN = /^prefetch\\s+when\\s/;\n\n/** Pattern to identify a `prefetch on` trigger. */\nconst PREFETCH_ON_PATTERN = /^prefetch\\s+on\\s/;\n\n/** Pattern to identify a `minimum` parameter in a block. */\nconst MINIMUM_PARAMETER_PATTERN = /^minimum\\s/;\n\n/** Pattern to identify a `after` parameter in a block. */\nconst AFTER_PARAMETER_PATTERN = /^after\\s/;\n\n/** Pattern to identify a `when` parameter in a block. */\nconst WHEN_PARAMETER_PATTERN = /^when\\s/;\n\n/** Pattern to identify a `on` parameter in a block. */\nconst ON_PARAMETER_PATTERN = /^on\\s/;\n\n/** Possible types of secondary deferred blocks. */\nexport enum SecondaryDeferredBlockType {\n  PLACEHOLDER = 'placeholder',\n  LOADING = 'loading',\n  ERROR = 'error',\n}\n\n/** Creates a deferred block from an HTML AST node. */\nexport function createDeferredBlock(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.DeferredBlock, errors: ParseError[]} {\n  const errors: ParseError[] = [];\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const {triggers, prefetchTriggers} =\n      parsePrimaryTriggers(primaryBlock.parameters, bindingParser, errors);\n  const {placeholder, loading, error} = parseSecondaryBlocks(secondaryBlocks, errors, visitor);\n\n  return {\n    node: new t.DeferredBlock(\n        html.visitAll(visitor, primaryBlock.children), triggers, prefetchTriggers, placeholder,\n        loading, error, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors,\n  };\n}\n\nfunction parseSecondaryBlocks(blocks: html.Block[], errors: ParseError[], visitor: html.Visitor) {\n  let placeholder: t.DeferredBlockPlaceholder|null = null;\n  let loading: t.DeferredBlockLoading|null = null;\n  let error: t.DeferredBlockError|null = null;\n\n  for (const block of blocks) {\n    try {\n      switch (block.name) {\n        case SecondaryDeferredBlockType.PLACEHOLDER:\n          if (placeholder !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${\n                    SecondaryDeferredBlockType.PLACEHOLDER}\" block`));\n          } else {\n            placeholder = parsePlaceholderBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.LOADING:\n          if (loading !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.LOADING}\" block`));\n          } else {\n            loading = parseLoadingBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.ERROR:\n          if (error !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.ERROR}\" block`));\n          } else {\n            error = parseErrorBlock(block, visitor);\n          }\n          break;\n\n        default:\n          errors.push(new ParseError(block.startSourceSpan, `Unrecognized block \"${block.name}\"`));\n          break;\n      }\n    } catch (e) {\n      errors.push(new ParseError(block.startSourceSpan, (e as Error).message));\n    }\n  }\n\n  return {placeholder, loading, error};\n}\n\nfunction parsePlaceholderBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockPlaceholder {\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${\n          SecondaryDeferredBlockType.PLACEHOLDER}\" block: \"${param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockPlaceholder(\n      html.visitAll(visitor, ast.children), minimumTime, ast.sourceSpan, ast.startSourceSpan,\n      ast.endSourceSpan);\n}\n\nfunction parseLoadingBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockLoading {\n  let afterTime: number|null = null;\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (AFTER_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"after\"`);\n      }\n\n      afterTime = parsedTime;\n    } else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${SecondaryDeferredBlockType.LOADING}\" block: \"${\n          param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockLoading(\n      html.visitAll(visitor, ast.children), afterTime, minimumTime, ast.sourceSpan,\n      ast.startSourceSpan, ast.endSourceSpan);\n}\n\n\nfunction parseErrorBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockError {\n  if (ast.parameters.length > 0) {\n    throw new Error(`\"${SecondaryDeferredBlockType.ERROR}\" block cannot have parameters`);\n  }\n\n  return new t.DeferredBlockError(\n      html.visitAll(visitor, ast.children), ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);\n}\n\nfunction parsePrimaryTriggers(\n    params: html.BlockParameter[], bindingParser: BindingParser, errors: ParseError[]) {\n  const triggers: t.DeferredTrigger[] = [];\n  const prefetchTriggers: t.DeferredTrigger[] = [];\n\n  for (const param of params) {\n    // The lexer ignores the leading spaces so we can assume\n    // that the expression starts with a keyword.\n    if (WHEN_PARAMETER_PATTERN.test(param.expression)) {\n      const result = parseWhenTrigger(param, bindingParser, errors);\n      result !== null && triggers.push(result);\n    } else if (ON_PARAMETER_PATTERN.test(param.expression)) {\n      triggers.push(...parseOnTrigger(param, errors));\n    } else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {\n      const result = parseWhenTrigger(param, bindingParser, errors);\n      result !== null && prefetchTriggers.push(result);\n    } else if (PREFETCH_ON_PATTERN.test(param.expression)) {\n      prefetchTriggers.push(...parseOnTrigger(param, errors));\n    } else {\n      errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));\n    }\n  }\n\n  return {triggers, prefetchTriggers};\n}\n"]}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_deferred_blocks.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_deferred_blocks.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 * as html from '../ml_parser/ast';\nimport {ParseError} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\nimport {getTriggerParametersStart, parseDeferredTime, parseOnTrigger, parseWhenTrigger} from './r3_deferred_triggers';\n\n/** Pattern to identify a `prefetch when` trigger. */\nconst PREFETCH_WHEN_PATTERN = /^prefetch\\s+when\\s/;\n\n/** Pattern to identify a `prefetch on` trigger. */\nconst PREFETCH_ON_PATTERN = /^prefetch\\s+on\\s/;\n\n/** Pattern to identify a `minimum` parameter in a block. */\nconst MINIMUM_PARAMETER_PATTERN = /^minimum\\s/;\n\n/** Pattern to identify a `after` parameter in a block. */\nconst AFTER_PARAMETER_PATTERN = /^after\\s/;\n\n/** Pattern to identify a `when` parameter in a block. */\nconst WHEN_PARAMETER_PATTERN = /^when\\s/;\n\n/** Pattern to identify a `on` parameter in a block. */\nconst ON_PARAMETER_PATTERN = /^on\\s/;\n\n/** Possible types of secondary deferred blocks. */\nexport enum SecondaryDeferredBlockType {\n  PLACEHOLDER = 'placeholder',\n  LOADING = 'loading',\n  ERROR = 'error',\n}\n\n/** Creates a deferred block from an HTML AST node. */\nexport function createDeferredBlock(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.DeferredBlock, errors: ParseError[]} {\n  const errors: ParseError[] = [];\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const {triggers, prefetchTriggers} =\n      parsePrimaryTriggers(primaryBlock.parameters, bindingParser, errors);\n  const {placeholder, loading, error} = parseSecondaryBlocks(secondaryBlocks, errors, visitor);\n\n  return {\n    node: new t.DeferredBlock(\n        html.visitAll(visitor, primaryBlock.children), triggers, prefetchTriggers, placeholder,\n        loading, error, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors,\n  };\n}\n\nfunction parseSecondaryBlocks(blocks: html.Block[], errors: ParseError[], visitor: html.Visitor) {\n  let placeholder: t.DeferredBlockPlaceholder|null = null;\n  let loading: t.DeferredBlockLoading|null = null;\n  let error: t.DeferredBlockError|null = null;\n\n  for (const block of blocks) {\n    try {\n      switch (block.name) {\n        case SecondaryDeferredBlockType.PLACEHOLDER:\n          if (placeholder !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${\n                    SecondaryDeferredBlockType.PLACEHOLDER}\" block`));\n          } else {\n            placeholder = parsePlaceholderBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.LOADING:\n          if (loading !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.LOADING}\" block`));\n          } else {\n            loading = parseLoadingBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.ERROR:\n          if (error !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.ERROR}\" block`));\n          } else {\n            error = parseErrorBlock(block, visitor);\n          }\n          break;\n\n        default:\n          errors.push(new ParseError(block.startSourceSpan, `Unrecognized block \"${block.name}\"`));\n          break;\n      }\n    } catch (e) {\n      errors.push(new ParseError(block.startSourceSpan, (e as Error).message));\n    }\n  }\n\n  return {placeholder, loading, error};\n}\n\nfunction parsePlaceholderBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockPlaceholder {\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      if (minimumTime != null) {\n        throw new Error(`Placeholder block can only have one \"minimum\" parameter`);\n      }\n\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${\n          SecondaryDeferredBlockType.PLACEHOLDER}\" block: \"${param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockPlaceholder(\n      html.visitAll(visitor, ast.children), minimumTime, ast.sourceSpan, ast.startSourceSpan,\n      ast.endSourceSpan);\n}\n\nfunction parseLoadingBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockLoading {\n  let afterTime: number|null = null;\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (AFTER_PARAMETER_PATTERN.test(param.expression)) {\n      if (afterTime != null) {\n        throw new Error(`Loading block can only have one \"after\" parameter`);\n      }\n\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"after\"`);\n      }\n\n      afterTime = parsedTime;\n    } else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      if (minimumTime != null) {\n        throw new Error(`Loading block can only have one \"minimum\" parameter`);\n      }\n\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${SecondaryDeferredBlockType.LOADING}\" block: \"${\n          param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockLoading(\n      html.visitAll(visitor, ast.children), afterTime, minimumTime, ast.sourceSpan,\n      ast.startSourceSpan, ast.endSourceSpan);\n}\n\n\nfunction parseErrorBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockError {\n  if (ast.parameters.length > 0) {\n    throw new Error(`\"${SecondaryDeferredBlockType.ERROR}\" block cannot have parameters`);\n  }\n\n  return new t.DeferredBlockError(\n      html.visitAll(visitor, ast.children), ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);\n}\n\nfunction parsePrimaryTriggers(\n    params: html.BlockParameter[], bindingParser: BindingParser, errors: ParseError[]) {\n  const triggers: t.DeferredBlockTriggers = {};\n  const prefetchTriggers: t.DeferredBlockTriggers = {};\n\n  for (const param of params) {\n    // The lexer ignores the leading spaces so we can assume\n    // that the expression starts with a keyword.\n    if (WHEN_PARAMETER_PATTERN.test(param.expression)) {\n      parseWhenTrigger(param, bindingParser, triggers, errors);\n    } else if (ON_PARAMETER_PATTERN.test(param.expression)) {\n      parseOnTrigger(param, triggers, errors);\n    } else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {\n      parseWhenTrigger(param, bindingParser, prefetchTriggers, errors);\n    } else if (PREFETCH_ON_PATTERN.test(param.expression)) {\n      parseOnTrigger(param, prefetchTriggers, errors);\n    } else {\n      errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));\n    }\n  }\n\n  return {triggers, prefetchTriggers};\n}\n"]}
|