@angular/core 10.0.6 → 10.0.10

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 (58) hide show
  1. package/bundles/core-testing.umd.js +1 -1
  2. package/bundles/core-testing.umd.min.js +1 -1
  3. package/bundles/core-testing.umd.min.js.map +1 -1
  4. package/bundles/core.umd.js +1678 -1387
  5. package/bundles/core.umd.js.map +1 -1
  6. package/bundles/core.umd.min.js +152 -124
  7. package/bundles/core.umd.min.js.map +1 -1
  8. package/core.d.ts +258 -58
  9. package/core.metadata.json +1 -1
  10. package/esm2015/core.js +3 -3
  11. package/esm2015/src/application_module.js +2 -2
  12. package/esm2015/src/application_ref.js +2 -2
  13. package/esm2015/src/core_render3_private_export.js +2 -1
  14. package/esm2015/src/metadata/di.js +1 -1
  15. package/esm2015/src/r3_symbols.js +2 -1
  16. package/esm2015/src/reflection/reflection_capabilities.js +38 -8
  17. package/esm2015/src/render3/bindings.js +3 -3
  18. package/esm2015/src/render3/component.js +3 -3
  19. package/esm2015/src/render3/di.js +1 -1
  20. package/esm2015/src/render3/i18n/i18n_apply.js +445 -0
  21. package/esm2015/src/render3/i18n/i18n_debug.js +170 -0
  22. package/esm2015/src/render3/i18n/i18n_locale_id.js +37 -0
  23. package/esm2015/src/render3/i18n/i18n_parse.js +635 -0
  24. package/esm2015/src/render3/i18n/i18n_postprocess.js +121 -0
  25. package/esm2015/src/render3/index.js +3 -2
  26. package/esm2015/src/render3/instructions/advance.js +3 -3
  27. package/esm2015/src/render3/instructions/element.js +3 -3
  28. package/esm2015/src/render3/instructions/element_container.js +3 -3
  29. package/esm2015/src/render3/instructions/i18n.js +164 -0
  30. package/esm2015/src/render3/instructions/listener.js +3 -3
  31. package/esm2015/src/render3/instructions/lview_debug.js +54 -213
  32. package/esm2015/src/render3/instructions/shared.js +54 -38
  33. package/esm2015/src/render3/instructions/text.js +3 -3
  34. package/esm2015/src/render3/interfaces/i18n.js +12 -3
  35. package/esm2015/src/render3/interfaces/node.js +13 -1
  36. package/esm2015/src/render3/interfaces/view.js +1 -1
  37. package/esm2015/src/render3/jit/directive.js +33 -10
  38. package/esm2015/src/render3/ng_module_ref.js +2 -2
  39. package/esm2015/src/render3/node_manipulation.js +1 -11
  40. package/esm2015/src/render3/pure_function.js +3 -3
  41. package/esm2015/src/render3/query.js +14 -12
  42. package/esm2015/src/render3/styling/style_binding_list.js +3 -3
  43. package/esm2015/src/render3/styling/styling_parser.js +3 -2
  44. package/esm2015/src/render3/util/debug_utils.js +31 -2
  45. package/esm2015/src/render3/util/discovery_utils.js +1 -1
  46. package/esm2015/src/render3/util/view_utils.js +5 -5
  47. package/esm2015/src/render3/view_engine_compatibility.js +13 -4
  48. package/esm2015/src/util/assert.js +2 -2
  49. package/esm2015/src/util/char_code.js +1 -1
  50. package/esm2015/src/version.js +1 -1
  51. package/fesm2015/core.js +1652 -1364
  52. package/fesm2015/core.js.map +1 -1
  53. package/fesm2015/testing.js +1 -1
  54. package/package.json +1 -1
  55. package/src/r3_symbols.d.ts +13 -1
  56. package/testing/testing.d.ts +1 -1
  57. package/testing.d.ts +1 -1
  58. package/esm2015/src/render3/i18n.js +0 -1225
