@angular/compiler 17.0.0-rc.0 → 17.0.0-rc.2

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 (101) hide show
  1. package/esm2022/src/ml_parser/ast.mjs +3 -2
  2. package/esm2022/src/ml_parser/html_whitespaces.mjs +2 -2
  3. package/esm2022/src/ml_parser/icu_ast_expander.mjs +2 -2
  4. package/esm2022/src/ml_parser/lexer.mjs +2 -13
  5. package/esm2022/src/ml_parser/parser.mjs +3 -3
  6. package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
  7. package/esm2022/src/render3/partial/directive.mjs +1 -1
  8. package/esm2022/src/render3/partial/factory.mjs +1 -1
  9. package/esm2022/src/render3/partial/injectable.mjs +1 -1
  10. package/esm2022/src/render3/partial/injector.mjs +1 -1
  11. package/esm2022/src/render3/partial/ng_module.mjs +1 -1
  12. package/esm2022/src/render3/partial/pipe.mjs +1 -1
  13. package/esm2022/src/render3/r3_ast.mjs +57 -62
  14. package/esm2022/src/render3/r3_control_flow.mjs +18 -14
  15. package/esm2022/src/render3/r3_deferred_blocks.mjs +7 -8
  16. package/esm2022/src/render3/r3_deferred_triggers.mjs +46 -27
  17. package/esm2022/src/render3/r3_template_transform.mjs +2 -2
  18. package/esm2022/src/render3/view/t2_api.mjs +1 -1
  19. package/esm2022/src/render3/view/t2_binder.mjs +22 -10
  20. package/esm2022/src/render3/view/template.mjs +54 -5
  21. package/esm2022/src/template/pipeline/ir/index.mjs +2 -1
  22. package/esm2022/src/template/pipeline/ir/src/enums.mjs +124 -38
  23. package/esm2022/src/template/pipeline/ir/src/expression.mjs +127 -64
  24. package/esm2022/src/template/pipeline/ir/src/handle.mjs +13 -0
  25. package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +71 -50
  26. package/esm2022/src/template/pipeline/ir/src/ops/shared.mjs +3 -2
  27. package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +19 -8
  28. package/esm2022/src/template/pipeline/ir/src/traits.mjs +1 -32
  29. package/esm2022/src/template/pipeline/ir/src/variable.mjs +1 -1
  30. package/esm2022/src/template/pipeline/src/compilation.mjs +6 -1
  31. package/esm2022/src/template/pipeline/src/conversion.mjs +2 -2
  32. package/esm2022/src/template/pipeline/src/emit.mjs +118 -99
  33. package/esm2022/src/template/pipeline/src/ingest.mjs +204 -50
  34. package/esm2022/src/template/pipeline/src/instruction.mjs +57 -12
  35. package/esm2022/src/template/pipeline/src/phases/any_cast.mjs +4 -3
  36. package/esm2022/src/template/pipeline/src/phases/apply_i18n_expressions.mjs +3 -3
  37. package/esm2022/src/template/pipeline/src/phases/assign_i18n_slot_dependencies.mjs +2 -2
  38. package/esm2022/src/template/pipeline/src/phases/attribute_extraction.mjs +4 -4
  39. package/esm2022/src/template/pipeline/src/phases/binding_specialization.mjs +2 -2
  40. package/esm2022/src/template/pipeline/src/phases/chaining.mjs +2 -2
  41. package/esm2022/src/template/pipeline/src/phases/collapse_singleton_interpolations.mjs +31 -0
  42. package/esm2022/src/template/pipeline/src/phases/conditionals.mjs +9 -6
  43. package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +2 -2
  44. package/esm2022/src/template/pipeline/src/phases/defer_configs.mjs +30 -0
  45. package/esm2022/src/template/pipeline/src/phases/defer_resolve_targets.mjs +93 -0
  46. package/esm2022/src/template/pipeline/src/phases/empty_elements.mjs +4 -4
  47. package/esm2022/src/template/pipeline/src/phases/expand_safe_reads.mjs +7 -4
  48. package/esm2022/src/template/pipeline/src/phases/format_i18n_params.mjs +114 -0
  49. package/esm2022/src/template/pipeline/src/phases/generate_advance.mjs +4 -4
  50. package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +2 -2
  51. package/esm2022/src/template/pipeline/src/phases/generate_variables.mjs +13 -6
  52. package/esm2022/src/template/pipeline/src/phases/has_const_expression_collection.mjs +28 -0
  53. package/esm2022/src/template/pipeline/src/phases/host_style_property_parsing.mjs +8 -2
  54. package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +129 -7
  55. package/esm2022/src/template/pipeline/src/phases/i18n_message_extraction.mjs +4 -105
  56. package/esm2022/src/template/pipeline/src/phases/i18n_text_extraction.mjs +9 -6
  57. package/esm2022/src/template/pipeline/src/phases/icu_extraction.mjs +8 -5
  58. package/esm2022/src/template/pipeline/src/phases/local_refs.mjs +2 -2
  59. package/esm2022/src/template/pipeline/src/phases/namespace.mjs +2 -2
  60. package/esm2022/src/template/pipeline/src/phases/naming.mjs +27 -11
  61. package/esm2022/src/template/pipeline/src/phases/next_context_merging.mjs +2 -2
  62. package/esm2022/src/template/pipeline/src/phases/ng_container.mjs +2 -2
  63. package/esm2022/src/template/pipeline/src/phases/nonbindable.mjs +2 -2
  64. package/esm2022/src/template/pipeline/src/phases/nullish_coalescing.mjs +8 -2
  65. package/esm2022/src/template/pipeline/src/phases/ordering.mjs +46 -32
  66. package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +3 -3
  67. package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +9 -9
  68. package/esm2022/src/template/pipeline/src/phases/pipe_creation.mjs +23 -6
  69. package/esm2022/src/template/pipeline/src/phases/pipe_variadic.mjs +7 -3
  70. package/esm2022/src/template/pipeline/src/phases/propagate_i18n_blocks.mjs +2 -2
  71. package/esm2022/src/template/pipeline/src/phases/propagate_i18n_placeholders.mjs +61 -0
  72. package/esm2022/src/template/pipeline/src/phases/pure_function_extraction.mjs +3 -2
  73. package/esm2022/src/template/pipeline/src/phases/pure_literal_structures.mjs +4 -4
  74. package/esm2022/src/template/pipeline/src/phases/reify.mjs +77 -26
  75. package/esm2022/src/template/pipeline/src/phases/remove_empty_bindings.mjs +5 -2
  76. package/esm2022/src/template/pipeline/src/phases/repeater_derived_vars.mjs +45 -0
  77. package/esm2022/src/template/pipeline/src/phases/resolve_contexts.mjs +7 -3
  78. package/esm2022/src/template/pipeline/src/phases/resolve_dollar_event.mjs +5 -5
  79. package/esm2022/src/template/pipeline/src/phases/resolve_i18n_element_placeholders.mjs +109 -0
  80. package/esm2022/src/template/pipeline/src/phases/resolve_i18n_expression_placeholders.mjs +58 -0
  81. package/esm2022/src/template/pipeline/src/phases/resolve_names.mjs +4 -3
  82. package/esm2022/src/template/pipeline/src/phases/resolve_sanitizers.mjs +5 -5
  83. package/esm2022/src/template/pipeline/src/phases/save_restore_view.mjs +16 -11
  84. package/esm2022/src/template/pipeline/src/phases/slot_allocation.mjs +5 -30
  85. package/esm2022/src/template/pipeline/src/phases/style_binding_specialization.mjs +3 -3
  86. package/esm2022/src/template/pipeline/src/phases/temporary_variables.mjs +3 -3
  87. package/esm2022/src/template/pipeline/src/phases/track_fn_generation.mjs +49 -0
  88. package/esm2022/src/template/pipeline/src/phases/track_fn_optimization.mjs +81 -0
  89. package/esm2022/src/template/pipeline/src/phases/track_variables.mjs +36 -0
  90. package/esm2022/src/template/pipeline/src/phases/var_counting.mjs +37 -4
  91. package/esm2022/src/template/pipeline/src/phases/variable_optimization.mjs +47 -7
  92. package/esm2022/src/template/pipeline/src/phases/wrap_icus.mjs +2 -2
  93. package/esm2022/src/template/pipeline/src/util/elements.mjs +6 -6
  94. package/esm2022/src/version.mjs +1 -1
  95. package/fesm2022/compiler.mjs +2004 -1048
  96. package/fesm2022/compiler.mjs.map +1 -1
  97. package/index.d.ts +45 -60
  98. package/package.json +3 -3
  99. package/esm2022/src/template/pipeline/src/phases/align_pipe_variadic_var_offset.mjs +0 -42
  100. package/esm2022/src/template/pipeline/src/phases/has_const_trait_collection.mjs +0 -34
  101. package/esm2022/src/template/pipeline/src/phases/resolve_i18n_placeholders.mjs +0 -288