@@ -0,0 +1,635 @@
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 '../../util/ng_dev_mode';
9
+ import '../../util/ng_i18n_closure_mode';
10
+ import { getTemplateContent, SRCSET_ATTRS, URI_ATTRS, VALID_ATTRS, VALID_ELEMENTS } from '../../sanitization/html_sanitizer';
11
+ import { getInertBodyHelper } from '../../sanitization/inert_body';
12
+ import { _sanitizeUrl, sanitizeSrcset } from '../../sanitization/url_sanitizer';
13
+ import { addAllToArray } from '../../util/array_utils';
14
+ import { assertEqual } from '../../util/assert';
15
+ import { allocExpando, elementAttributeInternal, setInputsForProperty, setNgReflectProperties } from '../instructions/shared';
16
+ import { getDocument } from '../interfaces/document';
17
+ import { COMMENT_MARKER, ELEMENT_MARKER } from '../interfaces/i18n';
18
+ import { HEADER_OFFSET, T_HOST } from '../interfaces/view';
19
+ import { getIsParent, getPreviousOrParentTNode } from '../state';
20
+ import { attachDebugGetter } from '../util/debug_utils';
21
+ import { getNativeByIndex, getTNode } from '../util/view_utils';
22
+ import { i18nMutateOpCodesToString, i18nUpdateOpCodesToString } from './i18n_debug';
23
+ const BINDING_REGEXP = /�(\d+):?\d*�/gi;
24
+ const ICU_REGEXP = /({\s*�\d+:?\d*�\s*,\s*\S{6}\s*,[\s\S]*})/gi;
25
+ const NESTED_ICU = /�(\d+)�/;
26
+ const ICU_BLOCK_REGEXP = /^\s*(�\d+:?\d*�)\s*,\s*(select|plural)\s*,/;
27
+ // Count for the number of vars that will be allocated for each i18n block.
28
+ // It is global because this is used in multiple functions that include loops and recursive calls.
29
+ // This is reset to 0 when `i18nStartFirstPass` is called.
30
+ let i18nVarsCount;
31
+ const parentIndexStack = [];
32
+ const MARKER = `�`;
33
+ const SUBTEMPLATE_REGEXP = /�\/?\*(\d+:\d+)�/gi;
34
+ const PH_REGEXP = /�(\/?[#*!]\d+):?\d*�/gi;
35
+ /**
36
+ * Angular Dart introduced &ngsp; as a placeholder for non-removable space, see:
37
+ * https://github.com/dart-lang/angular/blob/0bb611387d29d65b5af7f9d2515ab571fd3fbee4/_tests/test/compiler/preserve_whitespace_test.dart#L25-L32
38
+ * In Angular Dart &ngsp; is converted to the 0xE500 PUA (Private Use Areas) unicode character
39
+ * and later on replaced by a space. We are re-implementing the same idea here, since translations
40
+ * might contain this special character.
41
+ */
42
+ const NGSP_UNICODE_REGEXP = /\uE500/g;
43
+ function replaceNgsp(value) {
44
+ return value.replace(NGSP_UNICODE_REGEXP, ' ');
45
+ }
46
+ /**
47
+ * See `i18nStart` above.
48
+ */
49
+ export function i18nStartFirstPass(lView, tView, index, message, subTemplateIndex) {
50
+ const startIndex = tView.blueprint.length - HEADER_OFFSET;
51
+ i18nVarsCount = 0;
52
+ const previousOrParentTNode = getPreviousOrParentTNode();
53
+ const parentTNode = getIsParent() ? previousOrParentTNode : previousOrParentTNode && previousOrParentTNode.parent;
54
+ let parentIndex = parentTNode && parentTNode !== lView[T_HOST] ? parentTNode.index - HEADER_OFFSET : index;
55
+ let parentIndexPointer = 0;
56
+ parentIndexStack[parentIndexPointer] = parentIndex;
57
+ const createOpCodes = [];
58
+ if (ngDevMode) {
59
+ attachDebugGetter(createOpCodes, i18nMutateOpCodesToString);
60
+ }
61
+ // If the previous node wasn't the direct parent then we have a translation without top level
62
+ // element and we need to keep a reference of the previous element if there is one. We should also
63
+ // keep track whether an element was a parent node or not, so that the logic that consumes
64
+ // the generated `I18nMutateOpCode`s can leverage this information to properly set TNode state
65
+ // (whether it's a parent or sibling).
66
+ if (index > 0 && previousOrParentTNode !== parentTNode) {
67
+ let previousTNodeIndex = previousOrParentTNode.index - HEADER_OFFSET;
68
+ // If current TNode is a sibling node, encode it using a negative index. This information is
69
+ // required when the `Select` action is processed (see the `readCreateOpCodes` function).
70
+ if (!getIsParent()) {
71
+ previousTNodeIndex = ~previousTNodeIndex;
72
+ }
73
+ // Create an OpCode to select the previous TNode
74
+ createOpCodes.push(previousTNodeIndex << 3 /* SHIFT_REF */ | 0 /* Select */);
75
+ }
76
+ const updateOpCodes = [];
77
+ if (ngDevMode) {
78
+ attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
79
+ }
80
+ const icuExpressions = [];
81
+ if (message === '' && isRootTemplateMessage(subTemplateIndex)) {
82
+ // If top level translation is an empty string, do not invoke additional processing
83
+ // and just create op codes for empty text node instead.
84
+ createOpCodes.push(message, allocNodeIndex(startIndex), parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
85
+ }
86
+ else {
87
+ const templateTranslation = getTranslationForTemplate(message, subTemplateIndex);
88
+ const msgParts = replaceNgsp(templateTranslation).split(PH_REGEXP);
89
+ for (let i = 0; i < msgParts.length; i++) {
90
+ let value = msgParts[i];
91
+ if (i & 1) {
92
+ // Odd indexes are placeholders (elements and sub-templates)
93
+ if (value.charAt(0) === '/') {
94
+ // It is a closing tag
95
+ if (value.charAt(1) === "#" /* ELEMENT */) {
96
+ const phIndex = parseInt(value.substr(2), 10);
97
+ parentIndex = parentIndexStack[--parentIndexPointer];
98
+ createOpCodes.push(phIndex << 3 /* SHIFT_REF */ | 5 /* ElementEnd */);
99
+ }
100
+ }
101
+ else {
102
+ const phIndex = parseInt(value.substr(1), 10);
103
+ const isElement = value.charAt(0) === "#" /* ELEMENT */;
104
+ // The value represents a placeholder that we move to the designated index.
105
+ // Note: positive indicies indicate that a TNode with a given index should also be marked
106
+ // as parent while executing `Select` instruction.
107
+ createOpCodes.push((isElement ? phIndex : ~phIndex) << 3 /* SHIFT_REF */ |
108
+ 0 /* Select */, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
109
+ if (isElement) {
110
+ parentIndexStack[++parentIndexPointer] = parentIndex = phIndex;
111
+ }
112
+ }
113
+ }
114
+ else {
115
+ // Even indexes are text (including bindings & ICU expressions)
116
+ const parts = extractParts(value);
117
+ for (let j = 0; j < parts.length; j++) {
118
+ if (j & 1) {
119
+ // Odd indexes are ICU expressions
120
+ const icuExpression = parts[j];
121
+ // Verify that ICU expression has the right shape. Translations might contain invalid
122
+ // constructions (while original messages were correct), so ICU parsing at runtime may
123
+ // not succeed (thus `icuExpression` remains a string).
124
+ if (typeof icuExpression !== 'object') {
125
+ throw new Error(`Unable to parse ICU expression in "${templateTranslation}" message.`);
126
+ }
127
+ // Create the comment node that will anchor the ICU expression
128
+ const icuNodeIndex = allocNodeIndex(startIndex);
129
+ createOpCodes.push(COMMENT_MARKER, ngDevMode ? `ICU ${icuNodeIndex}` : '', icuNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
130
+ // Update codes for the ICU expression
131
+ const mask = getBindingMask(icuExpression);
132
+ icuStart(icuExpressions, icuExpression, icuNodeIndex, icuNodeIndex);
133
+ // Since this is recursive, the last TIcu that was pushed is the one we want
134
+ const tIcuIndex = icuExpressions.length - 1;
135
+ updateOpCodes.push(toMaskBit(icuExpression.mainBinding), // mask of the main binding
136
+ 3, // skip 3 opCodes if not changed
137
+ -1 - icuExpression.mainBinding, icuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */, tIcuIndex, mask, // mask of all the bindings of this ICU expression
138
+ 2, // skip 2 opCodes if not changed
139
+ icuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, tIcuIndex);
140
+ }
141
+ else if (parts[j] !== '') {
142
+ const text = parts[j];
143
+ // Even indexes are text (including bindings)
144
+ const hasBinding = text.match(BINDING_REGEXP);
145
+ // Create text nodes
146
+ const textNodeIndex = allocNodeIndex(startIndex);
147
+ createOpCodes.push(
148
+ // If there is a binding, the value will be set during update
149
+ hasBinding ? '' : text, textNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
150
+ if (hasBinding) {
151
+ addAllToArray(generateBindingUpdateOpCodes(text, textNodeIndex), updateOpCodes);
152
+ }
153
+ }
154
+ }
155
+ }
156
+ }
157
+ }
158
+ if (i18nVarsCount > 0) {
159
+ allocExpando(tView, lView, i18nVarsCount);
160
+ }
161
+ // NOTE: local var needed to properly assert the type of `TI18n`.
162
+ const tI18n = {
163
+ vars: i18nVarsCount,
164
+ create: createOpCodes,
165
+ update: updateOpCodes,
166
+ icus: icuExpressions.length ? icuExpressions : null,
167
+ };
168
+ tView.data[index + HEADER_OFFSET] = tI18n;
169
+ }
170
+ /**
171
+ * See `i18nAttributes` above.
172
+ */
173
+ export function i18nAttributesFirstPass(lView, tView, index, values) {
174
+ const previousElement = getPreviousOrParentTNode();
175
+ const previousElementIndex = previousElement.index - HEADER_OFFSET;
176
+ const updateOpCodes = [];
177
+ if (ngDevMode) {
178
+ attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
179
+ }
180
+ for (let i = 0; i < values.length; i += 2) {
181
+ const attrName = values[i];
182
+ const message = values[i + 1];
183
+ const parts = message.split(ICU_REGEXP);
184
+ for (let j = 0; j < parts.length; j++) {
185
+ const value = parts[j];
186
+ if (j & 1) {
187
+ // Odd indexes are ICU expressions
188
+ // TODO(ocombe): support ICU expressions in attributes
189
+ throw new Error('ICU expressions are not yet supported in attributes');
190
+ }
191
+ else if (value !== '') {
192
+ // Even indexes are text (including bindings)
193
+ const hasBinding = !!value.match(BINDING_REGEXP);
194
+ if (hasBinding) {
195
+ if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
196
+ addAllToArray(generateBindingUpdateOpCodes(value, previousElementIndex, attrName), updateOpCodes);
197
+ }
198
+ }
199
+ else {
200
+ const tNode = getTNode(tView, previousElementIndex);
201
+ // Set attributes for Elements only, for other types (like ElementContainer),
202
+ // only set inputs below
203
+ if (tNode.type === 3 /* Element */) {
204
+ elementAttributeInternal(tNode, lView, attrName, value, null, null);
205
+ }
206
+ // Check if that attribute is a directive input
207
+ const dataValue = tNode.inputs !== null && tNode.inputs[attrName];
208
+ if (dataValue) {
209
+ setInputsForProperty(tView, lView, dataValue, attrName, value);
210
+ if (ngDevMode) {
211
+ const element = getNativeByIndex(previousElementIndex, lView);
212
+ setNgReflectProperties(lView, element, tNode.type, dataValue, value);
213
+ }
214
+ }
215
+ }
216
+ }
217
+ }
218
+ }
219
+ if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
220
+ tView.data[index + HEADER_OFFSET] = updateOpCodes;
221
+ }
222
+ }
223
+ /**
224
+ * Generate the OpCodes to update the bindings of a string.
225
+ *
226
+ * @param str The string containing the bindings.
227
+ * @param destinationNode Index of the destination node which will receive the binding.
228
+ * @param attrName Name of the attribute, if the string belongs to an attribute.
229
+ * @param sanitizeFn Sanitization function used to sanitize the string after update, if necessary.
230
+ */
231
+ export function generateBindingUpdateOpCodes(str, destinationNode, attrName, sanitizeFn = null) {
232
+ const updateOpCodes = [null, null]; // Alloc space for mask and size
233
+ if (ngDevMode) {
234
+ attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
235
+ }
236
+ const textParts = str.split(BINDING_REGEXP);
237
+ let mask = 0;
238
+ for (let j = 0; j < textParts.length; j++) {
239
+ const textValue = textParts[j];
240
+ if (j & 1) {
241
+ // Odd indexes are bindings
242
+ const bindingIndex = parseInt(textValue, 10);
243
+ updateOpCodes.push(-1 - bindingIndex);
244
+ mask = mask | toMaskBit(bindingIndex);
245
+ }
246
+ else if (textValue !== '') {
247
+ // Even indexes are text
248
+ updateOpCodes.push(textValue);
249
+ }
250
+ }
251
+ updateOpCodes.push(destinationNode << 2 /* SHIFT_REF */ |
252
+ (attrName ? 1 /* Attr */ : 0 /* Text */));
253
+ if (attrName) {
254
+ updateOpCodes.push(attrName, sanitizeFn);
255
+ }
256
+ updateOpCodes[0] = mask;
257
+ updateOpCodes[1] = updateOpCodes.length - 2;
258
+ return updateOpCodes;
259
+ }
260
+ function getBindingMask(icuExpression, mask = 0) {
261
+ mask = mask | toMaskBit(icuExpression.mainBinding);
262
+ let match;
263
+ for (let i = 0; i < icuExpression.values.length; i++) {
264
+ const valueArr = icuExpression.values[i];
265
+ for (let j = 0; j < valueArr.length; j++) {
266
+ const value = valueArr[j];
267
+ if (typeof value === 'string') {
268
+ while (match = BINDING_REGEXP.exec(value)) {
269
+ mask = mask | toMaskBit(parseInt(match[1], 10));
270
+ }
271
+ }
272
+ else {
273
+ mask = getBindingMask(value, mask);
274
+ }
275
+ }
276
+ }
277
+ return mask;
278
+ }
279
+ function allocNodeIndex(startIndex) {
280
+ return startIndex + i18nVarsCount++;
281
+ }
282
+ /**
283
+ * Convert binding index to mask bit.
284
+ *
285
+ * Each index represents a single bit on the bit-mask. Because bit-mask only has 32 bits, we make
286
+ * the 32nd bit share all masks for all bindings higher than 32. Since it is extremely rare to have
287
+ * more than 32 bindings this will be hit very rarely. The downside of hitting this corner case is
288
+ * that we will execute binding code more often than necessary. (penalty of performance)
289
+ */
290
+ function toMaskBit(bindingIndex) {
291
+ return 1 << Math.min(bindingIndex, 31);
292
+ }
293
+ export function isRootTemplateMessage(subTemplateIndex) {
294
+ return subTemplateIndex === undefined;
295
+ }
296
+ /**
297
+ * Removes everything inside the sub-templates of a message.
298
+ */
299
+ function removeInnerTemplateTranslation(message) {
300
+ let match;
301
+ let res = '';
302
+ let index = 0;
303
+ let inTemplate = false;
304
+ let tagMatched;
305
+ while ((match = SUBTEMPLATE_REGEXP.exec(message)) !== null) {
306
+ if (!inTemplate) {
307
+ res += message.substring(index, match.index + match[0].length);
308
+ tagMatched = match[1];
309
+ inTemplate = true;
310
+ }
311
+ else {
312
+ if (match[0] === `${MARKER}/*${tagMatched}${MARKER}`) {
313
+ index = match.index;
314
+ inTemplate = false;
315
+ }
316
+ }
317
+ }
318
+ ngDevMode &&
319
+ assertEqual(inTemplate, false, `Tag mismatch: unable to find the end of the sub-template in the translation "${message}"`);
320
+ res += message.substr(index);
321
+ return res;
322
+ }
323
+ /**
324
+ * Extracts a part of a message and removes the rest.
325
+ *
326
+ * This method is used for extracting a part of the message associated with a template. A translated
327
+ * message can span multiple templates.
328
+ *
329
+ * Example:
330
+ * ```
331
+ * <div i18n>Translate <span *ngIf>me</span>!</div>
332
+ * ```
333
+ *
334
+ * @param message The message to crop
335
+ * @param subTemplateIndex Index of the sub-template to extract. If undefined it returns the
336
+ * external template and removes all sub-templates.
337
+ */
338
+ export function getTranslationForTemplate(message, subTemplateIndex) {
339
+ if (isRootTemplateMessage(subTemplateIndex)) {
340
+ // We want the root template message, ignore all sub-templates
341
+ return removeInnerTemplateTranslation(message);
342
+ }
343
+ else {
344
+ // We want a specific sub-template
345
+ const start = message.indexOf(`:${subTemplateIndex}${MARKER}`) + 2 + subTemplateIndex.toString().length;
346
+ const end = message.search(new RegExp(`${MARKER}\\/\\*\\d+:${subTemplateIndex}${MARKER}`));
347
+ return removeInnerTemplateTranslation(message.substring(start, end));
348
+ }
349
+ }
350
+ /**
351
+ * Generate the OpCodes for ICU expressions.
352
+ *
353
+ * @param tIcus
354
+ * @param icuExpression
355
+ * @param startIndex
356
+ * @param expandoStartIndex
357
+ */
358
+ export function icuStart(tIcus, icuExpression, startIndex, expandoStartIndex) {
359
+ const createCodes = [];
360
+ const removeCodes = [];
361
+ const updateCodes = [];
362
+ const vars = [];
363
+ const childIcus = [];
364
+ const values = icuExpression.values;
365
+ for (let i = 0; i < values.length; i++) {
366
+ // Each value is an array of strings & other ICU expressions
367
+ const valueArr = values[i];
368
+ const nestedIcus = [];
369
+ for (let j = 0; j < valueArr.length; j++) {
370
+ const value = valueArr[j];
371
+ if (typeof value !== 'string') {
372
+ // It is an nested ICU expression
373
+ const icuIndex = nestedIcus.push(value) - 1;
374
+ // Replace nested ICU expression by a comment node
375
+ valueArr[j] = `<!--�${icuIndex}�-->`;
376
+ }
377
+ }
378
+ const icuCase = parseIcuCase(valueArr.join(''), startIndex, nestedIcus, tIcus, expandoStartIndex);
379
+ createCodes.push(icuCase.create);
380
+ removeCodes.push(icuCase.remove);
381
+ updateCodes.push(icuCase.update);
382
+ vars.push(icuCase.vars);
383
+ childIcus.push(icuCase.childIcus);
384
+ }
385
+ const tIcu = {
386
+ type: icuExpression.type,
387
+ vars,
388
+ currentCaseLViewIndex: HEADER_OFFSET +
389
+ expandoStartIndex // expandoStartIndex does not include the header so add it.
390
+ + 1,
391
+ childIcus,
392
+ cases: icuExpression.cases,
393
+ create: createCodes,
394
+ remove: removeCodes,
395
+ update: updateCodes
396
+ };
397
+ tIcus.push(tIcu);
398
+ // Adding the maximum possible of vars needed (based on the cases with the most vars)
399
+ i18nVarsCount += Math.max(...vars);
400
+ }
401
+ /**
402
+ * Parses text containing an ICU expression and produces a JSON object for it.
403
+ * Original code from closure library, modified for Angular.
404
+ *
405
+ * @param pattern Text containing an ICU expression that needs to be parsed.
406
+ *
407
+ */
408
+ export function parseICUBlock(pattern) {
409
+ const cases = [];
410
+ const values = [];
411
+ let icuType = 1 /* plural */;
412
+ let mainBinding = 0;
413
+ pattern = pattern.replace(ICU_BLOCK_REGEXP, function (str, binding, type) {
414
+ if (type === 'select') {
415
+ icuType = 0 /* select */;
416
+ }
417
+ else {
418
+ icuType = 1 /* plural */;
419
+ }
420
+ mainBinding = parseInt(binding.substr(1), 10);
421
+ return '';
422
+ });
423
+ const parts = extractParts(pattern);
424
+ // Looking for (key block)+ sequence. One of the keys has to be "other".
425
+ for (let pos = 0; pos < parts.length;) {
426
+ let key = parts[pos++].trim();
427
+ if (icuType === 1 /* plural */) {
428
+ // Key can be "=x", we just want "x"
429
+ key = key.replace(/\s*(?:=)?(\w+)\s*/, '$1');
430
+ }
431
+ if (key.length) {
432
+ cases.push(key);
433
+ }
434
+ const blocks = extractParts(parts[pos++]);
435
+ if (cases.length > values.length) {
436
+ values.push(blocks);
437
+ }
438
+ }
439
+ // TODO(ocombe): support ICU expressions in attributes, see #21615
440
+ return { type: icuType, mainBinding: mainBinding, cases, values };
441
+ }
442
+ /**
443
+ * Transforms a string template into an HTML template and a list of instructions used to update
444
+ * attributes or nodes that contain bindings.
445
+ *
446
+ * @param unsafeHtml The string to parse
447
+ * @param parentIndex
448
+ * @param nestedIcus
449
+ * @param tIcus
450
+ * @param expandoStartIndex
451
+ */
452
+ function parseIcuCase(unsafeHtml, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
453
+ const inertBodyHelper = getInertBodyHelper(getDocument());
454
+ const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);
455
+ if (!inertBodyElement) {
456
+ throw new Error('Unable to generate inert body element');
457
+ }
458
+ const wrapper = getTemplateContent(inertBodyElement) || inertBodyElement;
459
+ const opCodes = {
460
+ vars: 1,
461
+ childIcus: [],
462
+ create: [],
463
+ remove: [],
464
+ update: []
465
+ };
466
+ if (ngDevMode) {
467
+ attachDebugGetter(opCodes.create, i18nMutateOpCodesToString);
468
+ attachDebugGetter(opCodes.remove, i18nMutateOpCodesToString);
469
+ attachDebugGetter(opCodes.update, i18nUpdateOpCodesToString);
470
+ }
471
+ parseNodes(wrapper.firstChild, opCodes, parentIndex, nestedIcus, tIcus, expandoStartIndex);
472
+ return opCodes;
473
+ }
474
+ /**
475
+ * Breaks pattern into strings and top level {...} blocks.
476
+ * Can be used to break a message into text and ICU expressions, or to break an ICU expression into
477
+ * keys and cases.
478
+ * Original code from closure library, modified for Angular.
479
+ *
480
+ * @param pattern (sub)Pattern to be broken.
481
+ *
482
+ */
483
+ function extractParts(pattern) {
484
+ if (!pattern) {
485
+ return [];
486
+ }
487
+ let prevPos = 0;
488
+ const braceStack = [];
489
+ const results = [];
490
+ const braces = /[{}]/g;
491
+ // lastIndex doesn't get set to 0 so we have to.
492
+ braces.lastIndex = 0;
493
+ let match;
494
+ while (match = braces.exec(pattern)) {
495
+ const pos = match.index;
496
+ if (match[0] == '}') {
497
+ braceStack.pop();
498
+ if (braceStack.length == 0) {
499
+ // End of the block.
500
+ const block = pattern.substring(prevPos, pos);
501
+ if (ICU_BLOCK_REGEXP.test(block)) {
502
+ results.push(parseICUBlock(block));
503
+ }
504
+ else {
505
+ results.push(block);
506
+ }
507
+ prevPos = pos + 1;
508
+ }
509
+ }
510
+ else {
511
+ if (braceStack.length == 0) {
512
+ const substring = pattern.substring(prevPos, pos);
513
+ results.push(substring);
514
+ prevPos = pos + 1;
515
+ }
516
+ braceStack.push('{');
517
+ }
518
+ }
519
+ const substring = pattern.substring(prevPos);
520
+ results.push(substring);
521
+ return results;
522
+ }
523
+ /**
524
+ * Parses a node, its children and its siblings, and generates the mutate & update OpCodes.
525
+ *
526
+ * @param currentNode The first node to parse
527
+ * @param icuCase The data for the ICU expression case that contains those nodes
528
+ * @param parentIndex Index of the current node's parent
529
+ * @param nestedIcus Data for the nested ICU expressions that this case contains
530
+ * @param tIcus Data for all ICU expressions of the current message
531
+ * @param expandoStartIndex Expando start index for the current ICU expression
532
+ */
533
+ export function parseNodes(currentNode, icuCase, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
534
+ if (currentNode) {
535
+ const nestedIcusToCreate = [];
536
+ while (currentNode) {
537
+ const nextNode = currentNode.nextSibling;
538
+ const newIndex = expandoStartIndex + ++icuCase.vars;
539
+ switch (currentNode.nodeType) {
540
+ case Node.ELEMENT_NODE:
541
+ const element = currentNode;
542
+ const tagName = element.tagName.toLowerCase();
543
+ if (!VALID_ELEMENTS.hasOwnProperty(tagName)) {
544
+ // This isn't a valid element, we won't create an element for it
545
+ icuCase.vars--;
546
+ }
547
+ else {
548
+ icuCase.create.push(ELEMENT_MARKER, tagName, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
549
+ const elAttrs = element.attributes;
550
+ for (let i = 0; i < elAttrs.length; i++) {
551
+ const attr = elAttrs.item(i);
552
+ const lowerAttrName = attr.name.toLowerCase();
553
+ const hasBinding = !!attr.value.match(BINDING_REGEXP);
554
+ // we assume the input string is safe, unless it's using a binding
555
+ if (hasBinding) {
556
+ if (VALID_ATTRS.hasOwnProperty(lowerAttrName)) {
557
+ if (URI_ATTRS[lowerAttrName]) {
558
+ addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, _sanitizeUrl), icuCase.update);
559
+ }
560
+ else if (SRCSET_ATTRS[lowerAttrName]) {
561
+ addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, sanitizeSrcset), icuCase.update);
562
+ }
563
+ else {
564
+ addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name), icuCase.update);
565
+ }
566
+ }
567
+ else {
568
+ ngDevMode &&
569
+ console.warn(`WARNING: ignoring unsafe attribute value ${lowerAttrName} on element ${tagName} (see http://g.co/ng/security#xss)`);
570
+ }
571
+ }
572
+ else {
573
+ icuCase.create.push(newIndex << 3 /* SHIFT_REF */ | 4 /* Attr */, attr.name, attr.value);
574
+ }
575
+ }
576
+ // Parse the children of this node (if any)
577
+ parseNodes(currentNode.firstChild, icuCase, newIndex, nestedIcus, tIcus, expandoStartIndex);
578
+ // Remove the parent node after the children
579
+ icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
580
+ }
581
+ break;
582
+ case Node.TEXT_NODE:
583
+ const value = currentNode.textContent || '';
584
+ const hasBinding = value.match(BINDING_REGEXP);
585
+ icuCase.create.push(hasBinding ? '' : value, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
586
+ icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
587
+ if (hasBinding) {
588
+ addAllToArray(generateBindingUpdateOpCodes(value, newIndex), icuCase.update);
589
+ }
590
+ break;
591
+ case Node.COMMENT_NODE:
592
+ // Check if the comment node is a placeholder for a nested ICU
593
+ const match = NESTED_ICU.exec(currentNode.textContent || '');
594
+ if (match) {
595
+ const nestedIcuIndex = parseInt(match[1], 10);
596
+ const newLocal = ngDevMode ? `nested ICU ${nestedIcuIndex}` : '';
597
+ // Create the comment node that will anchor the ICU expression
598
+ icuCase.create.push(COMMENT_MARKER, newLocal, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
599
+ const nestedIcu = nestedIcus[nestedIcuIndex];
600
+ nestedIcusToCreate.push([nestedIcu, newIndex]);
601
+ }
602
+ else {
603
+ // We do not handle any other type of comment
604
+ icuCase.vars--;
605
+ }
606
+ break;
607
+ default:
608
+ // We do not handle any other type of element
609
+ icuCase.vars--;
610
+ }
611
+ currentNode = nextNode;
612
+ }
613
+ for (let i = 0; i < nestedIcusToCreate.length; i++) {
614
+ const nestedIcu = nestedIcusToCreate[i][0];
615
+ const nestedIcuNodeIndex = nestedIcusToCreate[i][1];
616
+ icuStart(tIcus, nestedIcu, nestedIcuNodeIndex, expandoStartIndex + icuCase.vars);
617
+ // Since this is recursive, the last TIcu that was pushed is the one we want
618
+ const nestTIcuIndex = tIcus.length - 1;
619
+ icuCase.vars += Math.max(...tIcus[nestTIcuIndex].vars);
620
+ icuCase.childIcus.push(nestTIcuIndex);
621
+ const mask = getBindingMask(nestedIcu);
622
+ icuCase.update.push(toMaskBit(nestedIcu.mainBinding), // mask of the main binding
623
+ 3, // skip 3 opCodes if not changed
624
+ -1 - nestedIcu.mainBinding, nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */,
625
+ // FIXME(misko): Index should be part of the opcode
626
+ nestTIcuIndex, mask, // mask of all the bindings of this ICU expression
627
+ 2, // skip 2 opCodes if not changed
628
+ nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, nestTIcuIndex);
629
+ icuCase.remove.push(nestTIcuIndex << 3 /* SHIFT_REF */ | 6 /* RemoveNestedIcu */,
630
+ // FIXME(misko): Index should be part of the opcode
631
+ nestedIcuNodeIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
632
+ }
633
+ }
634
+ }
635
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n_parse.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/i18n/i18n_parse.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 '../../util/ng_dev_mode';\nimport '../../util/ng_i18n_closure_mode';\n\nimport {getTemplateContent, SRCSET_ATTRS, URI_ATTRS, VALID_ATTRS, VALID_ELEMENTS} from '../../sanitization/html_sanitizer';\nimport {getInertBodyHelper} from '../../sanitization/inert_body';\nimport {_sanitizeUrl, sanitizeSrcset} from '../../sanitization/url_sanitizer';\nimport {addAllToArray} from '../../util/array_utils';\nimport {assertEqual} from '../../util/assert';\nimport {allocExpando, elementAttributeInternal, setInputsForProperty, setNgReflectProperties} from '../instructions/shared';\nimport {getDocument} from '../interfaces/document';\nimport {COMMENT_MARKER, ELEMENT_MARKER, I18nMutateOpCode, I18nMutateOpCodes, I18nUpdateOpCode, I18nUpdateOpCodes, IcuCase, IcuExpression, IcuType, TI18n, TIcu} from '../interfaces/i18n';\nimport {TNodeType} from '../interfaces/node';\nimport {RComment, RElement} from '../interfaces/renderer';\nimport {SanitizerFn} from '../interfaces/sanitization';\nimport {HEADER_OFFSET, LView, T_HOST, TView} from '../interfaces/view';\nimport {getIsParent, getPreviousOrParentTNode} from '../state';\nimport {attachDebugGetter} from '../util/debug_utils';\nimport {getNativeByIndex, getTNode} from '../util/view_utils';\n\nimport {i18nMutateOpCodesToString, i18nUpdateOpCodesToString} from './i18n_debug';\n\n\n\nconst BINDING_REGEXP = /�(\\d+):?\\d*�/gi;\nconst ICU_REGEXP = /({\\s*�\\d+:?\\d*�\\s*,\\s*\\S{6}\\s*,[\\s\\S]*})/gi;\nconst NESTED_ICU = /�(\\d+)�/;\nconst ICU_BLOCK_REGEXP = /^\\s*(�\\d+:?\\d*�)\\s*,\\s*(select|plural)\\s*,/;\n\n\n// Count for the number of vars that will be allocated for each i18n block.\n// It is global because this is used in multiple functions that include loops and recursive calls.\n// This is reset to 0 when `i18nStartFirstPass` is called.\nlet i18nVarsCount: number;\n\nconst parentIndexStack: number[] = [];\n\nconst MARKER = `�`;\nconst SUBTEMPLATE_REGEXP = /�\\/?\\*(\\d+:\\d+)�/gi;\nconst PH_REGEXP = /�(\\/?[#*!]\\d+):?\\d*�/gi;\nconst enum TagType {\n  ELEMENT = '#',\n  TEMPLATE = '*',\n  PROJECTION = '!',\n}\n\n/**\n * Angular Dart introduced &ngsp; as a placeholder for non-removable space, see:\n * https://github.com/dart-lang/angular/blob/0bb611387d29d65b5af7f9d2515ab571fd3fbee4/_tests/test/compiler/preserve_whitespace_test.dart#L25-L32\n * In Angular Dart &ngsp; is converted to the 0xE500 PUA (Private Use Areas) unicode character\n * and later on replaced by a space. We are re-implementing the same idea here, since translations\n * might contain this special character.\n */\nconst NGSP_UNICODE_REGEXP = /\\uE500/g;\nfunction replaceNgsp(value: string): string {\n  return value.replace(NGSP_UNICODE_REGEXP, ' ');\n}\n\n\n/**\n * See `i18nStart` above.\n */\nexport function i18nStartFirstPass(\n    lView: LView, tView: TView, index: number, message: string, subTemplateIndex?: number) {\n  const startIndex = tView.blueprint.length - HEADER_OFFSET;\n  i18nVarsCount = 0;\n  const previousOrParentTNode = getPreviousOrParentTNode();\n  const parentTNode =\n      getIsParent() ? previousOrParentTNode : previousOrParentTNode && previousOrParentTNode.parent;\n  let parentIndex =\n      parentTNode && parentTNode !== lView[T_HOST] ? parentTNode.index - HEADER_OFFSET : index;\n  let parentIndexPointer = 0;\n  parentIndexStack[parentIndexPointer] = parentIndex;\n  const createOpCodes: I18nMutateOpCodes = [];\n  if (ngDevMode) {\n    attachDebugGetter(createOpCodes, i18nMutateOpCodesToString);\n  }\n  // If the previous node wasn't the direct parent then we have a translation without top level\n  // element and we need to keep a reference of the previous element if there is one. We should also\n  // keep track whether an element was a parent node or not, so that the logic that consumes\n  // the generated `I18nMutateOpCode`s can leverage this information to properly set TNode state\n  // (whether it's a parent or sibling).\n  if (index > 0 && previousOrParentTNode !== parentTNode) {\n    let previousTNodeIndex = previousOrParentTNode.index - HEADER_OFFSET;\n    // If current TNode is a sibling node, encode it using a negative index. This information is\n    // required when the `Select` action is processed (see the `readCreateOpCodes` function).\n    if (!getIsParent()) {\n      previousTNodeIndex = ~previousTNodeIndex;\n    }\n    // Create an OpCode to select the previous TNode\n    createOpCodes.push(previousTNodeIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Select);\n  }\n  const updateOpCodes: I18nUpdateOpCodes = [];\n  if (ngDevMode) {\n    attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);\n  }\n  const icuExpressions: TIcu[] = [];\n\n  if (message === '' && isRootTemplateMessage(subTemplateIndex)) {\n    // If top level translation is an empty string, do not invoke additional processing\n    // and just create op codes for empty text node instead.\n    createOpCodes.push(\n        message, allocNodeIndex(startIndex),\n        parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n  } else {\n    const templateTranslation = getTranslationForTemplate(message, subTemplateIndex);\n    const msgParts = replaceNgsp(templateTranslation).split(PH_REGEXP);\n    for (let i = 0; i < msgParts.length; i++) {\n      let value = msgParts[i];\n      if (i & 1) {\n        // Odd indexes are placeholders (elements and sub-templates)\n        if (value.charAt(0) === '/') {\n          // It is a closing tag\n          if (value.charAt(1) === TagType.ELEMENT) {\n            const phIndex = parseInt(value.substr(2), 10);\n            parentIndex = parentIndexStack[--parentIndexPointer];\n            createOpCodes.push(phIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.ElementEnd);\n          }\n        } else {\n          const phIndex = parseInt(value.substr(1), 10);\n          const isElement = value.charAt(0) === TagType.ELEMENT;\n          // The value represents a placeholder that we move to the designated index.\n          // Note: positive indicies indicate that a TNode with a given index should also be marked\n          // as parent while executing `Select` instruction.\n          createOpCodes.push(\n              (isElement ? phIndex : ~phIndex) << I18nMutateOpCode.SHIFT_REF |\n                  I18nMutateOpCode.Select,\n              parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n\n          if (isElement) {\n            parentIndexStack[++parentIndexPointer] = parentIndex = phIndex;\n          }\n        }\n      } else {\n        // Even indexes are text (including bindings & ICU expressions)\n        const parts = extractParts(value);\n        for (let j = 0; j < parts.length; j++) {\n          if (j & 1) {\n            // Odd indexes are ICU expressions\n            const icuExpression = parts[j] as IcuExpression;\n\n            // Verify that ICU expression has the right shape. Translations might contain invalid\n            // constructions (while original messages were correct), so ICU parsing at runtime may\n            // not succeed (thus `icuExpression` remains a string).\n            if (typeof icuExpression !== 'object') {\n              throw new Error(\n                  `Unable to parse ICU expression in \"${templateTranslation}\" message.`);\n            }\n\n            // Create the comment node that will anchor the ICU expression\n            const icuNodeIndex = allocNodeIndex(startIndex);\n            createOpCodes.push(\n                COMMENT_MARKER, ngDevMode ? `ICU ${icuNodeIndex}` : '', icuNodeIndex,\n                parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n\n            // Update codes for the ICU expression\n            const mask = getBindingMask(icuExpression);\n            icuStart(icuExpressions, icuExpression, icuNodeIndex, icuNodeIndex);\n            // Since this is recursive, the last TIcu that was pushed is the one we want\n            const tIcuIndex = icuExpressions.length - 1;\n            updateOpCodes.push(\n                toMaskBit(icuExpression.mainBinding),  // mask of the main binding\n                3,                                     // skip 3 opCodes if not changed\n                -1 - icuExpression.mainBinding,\n                icuNodeIndex << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuSwitch, tIcuIndex,\n                mask,  // mask of all the bindings of this ICU expression\n                2,     // skip 2 opCodes if not changed\n                icuNodeIndex << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuUpdate, tIcuIndex);\n          } else if (parts[j] !== '') {\n            const text = parts[j] as string;\n            // Even indexes are text (including bindings)\n            const hasBinding = text.match(BINDING_REGEXP);\n            // Create text nodes\n            const textNodeIndex = allocNodeIndex(startIndex);\n            createOpCodes.push(\n                // If there is a binding, the value will be set during update\n                hasBinding ? '' : text, textNodeIndex,\n                parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n\n            if (hasBinding) {\n              addAllToArray(generateBindingUpdateOpCodes(text, textNodeIndex), updateOpCodes);\n            }\n          }\n        }\n      }\n    }\n  }\n\n  if (i18nVarsCount > 0) {\n    allocExpando(tView, lView, i18nVarsCount);\n  }\n\n  // NOTE: local var needed to properly assert the type of `TI18n`.\n  const tI18n: TI18n = {\n    vars: i18nVarsCount,\n    create: createOpCodes,\n    update: updateOpCodes,\n    icus: icuExpressions.length ? icuExpressions : null,\n  };\n\n  tView.data[index + HEADER_OFFSET] = tI18n;\n}\n\n/**\n * See `i18nAttributes` above.\n */\nexport function i18nAttributesFirstPass(\n    lView: LView, tView: TView, index: number, values: string[]) {\n  const previousElement = getPreviousOrParentTNode();\n  const previousElementIndex = previousElement.index - HEADER_OFFSET;\n  const updateOpCodes: I18nUpdateOpCodes = [];\n  if (ngDevMode) {\n    attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);\n  }\n  for (let i = 0; i < values.length; i += 2) {\n    const attrName = values[i];\n    const message = values[i + 1];\n    const parts = message.split(ICU_REGEXP);\n    for (let j = 0; j < parts.length; j++) {\n      const value = parts[j];\n\n      if (j & 1) {\n        // Odd indexes are ICU expressions\n        // TODO(ocombe): support ICU expressions in attributes\n        throw new Error('ICU expressions are not yet supported in attributes');\n      } else if (value !== '') {\n        // Even indexes are text (including bindings)\n        const hasBinding = !!value.match(BINDING_REGEXP);\n        if (hasBinding) {\n          if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {\n            addAllToArray(\n                generateBindingUpdateOpCodes(value, previousElementIndex, attrName), updateOpCodes);\n          }\n        } else {\n          const tNode = getTNode(tView, previousElementIndex);\n          // Set attributes for Elements only, for other types (like ElementContainer),\n          // only set inputs below\n          if (tNode.type === TNodeType.Element) {\n            elementAttributeInternal(tNode, lView, attrName, value, null, null);\n          }\n          // Check if that attribute is a directive input\n          const dataValue = tNode.inputs !== null && tNode.inputs[attrName];\n          if (dataValue) {\n            setInputsForProperty(tView, lView, dataValue, attrName, value);\n            if (ngDevMode) {\n              const element = getNativeByIndex(previousElementIndex, lView) as RElement | RComment;\n              setNgReflectProperties(lView, element, tNode.type, dataValue, value);\n            }\n          }\n        }\n      }\n    }\n  }\n\n  if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {\n    tView.data[index + HEADER_OFFSET] = updateOpCodes;\n  }\n}\n\n\n/**\n * Generate the OpCodes to update the bindings of a string.\n *\n * @param str The string containing the bindings.\n * @param destinationNode Index of the destination node which will receive the binding.\n * @param attrName Name of the attribute, if the string belongs to an attribute.\n * @param sanitizeFn Sanitization function used to sanitize the string after update, if necessary.\n */\nexport function generateBindingUpdateOpCodes(\n    str: string, destinationNode: number, attrName?: string,\n    sanitizeFn: SanitizerFn|null = null): I18nUpdateOpCodes {\n  const updateOpCodes: I18nUpdateOpCodes = [null, null];  // Alloc space for mask and size\n  if (ngDevMode) {\n    attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);\n  }\n  const textParts = str.split(BINDING_REGEXP);\n  let mask = 0;\n\n  for (let j = 0; j < textParts.length; j++) {\n    const textValue = textParts[j];\n\n    if (j & 1) {\n      // Odd indexes are bindings\n      const bindingIndex = parseInt(textValue, 10);\n      updateOpCodes.push(-1 - bindingIndex);\n      mask = mask | toMaskBit(bindingIndex);\n    } else if (textValue !== '') {\n      // Even indexes are text\n      updateOpCodes.push(textValue);\n    }\n  }\n\n  updateOpCodes.push(\n      destinationNode << I18nUpdateOpCode.SHIFT_REF |\n      (attrName ? I18nUpdateOpCode.Attr : I18nUpdateOpCode.Text));\n  if (attrName) {\n    updateOpCodes.push(attrName, sanitizeFn);\n  }\n  updateOpCodes[0] = mask;\n  updateOpCodes[1] = updateOpCodes.length - 2;\n  return updateOpCodes;\n}\n\nfunction getBindingMask(icuExpression: IcuExpression, mask = 0): number {\n  mask = mask | toMaskBit(icuExpression.mainBinding);\n  let match;\n  for (let i = 0; i < icuExpression.values.length; i++) {\n    const valueArr = icuExpression.values[i];\n    for (let j = 0; j < valueArr.length; j++) {\n      const value = valueArr[j];\n      if (typeof value === 'string') {\n        while (match = BINDING_REGEXP.exec(value)) {\n          mask = mask | toMaskBit(parseInt(match[1], 10));\n        }\n      } else {\n        mask = getBindingMask(value as IcuExpression, mask);\n      }\n    }\n  }\n  return mask;\n}\n\nfunction allocNodeIndex(startIndex: number): number {\n  return startIndex + i18nVarsCount++;\n}\n\n\n/**\n * Convert binding index to mask bit.\n *\n * Each index represents a single bit on the bit-mask. Because bit-mask only has 32 bits, we make\n * the 32nd bit share all masks for all bindings higher than 32. Since it is extremely rare to have\n * more than 32 bindings this will be hit very rarely. The downside of hitting this corner case is\n * that we will execute binding code more often than necessary. (penalty of performance)\n */\nfunction toMaskBit(bindingIndex: number): number {\n  return 1 << Math.min(bindingIndex, 31);\n}\n\nexport function isRootTemplateMessage(subTemplateIndex: number|\n                                      undefined): subTemplateIndex is undefined {\n  return subTemplateIndex === undefined;\n}\n\n\n/**\n * Removes everything inside the sub-templates of a message.\n */\nfunction removeInnerTemplateTranslation(message: string): string {\n  let match;\n  let res = '';\n  let index = 0;\n  let inTemplate = false;\n  let tagMatched;\n\n  while ((match = SUBTEMPLATE_REGEXP.exec(message)) !== null) {\n    if (!inTemplate) {\n      res += message.substring(index, match.index + match[0].length);\n      tagMatched = match[1];\n      inTemplate = true;\n    } else {\n      if (match[0] === `${MARKER}/*${tagMatched}${MARKER}`) {\n        index = match.index;\n        inTemplate = false;\n      }\n    }\n  }\n\n  ngDevMode &&\n      assertEqual(\n          inTemplate, false,\n          `Tag mismatch: unable to find the end of the sub-template in the translation \"${\n              message}\"`);\n\n  res += message.substr(index);\n  return res;\n}\n\n\n/**\n * Extracts a part of a message and removes the rest.\n *\n * This method is used for extracting a part of the message associated with a template. A translated\n * message can span multiple templates.\n *\n * Example:\n * ```\n * <div i18n>Translate <span *ngIf>me</span>!</div>\n * ```\n *\n * @param message The message to crop\n * @param subTemplateIndex Index of the sub-template to extract. If undefined it returns the\n * external template and removes all sub-templates.\n */\nexport function getTranslationForTemplate(message: string, subTemplateIndex?: number) {\n  if (isRootTemplateMessage(subTemplateIndex)) {\n    // We want the root template message, ignore all sub-templates\n    return removeInnerTemplateTranslation(message);\n  } else {\n    // We want a specific sub-template\n    const start =\n        message.indexOf(`:${subTemplateIndex}${MARKER}`) + 2 + subTemplateIndex.toString().length;\n    const end = message.search(new RegExp(`${MARKER}\\\\/\\\\*\\\\d+:${subTemplateIndex}${MARKER}`));\n    return removeInnerTemplateTranslation(message.substring(start, end));\n  }\n}\n\n/**\n * Generate the OpCodes for ICU expressions.\n *\n * @param tIcus\n * @param icuExpression\n * @param startIndex\n * @param expandoStartIndex\n */\nexport function icuStart(\n    tIcus: TIcu[], icuExpression: IcuExpression, startIndex: number,\n    expandoStartIndex: number): void {\n  const createCodes: I18nMutateOpCodes[] = [];\n  const removeCodes: I18nMutateOpCodes[] = [];\n  const updateCodes: I18nUpdateOpCodes[] = [];\n  const vars = [];\n  const childIcus: number[][] = [];\n  const values = icuExpression.values;\n  for (let i = 0; i < values.length; i++) {\n    // Each value is an array of strings & other ICU expressions\n    const valueArr = values[i];\n    const nestedIcus: IcuExpression[] = [];\n    for (let j = 0; j < valueArr.length; j++) {\n      const value = valueArr[j];\n      if (typeof value !== 'string') {\n        // It is an nested ICU expression\n        const icuIndex = nestedIcus.push(value as IcuExpression) - 1;\n        // Replace nested ICU expression by a comment node\n        valueArr[j] = `<!--�${icuIndex}�-->`;\n      }\n    }\n    const icuCase: IcuCase =\n        parseIcuCase(valueArr.join(''), startIndex, nestedIcus, tIcus, expandoStartIndex);\n    createCodes.push(icuCase.create);\n    removeCodes.push(icuCase.remove);\n    updateCodes.push(icuCase.update);\n    vars.push(icuCase.vars);\n    childIcus.push(icuCase.childIcus);\n  }\n  const tIcu: TIcu = {\n    type: icuExpression.type,\n    vars,\n    currentCaseLViewIndex: HEADER_OFFSET +\n        expandoStartIndex  // expandoStartIndex does not include the header so add it.\n        + 1,               // The first item stored is the `<!--ICU #-->` anchor so skip it.\n    childIcus,\n    cases: icuExpression.cases,\n    create: createCodes,\n    remove: removeCodes,\n    update: updateCodes\n  };\n  tIcus.push(tIcu);\n  // Adding the maximum possible of vars needed (based on the cases with the most vars)\n  i18nVarsCount += Math.max(...vars);\n}\n\n/**\n * Parses text containing an ICU expression and produces a JSON object for it.\n * Original code from closure library, modified for Angular.\n *\n * @param pattern Text containing an ICU expression that needs to be parsed.\n *\n */\nexport function parseICUBlock(pattern: string): IcuExpression {\n  const cases = [];\n  const values: (string|IcuExpression)[][] = [];\n  let icuType = IcuType.plural;\n  let mainBinding = 0;\n  pattern = pattern.replace(ICU_BLOCK_REGEXP, function(str: string, binding: string, type: string) {\n    if (type === 'select') {\n      icuType = IcuType.select;\n    } else {\n      icuType = IcuType.plural;\n    }\n    mainBinding = parseInt(binding.substr(1), 10);\n    return '';\n  });\n\n  const parts = extractParts(pattern) as string[];\n  // Looking for (key block)+ sequence. One of the keys has to be \"other\".\n  for (let pos = 0; pos < parts.length;) {\n    let key = parts[pos++].trim();\n    if (icuType === IcuType.plural) {\n      // Key can be \"=x\", we just want \"x\"\n      key = key.replace(/\\s*(?:=)?(\\w+)\\s*/, '$1');\n    }\n    if (key.length) {\n      cases.push(key);\n    }\n\n    const blocks = extractParts(parts[pos++]) as string[];\n    if (cases.length > values.length) {\n      values.push(blocks);\n    }\n  }\n\n  // TODO(ocombe): support ICU expressions in attributes, see #21615\n  return {type: icuType, mainBinding: mainBinding, cases, values};\n}\n\n\n/**\n * Transforms a string template into an HTML template and a list of instructions used to update\n * attributes or nodes that contain bindings.\n *\n * @param unsafeHtml The string to parse\n * @param parentIndex\n * @param nestedIcus\n * @param tIcus\n * @param expandoStartIndex\n */\nfunction parseIcuCase(\n    unsafeHtml: string, parentIndex: number, nestedIcus: IcuExpression[], tIcus: TIcu[],\n    expandoStartIndex: number): IcuCase {\n  const inertBodyHelper = getInertBodyHelper(getDocument());\n  const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);\n  if (!inertBodyElement) {\n    throw new Error('Unable to generate inert body element');\n  }\n  const wrapper = getTemplateContent(inertBodyElement!) as Element || inertBodyElement;\n  const opCodes: IcuCase = {\n    vars: 1,  // allocate space for `TIcu.currentCaseLViewIndex`\n    childIcus: [],\n    create: [],\n    remove: [],\n    update: []\n  };\n  if (ngDevMode) {\n    attachDebugGetter(opCodes.create, i18nMutateOpCodesToString);\n    attachDebugGetter(opCodes.remove, i18nMutateOpCodesToString);\n    attachDebugGetter(opCodes.update, i18nUpdateOpCodesToString);\n  }\n  parseNodes(wrapper.firstChild, opCodes, parentIndex, nestedIcus, tIcus, expandoStartIndex);\n  return opCodes;\n}\n\n/**\n * Breaks pattern into strings and top level {...} blocks.\n * Can be used to break a message into text and ICU expressions, or to break an ICU expression into\n * keys and cases.\n * Original code from closure library, modified for Angular.\n *\n * @param pattern (sub)Pattern to be broken.\n *\n */\nfunction extractParts(pattern: string): (string|IcuExpression)[] {\n  if (!pattern) {\n    return [];\n  }\n\n  let prevPos = 0;\n  const braceStack = [];\n  const results: (string|IcuExpression)[] = [];\n  const braces = /[{}]/g;\n  // lastIndex doesn't get set to 0 so we have to.\n  braces.lastIndex = 0;\n\n  let match;\n  while (match = braces.exec(pattern)) {\n    const pos = match.index;\n    if (match[0] == '}') {\n      braceStack.pop();\n\n      if (braceStack.length == 0) {\n        // End of the block.\n        const block = pattern.substring(prevPos, pos);\n        if (ICU_BLOCK_REGEXP.test(block)) {\n          results.push(parseICUBlock(block));\n        } else {\n          results.push(block);\n        }\n\n        prevPos = pos + 1;\n      }\n    } else {\n      if (braceStack.length == 0) {\n        const substring = pattern.substring(prevPos, pos);\n        results.push(substring);\n        prevPos = pos + 1;\n      }\n      braceStack.push('{');\n    }\n  }\n\n  const substring = pattern.substring(prevPos);\n  results.push(substring);\n  return results;\n}\n\n\n/**\n * Parses a node, its children and its siblings, and generates the mutate & update OpCodes.\n *\n * @param currentNode The first node to parse\n * @param icuCase The data for the ICU expression case that contains those nodes\n * @param parentIndex Index of the current node's parent\n * @param nestedIcus Data for the nested ICU expressions that this case contains\n * @param tIcus Data for all ICU expressions of the current message\n * @param expandoStartIndex Expando start index for the current ICU expression\n */\nexport function parseNodes(\n    currentNode: Node|null, icuCase: IcuCase, parentIndex: number, nestedIcus: IcuExpression[],\n    tIcus: TIcu[], expandoStartIndex: number) {\n  if (currentNode) {\n    const nestedIcusToCreate: [IcuExpression, number][] = [];\n    while (currentNode) {\n      const nextNode: Node|null = currentNode.nextSibling;\n      const newIndex = expandoStartIndex + ++icuCase.vars;\n      switch (currentNode.nodeType) {\n        case Node.ELEMENT_NODE:\n          const element = currentNode as Element;\n          const tagName = element.tagName.toLowerCase();\n          if (!VALID_ELEMENTS.hasOwnProperty(tagName)) {\n            // This isn't a valid element, we won't create an element for it\n            icuCase.vars--;\n          } else {\n            icuCase.create.push(\n                ELEMENT_MARKER, tagName, newIndex,\n                parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n            const elAttrs = element.attributes;\n            for (let i = 0; i < elAttrs.length; i++) {\n              const attr = elAttrs.item(i)!;\n              const lowerAttrName = attr.name.toLowerCase();\n              const hasBinding = !!attr.value.match(BINDING_REGEXP);\n              // we assume the input string is safe, unless it's using a binding\n              if (hasBinding) {\n                if (VALID_ATTRS.hasOwnProperty(lowerAttrName)) {\n                  if (URI_ATTRS[lowerAttrName]) {\n                    addAllToArray(\n                        generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, _sanitizeUrl),\n                        icuCase.update);\n                  } else if (SRCSET_ATTRS[lowerAttrName]) {\n                    addAllToArray(\n                        generateBindingUpdateOpCodes(\n                            attr.value, newIndex, attr.name, sanitizeSrcset),\n                        icuCase.update);\n                  } else {\n                    addAllToArray(\n                        generateBindingUpdateOpCodes(attr.value, newIndex, attr.name),\n                        icuCase.update);\n                  }\n                } else {\n                  ngDevMode &&\n                      console.warn(`WARNING: ignoring unsafe attribute value ${\n                          lowerAttrName} on element ${tagName} (see http://g.co/ng/security#xss)`);\n                }\n              } else {\n                icuCase.create.push(\n                    newIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Attr, attr.name,\n                    attr.value);\n              }\n            }\n            // Parse the children of this node (if any)\n            parseNodes(\n                currentNode.firstChild, icuCase, newIndex, nestedIcus, tIcus, expandoStartIndex);\n            // Remove the parent node after the children\n            icuCase.remove.push(newIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Remove);\n          }\n          break;\n        case Node.TEXT_NODE:\n          const value = currentNode.textContent || '';\n          const hasBinding = value.match(BINDING_REGEXP);\n          icuCase.create.push(\n              hasBinding ? '' : value, newIndex,\n              parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n          icuCase.remove.push(newIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Remove);\n          if (hasBinding) {\n            addAllToArray(generateBindingUpdateOpCodes(value, newIndex), icuCase.update);\n          }\n          break;\n        case Node.COMMENT_NODE:\n          // Check if the comment node is a placeholder for a nested ICU\n          const match = NESTED_ICU.exec(currentNode.textContent || '');\n          if (match) {\n            const nestedIcuIndex = parseInt(match[1], 10);\n            const newLocal = ngDevMode ? `nested ICU ${nestedIcuIndex}` : '';\n            // Create the comment node that will anchor the ICU expression\n            icuCase.create.push(\n                COMMENT_MARKER, newLocal, newIndex,\n                parentIndex << I18nMutateOpCode.SHIFT_PARENT | I18nMutateOpCode.AppendChild);\n            const nestedIcu = nestedIcus[nestedIcuIndex];\n            nestedIcusToCreate.push([nestedIcu, newIndex]);\n          } else {\n            // We do not handle any other type of comment\n            icuCase.vars--;\n          }\n          break;\n        default:\n          // We do not handle any other type of element\n          icuCase.vars--;\n      }\n      currentNode = nextNode!;\n    }\n\n    for (let i = 0; i < nestedIcusToCreate.length; i++) {\n      const nestedIcu = nestedIcusToCreate[i][0];\n      const nestedIcuNodeIndex = nestedIcusToCreate[i][1];\n      icuStart(tIcus, nestedIcu, nestedIcuNodeIndex, expandoStartIndex + icuCase.vars);\n      // Since this is recursive, the last TIcu that was pushed is the one we want\n      const nestTIcuIndex = tIcus.length - 1;\n      icuCase.vars += Math.max(...tIcus[nestTIcuIndex].vars);\n      icuCase.childIcus.push(nestTIcuIndex);\n      const mask = getBindingMask(nestedIcu);\n      icuCase.update.push(\n          toMaskBit(nestedIcu.mainBinding),  // mask of the main binding\n          3,                                 // skip 3 opCodes if not changed\n          -1 - nestedIcu.mainBinding,\n          nestedIcuNodeIndex << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuSwitch,\n          // FIXME(misko): Index should be part of the opcode\n          nestTIcuIndex,\n          mask,  // mask of all the bindings of this ICU expression\n          2,     // skip 2 opCodes if not changed\n          nestedIcuNodeIndex << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuUpdate,\n          nestTIcuIndex);\n      icuCase.remove.push(\n          nestTIcuIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.RemoveNestedIcu,\n          // FIXME(misko): Index should be part of the opcode\n          nestedIcuNodeIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Remove);\n    }\n  }\n}\n"]}