@@ -1,288 +0,0 @@
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 o from '../../../../output/output_ast';
9
- import * as ir from '../../ir';
10
- /**
11
- * The escape sequence used indicate message param values.
12
- */
13
- const ESCAPE = '\uFFFD';
14
- /**
15
- * Marker used to indicate an element tag.
16
- */
17
- const ELEMENT_MARKER = '#';
18
- /**
19
- * Marker used to indicate a template tag.
20
- */
21
- const TEMPLATE_MARKER = '*';
22
- /**
23
- * Marker used to indicate closing of an element or template tag.
24
- */
25
- const TAG_CLOSE_MARKER = '/';
26
- /**
27
- * Marker used to indicate the sub-template context.
28
- */
29
- const CONTEXT_MARKER = ':';
30
- /**
31
- * Marker used to indicate the start of a list of values.
32
- */
33
- const LIST_START_MARKER = '[';
34
- /**
35
- * Marker used to indicate the end of a list of values.
36
- */
37
- const LIST_END_MARKER = ']';
38
- /**
39
- * Delimiter used to separate multiple values in a list.
40
- */
41
- const LIST_DELIMITER = '|';
42
- /**
43
- * Flags that describe what an i18n param value. These determine how the value is serialized into
44
- * the final map.
45
- */
46
- var I18nParamValueFlags;
47
- (function (I18nParamValueFlags) {
48
- I18nParamValueFlags[I18nParamValueFlags["None"] = 0] = "None";
49
- /**
50
- * This value represtents an element tag.
51
- */
52
- I18nParamValueFlags[I18nParamValueFlags["ElementTag"] = 1] = "ElementTag";
53
- /**
54
- * This value represents a template tag.
55
- */
56
- I18nParamValueFlags[I18nParamValueFlags["TemplateTag"] = 2] = "TemplateTag";
57
- /**
58
- * This value represents the opening of a tag.
59
- */
60
- I18nParamValueFlags[I18nParamValueFlags["OpenTag"] = 4] = "OpenTag";
61
- /**
62
- * This value represents the closing of a tag.
63
- */
64
- I18nParamValueFlags[I18nParamValueFlags["CloseTag"] = 8] = "CloseTag";
65
- })(I18nParamValueFlags || (I18nParamValueFlags = {}));
66
- /**
67
- * Represents the complete i18n params map for an i18n op.
68
- */
69
- class I18nPlaceholderParams {
70
- constructor() {
71
- this.values = new Map();
72
- }
73
- /**
74
- * Adds a new value to the params map.
75
- */
76
- addValue(placeholder, value, subTemplateIndex, resolutionTime, flags) {
77
- const placeholderValues = this.values.get(placeholder) ?? [];
78
- placeholderValues.push({ value, subTemplateIndex, resolutionTime, flags });
79
- this.values.set(placeholder, placeholderValues);
80
- }
81
- /**
82
- * Saves the params map, in serialized form, into the given i18n op.
83
- */
84
- saveToOp(op) {
85
- for (const [placeholder, placeholderValues] of this.values) {
86
- // We need to run post-processing for any 1i8n ops that contain parameters with more than
87
- // one value, even if there are no parameters resolved at post-processing time.
88
- const creationValues = placeholderValues.filter(({ resolutionTime }) => resolutionTime === ir.I18nParamResolutionTime.Creation);
89
- if (creationValues.length > 1) {
90
- op.needsPostprocessing = true;
91
- }
92
- // Save creation time params to op.
93
- const serializedCreationValues = this.serializeValues(creationValues);
94
- if (serializedCreationValues !== null) {
95
- op.params.set(placeholder, o.literal(serializedCreationValues));
96
- }
97
- // Save post-processing time params to op.
98
- const serializedPostprocessingValues = this.serializeValues(placeholderValues.filter(({ resolutionTime }) => resolutionTime === ir.I18nParamResolutionTime.Postproccessing));
99
- if (serializedPostprocessingValues !== null) {
100
- op.needsPostprocessing = true;
101
- op.postprocessingParams.set(placeholder, o.literal(serializedPostprocessingValues));
102
- }
103
- }
104
- }
105
- /**
106
- * Merges another param map into this one.
107
- */
108
- merge(other) {
109
- for (const [placeholder, otherValues] of other.values) {
110
- const currentValues = this.values.get(placeholder) || [];
111
- // Child element close tag params should be prepended to maintain the same order as
112
- // TemplateDefinitionBuilder.
113
- const flags = otherValues[0].flags;
114
- if ((flags & I18nParamValueFlags.CloseTag) && !(flags & I18nParamValueFlags.OpenTag)) {
115
- this.values.set(placeholder, [...otherValues, ...currentValues]);
116
- }
117
- else {
118
- this.values.set(placeholder, [...currentValues, ...otherValues]);
119
- }
120
- }
121
- }
122
- /**
123
- * Serializes a list of i18n placeholder values.
124
- */
125
- serializeValues(values) {
126
- if (values.length === 0) {
127
- return null;
128
- }
129
- const serializedValues = values.map(value => this.serializeValue(value));
130
- return serializedValues.length === 1 ?
131
- serializedValues[0] :
132
- `${LIST_START_MARKER}${serializedValues.join(LIST_DELIMITER)}${LIST_END_MARKER}`;
133
- }
134
- /**
135
- * Serializes a single i18n placeholder value.
136
- */
137
- serializeValue(value) {
138
- let tagMarker = '';
139
- let closeMarker = '';
140
- if (value.flags & I18nParamValueFlags.ElementTag) {
141
- tagMarker = ELEMENT_MARKER;
142
- }
143
- else if (value.flags & I18nParamValueFlags.TemplateTag) {
144
- tagMarker = TEMPLATE_MARKER;
145
- }
146
- if (tagMarker !== '') {
147
- closeMarker = value.flags & I18nParamValueFlags.CloseTag ? TAG_CLOSE_MARKER : '';
148
- }
149
- const context = value.subTemplateIndex === null ? '' : `${CONTEXT_MARKER}${value.subTemplateIndex}`;
150
- // Self-closing tags use a special form that concatenates the start and close tag values.
151
- if ((value.flags & I18nParamValueFlags.OpenTag) &&
152
- (value.flags & I18nParamValueFlags.CloseTag)) {
153
- return `${ESCAPE}${tagMarker}${value.value}${context}${ESCAPE}${ESCAPE}${closeMarker}${tagMarker}${value.value}${context}${ESCAPE}`;
154
- }
155
- return `${ESCAPE}${closeMarker}${tagMarker}${value.value}${context}${ESCAPE}`;
156
- }
157
- }
158
- /**
159
- * Resolve the placeholders in i18n messages.
160
- */
161
- export function phaseResolveI18nPlaceholders(job) {
162
- const params = new Map();
163
- const i18nOps = new Map();
164
- resolvePlaceholders(job, params, i18nOps);
165
- propagatePlaceholders(params, i18nOps);
166
- // After colleccting all params, save them to the i18n ops.
167
- for (const [xref, i18nOpParams] of params) {
168
- i18nOpParams.saveToOp(i18nOps.get(xref));
169
- }
170
- // Validate the root i18n ops have all placeholders filled in.
171
- for (const op of i18nOps.values()) {
172
- if (op.xref === op.root) {
173
- for (const placeholder in op.message.placeholders) {
174
- if (!op.params.has(placeholder) && !op.postprocessingParams.has(placeholder)) {
175
- throw Error(`Failed to resolve i18n placeholder: ${placeholder}`);
176
- }
177
- }
178
- }
179
- }
180
- }
181
- /**
182
- * Resolve placeholders for each i18n op.
183
- */
184
- function resolvePlaceholders(job, params, i18nOps) {
185
- for (const unit of job.units) {
186
- const elements = new Map();
187
- let currentI18nOp = null;
188
- // Record slots for tag name placeholders.
189
- for (const op of unit.create) {
190
- switch (op.kind) {
191
- case ir.OpKind.I18nStart:
192
- i18nOps.set(op.xref, op);
193
- currentI18nOp = op.kind === ir.OpKind.I18nStart ? op : null;
194
- break;
195
- case ir.OpKind.I18nEnd:
196
- currentI18nOp = null;
197
- break;
198
- case ir.OpKind.ElementStart:
199
- // For elements with i18n placeholders, record its slot value in the params map under the
200
- // corresponding tag start placeholder.
201
- if (op.i18nPlaceholder !== undefined) {
202
- if (currentI18nOp === null) {
203
- throw Error('i18n tag placeholder should only occur inside an i18n block');
204
- }
205
- elements.set(op.xref, op);
206
- const { startName, closeName } = op.i18nPlaceholder;
207
- let flags = I18nParamValueFlags.ElementTag | I18nParamValueFlags.OpenTag;
208
- // For self-closing tags, there is no close tag placeholder. Instead, the start tag
209
- // placeholder accounts for the start and close of the element.
210
- if (closeName === '') {
211
- flags |= I18nParamValueFlags.CloseTag;
212
- }
213
- addParam(params, currentI18nOp, startName, op.slot, currentI18nOp.subTemplateIndex, ir.I18nParamResolutionTime.Creation, flags);
214
- }
215
- break;
216
- case ir.OpKind.ElementEnd:
217
- const startOp = elements.get(op.xref);
218
- if (startOp && startOp.i18nPlaceholder !== undefined) {
219
- if (currentI18nOp === null) {
220
- throw Error('i18n tag placeholder should only occur inside an i18n block');
221
- }
222
- const { closeName } = startOp.i18nPlaceholder;
223
- // Self-closing tags don't have a closing tag placeholder.
224
- if (closeName !== '') {
225
- addParam(params, currentI18nOp, closeName, startOp.slot, currentI18nOp.subTemplateIndex, ir.I18nParamResolutionTime.Creation, I18nParamValueFlags.ElementTag | I18nParamValueFlags.CloseTag);
226
- }
227
- }
228
- break;
229
- case ir.OpKind.Template:
230
- if (op.i18nPlaceholder !== undefined) {
231
- if (currentI18nOp === null) {
232
- throw Error('i18n tag placeholder should only occur inside an i18n block');
233
- }
234
- const subTemplateIndex = getSubTemplateIndexForTemplateTag(job, currentI18nOp, op);
235
- addParam(params, currentI18nOp, op.i18nPlaceholder.startName, op.slot, subTemplateIndex, ir.I18nParamResolutionTime.Creation, I18nParamValueFlags.TemplateTag);
236
- addParam(params, currentI18nOp, op.i18nPlaceholder.closeName, op.slot, subTemplateIndex, ir.I18nParamResolutionTime.Creation, I18nParamValueFlags.TemplateTag | I18nParamValueFlags.CloseTag);
237
- }
238
- break;
239
- }
240
- }
241
- // Fill in values for each of the i18n expression placeholders.
242
- const i18nBlockPlaceholderIndices = new Map();
243
- for (const op of unit.update) {
244
- if (op.kind === ir.OpKind.I18nExpression) {
245
- const i18nOp = i18nOps.get(op.owner);
246
- let index = i18nBlockPlaceholderIndices.get(op.owner) || 0;
247
- if (!i18nOp) {
248
- throw Error('Cannot find corresponding i18nStart for i18nExpr');
249
- }
250
- addParam(params, i18nOp, op.i18nPlaceholder, index++, i18nOp.subTemplateIndex, op.resolutionTime);
251
- i18nBlockPlaceholderIndices.set(op.owner, index);
252
- }
253
- }
254
- }
255
- }
256
- /**
257
- * Add a param to the params map for the given i18n op.
258
- */
259
- function addParam(params, i18nOp, placeholder, value, subTemplateIndex, resolutionTime, flags = I18nParamValueFlags.None) {
260
- const i18nOpParams = params.get(i18nOp.xref) || new I18nPlaceholderParams();
261
- i18nOpParams.addValue(placeholder, value, subTemplateIndex, resolutionTime, flags);
262
- params.set(i18nOp.xref, i18nOpParams);
263
- }
264
- /**
265
- * Get the subTemplateIndex for the given template op. For template ops, use the subTemplateIndex of
266
- * the child i18n block inside the template.
267
- */
268
- function getSubTemplateIndexForTemplateTag(job, i18nOp, op) {
269
- for (const childOp of job.views.get(op.xref).create) {
270
- if (childOp.kind === ir.OpKind.I18nStart) {
271
- return childOp.subTemplateIndex;
272
- }
273
- }
274
- return i18nOp.subTemplateIndex;
275
- }
276
- /**
277
- * Propagate placeholders up to their root i18n op.
278
- */
279
- function propagatePlaceholders(params, i18nOps) {
280
- for (const [xref, opParams] of params) {
281
- const op = i18nOps.get(xref);
282
- if (op.xref !== op.root) {
283
- const rootParams = params.get(op.root) || new I18nPlaceholderParams();
284
- rootParams.merge(opParams);
285
- }
286
- }
287
- }
288
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolve_i18n_placeholders.js","sourceRoot":"","sources":["../../../../../../../../../../packages/compiler/src/template/pipeline/src/phases/resolve_i18n_placeholders.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 o from '../../../../output/output_ast';\nimport * as ir from '../../ir';\nimport {ComponentCompilationJob} from '../compilation';\n\n/**\n * The escape sequence used indicate message param values.\n */\nconst ESCAPE = '\\uFFFD';\n\n/**\n * Marker used to indicate an element tag.\n */\nconst ELEMENT_MARKER = '#';\n\n/**\n * Marker used to indicate a template tag.\n */\nconst TEMPLATE_MARKER = '*';\n\n/**\n * Marker used to indicate closing of an element or template tag.\n */\nconst TAG_CLOSE_MARKER = '/';\n\n/**\n * Marker used to indicate the sub-template context.\n */\nconst CONTEXT_MARKER = ':';\n\n/**\n * Marker used to indicate the start of a list of values.\n */\nconst LIST_START_MARKER = '[';\n\n/**\n * Marker used to indicate the end of a list of values.\n */\nconst LIST_END_MARKER = ']';\n\n/**\n * Delimiter used to separate multiple values in a list.\n */\nconst LIST_DELIMITER = '|';\n\n/**\n * Flags that describe what an i18n param value. These determine how the value is serialized into\n * the final map.\n */\nenum I18nParamValueFlags {\n  None = 0b0000,\n\n  /**\n   *  This value represtents an element tag.\n   */\n  ElementTag = 0b001,\n\n  /**\n   * This value represents a template tag.\n   */\n  TemplateTag = 0b0010,\n\n  /**\n   * This value represents the opening of a tag.\n   */\n  OpenTag = 0b0100,\n\n  /**\n   * This value represents the closing of a tag.\n   */\n  CloseTag = 0b1000,\n}\n\n/**\n * Represents a single placeholder value in the i18n params map. The map may contain multiple\n * I18nPlaceholderValue per placeholder.\n */\ninterface I18nPlaceholderValue {\n  /**\n   * The value.\n   */\n  value: string|number;\n\n  /**\n   * The sub-template index associated with the value.\n   */\n  subTemplateIndex: number|null;\n\n  /**\n   * Flags associated with the value.\n   */\n  flags: I18nParamValueFlags;\n\n  /**\n   * The time when the placeholder value is resolved.\n   */\n  resolutionTime: ir.I18nParamResolutionTime;\n}\n\n/**\n * Represents the complete i18n params map for an i18n op.\n */\nclass I18nPlaceholderParams {\n  values = new Map<string, I18nPlaceholderValue[]>();\n\n  /**\n   * Adds a new value to the params map.\n   */\n  addValue(\n      placeholder: string, value: string|number, subTemplateIndex: number|null,\n      resolutionTime: ir.I18nParamResolutionTime, flags: I18nParamValueFlags) {\n    const placeholderValues = this.values.get(placeholder) ?? [];\n    placeholderValues.push({value, subTemplateIndex, resolutionTime, flags});\n    this.values.set(placeholder, placeholderValues);\n  }\n\n  /**\n   * Saves the params map, in serialized form, into the given i18n op.\n   */\n  saveToOp(op: ir.I18nStartOp) {\n    for (const [placeholder, placeholderValues] of this.values) {\n      // We need to run post-processing for any 1i8n ops that contain parameters with more than\n      // one value, even if there are no parameters resolved at post-processing time.\n      const creationValues = placeholderValues.filter(\n          ({resolutionTime}) => resolutionTime === ir.I18nParamResolutionTime.Creation);\n      if (creationValues.length > 1) {\n        op.needsPostprocessing = true;\n      }\n\n      // Save creation time params to op.\n      const serializedCreationValues = this.serializeValues(creationValues);\n      if (serializedCreationValues !== null) {\n        op.params.set(placeholder, o.literal(serializedCreationValues));\n      }\n\n      // Save post-processing time params to op.\n      const serializedPostprocessingValues = this.serializeValues(placeholderValues.filter(\n          ({resolutionTime}) => resolutionTime === ir.I18nParamResolutionTime.Postproccessing));\n      if (serializedPostprocessingValues !== null) {\n        op.needsPostprocessing = true;\n        op.postprocessingParams.set(placeholder, o.literal(serializedPostprocessingValues));\n      }\n    }\n  }\n\n  /**\n   * Merges another param map into this one.\n   */\n  merge(other: I18nPlaceholderParams) {\n    for (const [placeholder, otherValues] of other.values) {\n      const currentValues = this.values.get(placeholder) || [];\n      // Child element close tag params should be prepended to maintain the same order as\n      // TemplateDefinitionBuilder.\n      const flags = otherValues[0]!.flags;\n      if ((flags & I18nParamValueFlags.CloseTag) && !(flags & I18nParamValueFlags.OpenTag)) {\n        this.values.set(placeholder, [...otherValues, ...currentValues]);\n      } else {\n        this.values.set(placeholder, [...currentValues, ...otherValues]);\n      }\n    }\n  }\n\n  /**\n   * Serializes a list of i18n placeholder values.\n   */\n  private serializeValues(values: I18nPlaceholderValue[]) {\n    if (values.length === 0) {\n      return null;\n    }\n    const serializedValues = values.map(value => this.serializeValue(value));\n    return serializedValues.length === 1 ?\n        serializedValues[0] :\n        `${LIST_START_MARKER}${serializedValues.join(LIST_DELIMITER)}${LIST_END_MARKER}`;\n  }\n\n  /**\n   * Serializes a single i18n placeholder value.\n   */\n  private serializeValue(value: I18nPlaceholderValue) {\n    let tagMarker = '';\n    let closeMarker = '';\n    if (value.flags & I18nParamValueFlags.ElementTag) {\n      tagMarker = ELEMENT_MARKER;\n    } else if (value.flags & I18nParamValueFlags.TemplateTag) {\n      tagMarker = TEMPLATE_MARKER;\n    }\n    if (tagMarker !== '') {\n      closeMarker = value.flags & I18nParamValueFlags.CloseTag ? TAG_CLOSE_MARKER : '';\n    }\n    const context =\n        value.subTemplateIndex === null ? '' : `${CONTEXT_MARKER}${value.subTemplateIndex}`;\n    // Self-closing tags use a special form that concatenates the start and close tag values.\n    if ((value.flags & I18nParamValueFlags.OpenTag) &&\n        (value.flags & I18nParamValueFlags.CloseTag)) {\n      return `${ESCAPE}${tagMarker}${value.value}${context}${ESCAPE}${ESCAPE}${closeMarker}${\n          tagMarker}${value.value}${context}${ESCAPE}`;\n    }\n    return `${ESCAPE}${closeMarker}${tagMarker}${value.value}${context}${ESCAPE}`;\n  }\n}\n\n/**\n * Resolve the placeholders in i18n messages.\n */\nexport function phaseResolveI18nPlaceholders(job: ComponentCompilationJob) {\n  const params = new Map<ir.XrefId, I18nPlaceholderParams>();\n  const i18nOps = new Map<ir.XrefId, ir.I18nStartOp>();\n\n  resolvePlaceholders(job, params, i18nOps);\n  propagatePlaceholders(params, i18nOps);\n\n  // After colleccting all params, save them to the i18n ops.\n  for (const [xref, i18nOpParams] of params) {\n    i18nOpParams.saveToOp(i18nOps.get(xref)!);\n  }\n\n  // Validate the root i18n ops have all placeholders filled in.\n  for (const op of i18nOps.values()) {\n    if (op.xref === op.root) {\n      for (const placeholder in op.message.placeholders) {\n        if (!op.params.has(placeholder) && !op.postprocessingParams.has(placeholder)) {\n          throw Error(`Failed to resolve i18n placeholder: ${placeholder}`);\n        }\n      }\n    }\n  }\n}\n\n/**\n * Resolve placeholders for each i18n op.\n */\nfunction resolvePlaceholders(\n    job: ComponentCompilationJob, params: Map<ir.XrefId, I18nPlaceholderParams>,\n    i18nOps: Map<ir.XrefId, ir.I18nStartOp>) {\n  for (const unit of job.units) {\n    const elements = new Map<ir.XrefId, ir.ElementStartOp>();\n    let currentI18nOp: ir.I18nStartOp|null = null;\n\n    // Record slots for tag name placeholders.\n    for (const op of unit.create) {\n      switch (op.kind) {\n        case ir.OpKind.I18nStart:\n          i18nOps.set(op.xref, op);\n          currentI18nOp = op.kind === ir.OpKind.I18nStart ? op : null;\n          break;\n        case ir.OpKind.I18nEnd:\n          currentI18nOp = null;\n          break;\n        case ir.OpKind.ElementStart:\n          // For elements with i18n placeholders, record its slot value in the params map under the\n          // corresponding tag start placeholder.\n          if (op.i18nPlaceholder !== undefined) {\n            if (currentI18nOp === null) {\n              throw Error('i18n tag placeholder should only occur inside an i18n block');\n            }\n            elements.set(op.xref, op);\n            const {startName, closeName} = op.i18nPlaceholder;\n            let flags = I18nParamValueFlags.ElementTag | I18nParamValueFlags.OpenTag;\n            // For self-closing tags, there is no close tag placeholder. Instead, the start tag\n            // placeholder accounts for the start and close of the element.\n            if (closeName === '') {\n              flags |= I18nParamValueFlags.CloseTag;\n            }\n            addParam(\n                params, currentI18nOp, startName, op.slot!, currentI18nOp.subTemplateIndex,\n                ir.I18nParamResolutionTime.Creation, flags);\n          }\n          break;\n        case ir.OpKind.ElementEnd:\n          const startOp = elements.get(op.xref);\n          if (startOp && startOp.i18nPlaceholder !== undefined) {\n            if (currentI18nOp === null) {\n              throw Error('i18n tag placeholder should only occur inside an i18n block');\n            }\n            const {closeName} = startOp.i18nPlaceholder;\n            // Self-closing tags don't have a closing tag placeholder.\n            if (closeName !== '') {\n              addParam(\n                  params, currentI18nOp, closeName, startOp.slot!, currentI18nOp.subTemplateIndex,\n                  ir.I18nParamResolutionTime.Creation,\n                  I18nParamValueFlags.ElementTag | I18nParamValueFlags.CloseTag);\n            }\n          }\n          break;\n        case ir.OpKind.Template:\n          if (op.i18nPlaceholder !== undefined) {\n            if (currentI18nOp === null) {\n              throw Error('i18n tag placeholder should only occur inside an i18n block');\n            }\n            const subTemplateIndex = getSubTemplateIndexForTemplateTag(job, currentI18nOp, op);\n            addParam(\n                params, currentI18nOp, op.i18nPlaceholder.startName, op.slot!, subTemplateIndex,\n                ir.I18nParamResolutionTime.Creation, I18nParamValueFlags.TemplateTag);\n            addParam(\n                params, currentI18nOp, op.i18nPlaceholder.closeName, op.slot!, subTemplateIndex,\n                ir.I18nParamResolutionTime.Creation,\n                I18nParamValueFlags.TemplateTag | I18nParamValueFlags.CloseTag);\n          }\n          break;\n      }\n    }\n\n    // Fill in values for each of the i18n expression placeholders.\n    const i18nBlockPlaceholderIndices = new Map<ir.XrefId, number>();\n    for (const op of unit.update) {\n      if (op.kind === ir.OpKind.I18nExpression) {\n        const i18nOp = i18nOps.get(op.owner);\n        let index = i18nBlockPlaceholderIndices.get(op.owner) || 0;\n        if (!i18nOp) {\n          throw Error('Cannot find corresponding i18nStart for i18nExpr');\n        }\n        addParam(\n            params, i18nOp, op.i18nPlaceholder, index++, i18nOp.subTemplateIndex,\n            op.resolutionTime);\n        i18nBlockPlaceholderIndices.set(op.owner, index);\n      }\n    }\n  }\n}\n\n/**\n * Add a param to the params map for the given i18n op.\n */\nfunction addParam(\n    params: Map<ir.XrefId, I18nPlaceholderParams>, i18nOp: ir.I18nStartOp, placeholder: string,\n    value: string|number, subTemplateIndex: number|null, resolutionTime: ir.I18nParamResolutionTime,\n    flags: I18nParamValueFlags = I18nParamValueFlags.None) {\n  const i18nOpParams = params.get(i18nOp.xref) || new I18nPlaceholderParams();\n  i18nOpParams.addValue(placeholder, value, subTemplateIndex, resolutionTime, flags);\n  params.set(i18nOp.xref, i18nOpParams);\n}\n\n/**\n * Get the subTemplateIndex for the given template op. For template ops, use the subTemplateIndex of\n * the child i18n block inside the template.\n */\nfunction getSubTemplateIndexForTemplateTag(\n    job: ComponentCompilationJob, i18nOp: ir.I18nStartOp, op: ir.TemplateOp): number|null {\n  for (const childOp of job.views.get(op.xref)!.create) {\n    if (childOp.kind === ir.OpKind.I18nStart) {\n      return childOp.subTemplateIndex;\n    }\n  }\n  return i18nOp.subTemplateIndex;\n}\n\n/**\n * Propagate placeholders up to their root i18n op.\n */\nfunction propagatePlaceholders(\n    params: Map<ir.XrefId, I18nPlaceholderParams>, i18nOps: Map<ir.XrefId, ir.I18nStartOp>) {\n  for (const [xref, opParams] of params) {\n    const op = i18nOps.get(xref)!;\n    if (op.xref !== op.root) {\n      const rootParams = params.get(op.root) || new I18nPlaceholderParams();\n      rootParams.merge(opParams);\n    }\n  }\n}\n"]}