source: trip-planner-front/node_modules/@angular/core/esm2015/src/render3/i18n/i18n_parse.js@ 6a3a178

Last change on this file since 6a3a178 was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 92.1 KB
Line 
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 */
8import '../../util/ng_dev_mode';
9import '../../util/ng_i18n_closure_mode';
10import { getTemplateContent, SRCSET_ATTRS, URI_ATTRS, VALID_ATTRS, VALID_ELEMENTS } from '../../sanitization/html_sanitizer';
11import { getInertBodyHelper } from '../../sanitization/inert_body';
12import { _sanitizeUrl, sanitizeSrcset } from '../../sanitization/url_sanitizer';
13import { assertDefined, assertEqual, assertGreaterThanOrEqual, assertOneOf, assertString } from '../../util/assert';
14import { loadIcuContainerVisitor } from '../instructions/i18n_icu_container_visitor';
15import { allocExpando, createTNodeAtIndex } from '../instructions/shared';
16import { getDocument } from '../interfaces/document';
17import { ELEMENT_MARKER, I18nCreateOpCode, ICU_MARKER } from '../interfaces/i18n';
18import { HEADER_OFFSET } from '../interfaces/view';
19import { getCurrentParentTNode, getCurrentTNode, setCurrentTNode } from '../state';
20import { attachDebugGetter } from '../util/debug_utils';
21import { i18nCreateOpCodesToString, i18nRemoveOpCodesToString, i18nUpdateOpCodesToString, icuCreateOpCodesToString } from './i18n_debug';
22import { addTNodeAndUpdateInsertBeforeIndex } from './i18n_insert_before_index';
23import { ensureIcuContainerVisitorLoaded } from './i18n_tree_shaking';
24import { createTNodePlaceholder, icuCreateOpCode, setTIcu, setTNodeInsertBeforeIndex } from './i18n_util';
25const BINDING_REGEXP = /�(\d+):?\d*�/gi;
26const ICU_REGEXP = /({\s*�\d+:?\d*�\s*,\s*\S{6}\s*,[\s\S]*})/gi;
27const NESTED_ICU = /�(\d+)�/;
28const ICU_BLOCK_REGEXP = /^\s*(�\d+:?\d*�)\s*,\s*(select|plural)\s*,/;
29const MARKER = `�`;
30const SUBTEMPLATE_REGEXP = /�\/?\*(\d+:\d+)�/gi;
31const PH_REGEXP = /�(\/?[#*]\d+):?\d*�/gi;
32/**
33 * Angular Dart introduced &ngsp; as a placeholder for non-removable space, see:
34 * https://github.com/dart-lang/angular/blob/0bb611387d29d65b5af7f9d2515ab571fd3fbee4/_tests/test/compiler/preserve_whitespace_test.dart#L25-L32
35 * In Angular Dart &ngsp; is converted to the 0xE500 PUA (Private Use Areas) unicode character
36 * and later on replaced by a space. We are re-implementing the same idea here, since translations
37 * might contain this special character.
38 */
39const NGSP_UNICODE_REGEXP = /\uE500/g;
40function replaceNgsp(value) {
41 return value.replace(NGSP_UNICODE_REGEXP, ' ');
42}
43/**
44 * Create dynamic nodes from i18n translation block.
45 *
46 * - Text nodes are created synchronously
47 * - TNodes are linked into tree lazily
48 *
49 * @param tView Current `TView`
50 * @parentTNodeIndex index to the parent TNode of this i18n block
51 * @param lView Current `LView`
52 * @param index Index of `ɵɵi18nStart` instruction.
53 * @param message Message to translate.
54 * @param subTemplateIndex Index into the sub template of message translation. (ie in case of
55 * `ngIf`) (-1 otherwise)
56 */
57export function i18nStartFirstCreatePass(tView, parentTNodeIndex, lView, index, message, subTemplateIndex) {
58 const rootTNode = getCurrentParentTNode();
59 const createOpCodes = [];
60 const updateOpCodes = [];
61 const existingTNodeStack = [[]];
62 if (ngDevMode) {
63 attachDebugGetter(createOpCodes, i18nCreateOpCodesToString);
64 attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
65 }
66 message = getTranslationForTemplate(message, subTemplateIndex);
67 const msgParts = replaceNgsp(message).split(PH_REGEXP);
68 for (let i = 0; i < msgParts.length; i++) {
69 let value = msgParts[i];
70 if ((i & 1) === 0) {
71 // Even indexes are text (including bindings & ICU expressions)
72 const parts = i18nParseTextIntoPartsAndICU(value);
73 for (let j = 0; j < parts.length; j++) {
74 let part = parts[j];
75 if ((j & 1) === 0) {
76 // `j` is odd therefore `part` is string
77 const text = part;
78 ngDevMode && assertString(text, 'Parsed ICU part should be string');
79 if (text !== '') {
80 i18nStartFirstCreatePassProcessTextNode(tView, rootTNode, existingTNodeStack[0], createOpCodes, updateOpCodes, lView, text);
81 }
82 }
83 else {
84 // `j` is Even therefor `part` is an `ICUExpression`
85 const icuExpression = part;
86 // Verify that ICU expression has the right shape. Translations might contain invalid
87 // constructions (while original messages were correct), so ICU parsing at runtime may
88 // not succeed (thus `icuExpression` remains a string).
89 // Note: we intentionally retain the error here by not using `ngDevMode`, because
90 // the value can change based on the locale and users aren't guaranteed to hit
91 // an invalid string while they're developing.
92 if (typeof icuExpression !== 'object') {
93 throw new Error(`Unable to parse ICU expression in "${message}" message.`);
94 }
95 const icuContainerTNode = createTNodeAndAddOpCode(tView, rootTNode, existingTNodeStack[0], lView, createOpCodes, ngDevMode ? `ICU ${index}:${icuExpression.mainBinding}` : '', true);
96 const icuNodeIndex = icuContainerTNode.index;
97 ngDevMode &&
98 assertGreaterThanOrEqual(icuNodeIndex, HEADER_OFFSET, 'Index must be in absolute LView offset');
99 icuStart(tView, lView, updateOpCodes, parentTNodeIndex, icuExpression, icuNodeIndex);
100 }
101 }
102 }
103 else {
104 // Odd indexes are placeholders (elements and sub-templates)
105 // At this point value is something like: '/#1:2' (originally coming from '�/#1:2�')
106 const isClosing = value.charCodeAt(0) === 47 /* SLASH */;
107 const type = value.charCodeAt(isClosing ? 1 : 0);
108 ngDevMode && assertOneOf(type, 42 /* STAR */, 35 /* HASH */);
109 const index = HEADER_OFFSET + Number.parseInt(value.substring((isClosing ? 2 : 1)));
110 if (isClosing) {
111 existingTNodeStack.shift();
112 setCurrentTNode(getCurrentParentTNode(), false);
113 }
114 else {
115 const tNode = createTNodePlaceholder(tView, existingTNodeStack[0], index);
116 existingTNodeStack.unshift([]);
117 setCurrentTNode(tNode, true);
118 }
119 }
120 }
121 tView.data[index] = {
122 create: createOpCodes,
123 update: updateOpCodes,
124 };
125}
126/**
127 * Allocate space in i18n Range add create OpCode instruction to crete a text or comment node.
128 *
129 * @param tView Current `TView` needed to allocate space in i18n range.
130 * @param rootTNode Root `TNode` of the i18n block. This node determines if the new TNode will be
131 * added as part of the `i18nStart` instruction or as part of the `TNode.insertBeforeIndex`.
132 * @param existingTNodes internal state for `addTNodeAndUpdateInsertBeforeIndex`.
133 * @param lView Current `LView` needed to allocate space in i18n range.
134 * @param createOpCodes Array storing `I18nCreateOpCodes` where new opCodes will be added.
135 * @param text Text to be added when the `Text` or `Comment` node will be created.
136 * @param isICU true if a `Comment` node for ICU (instead of `Text`) node should be created.
137 */
138function createTNodeAndAddOpCode(tView, rootTNode, existingTNodes, lView, createOpCodes, text, isICU) {
139 const i18nNodeIdx = allocExpando(tView, lView, 1, null);
140 let opCode = i18nNodeIdx << I18nCreateOpCode.SHIFT;
141 let parentTNode = getCurrentParentTNode();
142 if (rootTNode === parentTNode) {
143 // FIXME(misko): A null `parentTNode` should represent when we fall of the `LView` boundary.
144 // (there is no parent), but in some circumstances (because we are inconsistent about how we set
145 // `previousOrParentTNode`) it could point to `rootTNode` So this is a work around.
146 parentTNode = null;
147 }
148 if (parentTNode === null) {
149 // If we don't have a parent that means that we can eagerly add nodes.
150 // If we have a parent than these nodes can't be added now (as the parent has not been created
151 // yet) and instead the `parentTNode` is responsible for adding it. See
152 // `TNode.insertBeforeIndex`
153 opCode |= I18nCreateOpCode.APPEND_EAGERLY;
154 }
155 if (isICU) {
156 opCode |= I18nCreateOpCode.COMMENT;
157 ensureIcuContainerVisitorLoaded(loadIcuContainerVisitor);
158 }
159 createOpCodes.push(opCode, text === null ? '' : text);
160 // We store `{{?}}` so that when looking at debug `TNodeType.template` we can see where the
161 // bindings are.
162 const tNode = createTNodeAtIndex(tView, i18nNodeIdx, isICU ? 32 /* Icu */ : 1 /* Text */, text === null ? (ngDevMode ? '{{?}}' : '') : text, null);
163 addTNodeAndUpdateInsertBeforeIndex(existingTNodes, tNode);
164 const tNodeIdx = tNode.index;
165 setCurrentTNode(tNode, false /* Text nodes are self closing */);
166 if (parentTNode !== null && rootTNode !== parentTNode) {
167 // We are a child of deeper node (rather than a direct child of `i18nStart` instruction.)
168 // We have to make sure to add ourselves to the parent.
169 setTNodeInsertBeforeIndex(parentTNode, tNodeIdx);
170 }
171 return tNode;
172}
173/**
174 * Processes text node in i18n block.
175 *
176 * Text nodes can have:
177 * - Create instruction in `createOpCodes` for creating the text node.
178 * - Allocate spec for text node in i18n range of `LView`
179 * - If contains binding:
180 * - bindings => allocate space in i18n range of `LView` to store the binding value.
181 * - populate `updateOpCodes` with update instructions.
182 *
183 * @param tView Current `TView`
184 * @param rootTNode Root `TNode` of the i18n block. This node determines if the new TNode will
185 * be added as part of the `i18nStart` instruction or as part of the
186 * `TNode.insertBeforeIndex`.
187 * @param existingTNodes internal state for `addTNodeAndUpdateInsertBeforeIndex`.
188 * @param createOpCodes Location where the creation OpCodes will be stored.
189 * @param lView Current `LView`
190 * @param text The translated text (which may contain binding)
191 */
192function i18nStartFirstCreatePassProcessTextNode(tView, rootTNode, existingTNodes, createOpCodes, updateOpCodes, lView, text) {
193 const hasBinding = text.match(BINDING_REGEXP);
194 const tNode = createTNodeAndAddOpCode(tView, rootTNode, existingTNodes, lView, createOpCodes, hasBinding ? null : text, false);
195 if (hasBinding) {
196 generateBindingUpdateOpCodes(updateOpCodes, text, tNode.index, null, 0, null);
197 }
198}
199/**
200 * See `i18nAttributes` above.
201 */
202export function i18nAttributesFirstPass(tView, index, values) {
203 const previousElement = getCurrentTNode();
204 const previousElementIndex = previousElement.index;
205 const updateOpCodes = [];
206 if (ngDevMode) {
207 attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
208 }
209 if (tView.firstCreatePass && tView.data[index] === null) {
210 for (let i = 0; i < values.length; i += 2) {
211 const attrName = values[i];
212 const message = values[i + 1];
213 if (message !== '') {
214 // Check if attribute value contains an ICU and throw an error if that's the case.
215 // ICUs in element attributes are not supported.
216 // Note: we intentionally retain the error here by not using `ngDevMode`, because
217 // the `value` can change based on the locale and users aren't guaranteed to hit
218 // an invalid string while they're developing.
219 if (ICU_REGEXP.test(message)) {
220 throw new Error(`ICU expressions are not supported in attributes. Message: "${message}".`);
221 }
222 // i18n attributes that hit this code path are guaranteed to have bindings, because
223 // the compiler treats static i18n attributes as regular attribute bindings.
224 // Since this may not be the first i18n attribute on this element we need to pass in how
225 // many previous bindings there have already been.
226 generateBindingUpdateOpCodes(updateOpCodes, message, previousElementIndex, attrName, countBindings(updateOpCodes), null);
227 }
228 }
229 tView.data[index] = updateOpCodes;
230 }
231}
232/**
233 * Generate the OpCodes to update the bindings of a string.
234 *
235 * @param updateOpCodes Place where the update opcodes will be stored.
236 * @param str The string containing the bindings.
237 * @param destinationNode Index of the destination node which will receive the binding.
238 * @param attrName Name of the attribute, if the string belongs to an attribute.
239 * @param sanitizeFn Sanitization function used to sanitize the string after update, if necessary.
240 * @param bindingStart The lView index of the next expression that can be bound via an opCode.
241 * @returns The mask value for these bindings
242 */
243function generateBindingUpdateOpCodes(updateOpCodes, str, destinationNode, attrName, bindingStart, sanitizeFn) {
244 ngDevMode &&
245 assertGreaterThanOrEqual(destinationNode, HEADER_OFFSET, 'Index must be in absolute LView offset');
246 const maskIndex = updateOpCodes.length; // Location of mask
247 const sizeIndex = maskIndex + 1; // location of size for skipping
248 updateOpCodes.push(null, null); // Alloc space for mask and size
249 const startIndex = maskIndex + 2; // location of first allocation.
250 if (ngDevMode) {
251 attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);
252 }
253 const textParts = str.split(BINDING_REGEXP);
254 let mask = 0;
255 for (let j = 0; j < textParts.length; j++) {
256 const textValue = textParts[j];
257 if (j & 1) {
258 // Odd indexes are bindings
259 const bindingIndex = bindingStart + parseInt(textValue, 10);
260 updateOpCodes.push(-1 - bindingIndex);
261 mask = mask | toMaskBit(bindingIndex);
262 }
263 else if (textValue !== '') {
264 // Even indexes are text
265 updateOpCodes.push(textValue);
266 }
267 }
268 updateOpCodes.push(destinationNode << 2 /* SHIFT_REF */ |
269 (attrName ? 1 /* Attr */ : 0 /* Text */));
270 if (attrName) {
271 updateOpCodes.push(attrName, sanitizeFn);
272 }
273 updateOpCodes[maskIndex] = mask;
274 updateOpCodes[sizeIndex] = updateOpCodes.length - startIndex;
275 return mask;
276}
277/**
278 * Count the number of bindings in the given `opCodes`.
279 *
280 * It could be possible to speed this up, by passing the number of bindings found back from
281 * `generateBindingUpdateOpCodes()` to `i18nAttributesFirstPass()` but this would then require more
282 * complexity in the code and/or transient objects to be created.
283 *
284 * Since this function is only called once when the template is instantiated, is trivial in the
285 * first instance (since `opCodes` will be an empty array), and it is not common for elements to
286 * contain multiple i18n bound attributes, it seems like this is a reasonable compromise.
287 */
288function countBindings(opCodes) {
289 let count = 0;
290 for (let i = 0; i < opCodes.length; i++) {
291 const opCode = opCodes[i];
292 // Bindings are negative numbers.
293 if (typeof opCode === 'number' && opCode < 0) {
294 count++;
295 }
296 }
297 return count;
298}
299/**
300 * Convert binding index to mask bit.
301 *
302 * Each index represents a single bit on the bit-mask. Because bit-mask only has 32 bits, we make
303 * the 32nd bit share all masks for all bindings higher than 32. Since it is extremely rare to
304 * have more than 32 bindings this will be hit very rarely. The downside of hitting this corner
305 * case is that we will execute binding code more often than necessary. (penalty of performance)
306 */
307function toMaskBit(bindingIndex) {
308 return 1 << Math.min(bindingIndex, 31);
309}
310export function isRootTemplateMessage(subTemplateIndex) {
311 return subTemplateIndex === -1;
312}
313/**
314 * Removes everything inside the sub-templates of a message.
315 */
316function removeInnerTemplateTranslation(message) {
317 let match;
318 let res = '';
319 let index = 0;
320 let inTemplate = false;
321 let tagMatched;
322 while ((match = SUBTEMPLATE_REGEXP.exec(message)) !== null) {
323 if (!inTemplate) {
324 res += message.substring(index, match.index + match[0].length);
325 tagMatched = match[1];
326 inTemplate = true;
327 }
328 else {
329 if (match[0] === `${MARKER}/*${tagMatched}${MARKER}`) {
330 index = match.index;
331 inTemplate = false;
332 }
333 }
334 }
335 ngDevMode &&
336 assertEqual(inTemplate, false, `Tag mismatch: unable to find the end of the sub-template in the translation "${message}"`);
337 res += message.substr(index);
338 return res;
339}
340/**
341 * Extracts a part of a message and removes the rest.
342 *
343 * This method is used for extracting a part of the message associated with a template. A
344 * translated message can span multiple templates.
345 *
346 * Example:
347 * ```
348 * <div i18n>Translate <span *ngIf>me</span>!</div>
349 * ```
350 *
351 * @param message The message to crop
352 * @param subTemplateIndex Index of the sub-template to extract. If undefined it returns the
353 * external template and removes all sub-templates.
354 */
355export function getTranslationForTemplate(message, subTemplateIndex) {
356 if (isRootTemplateMessage(subTemplateIndex)) {
357 // We want the root template message, ignore all sub-templates
358 return removeInnerTemplateTranslation(message);
359 }
360 else {
361 // We want a specific sub-template
362 const start = message.indexOf(`:${subTemplateIndex}${MARKER}`) + 2 + subTemplateIndex.toString().length;
363 const end = message.search(new RegExp(`${MARKER}\\/\\*\\d+:${subTemplateIndex}${MARKER}`));
364 return removeInnerTemplateTranslation(message.substring(start, end));
365 }
366}
367/**
368 * Generate the OpCodes for ICU expressions.
369 *
370 * @param icuExpression
371 * @param index Index where the anchor is stored and an optional `TIcuContainerNode`
372 * - `lView[anchorIdx]` points to a `Comment` node representing the anchor for the ICU.
373 * - `tView.data[anchorIdx]` points to the `TIcuContainerNode` if ICU is root (`null` otherwise)
374 */
375export function icuStart(tView, lView, updateOpCodes, parentIdx, icuExpression, anchorIdx) {
376 ngDevMode && assertDefined(icuExpression, 'ICU expression must be defined');
377 let bindingMask = 0;
378 const tIcu = {
379 type: icuExpression.type,
380 currentCaseLViewIndex: allocExpando(tView, lView, 1, null),
381 anchorIdx,
382 cases: [],
383 create: [],
384 remove: [],
385 update: []
386 };
387 addUpdateIcuSwitch(updateOpCodes, icuExpression, anchorIdx);
388 setTIcu(tView, anchorIdx, tIcu);
389 const values = icuExpression.values;
390 for (let i = 0; i < values.length; i++) {
391 // Each value is an array of strings & other ICU expressions
392 const valueArr = values[i];
393 const nestedIcus = [];
394 for (let j = 0; j < valueArr.length; j++) {
395 const value = valueArr[j];
396 if (typeof value !== 'string') {
397 // It is an nested ICU expression
398 const icuIndex = nestedIcus.push(value) - 1;
399 // Replace nested ICU expression by a comment node
400 valueArr[j] = `<!--�${icuIndex}�-->`;
401 }
402 }
403 bindingMask = parseIcuCase(tView, tIcu, lView, updateOpCodes, parentIdx, icuExpression.cases[i], valueArr.join(''), nestedIcus) |
404 bindingMask;
405 }
406 if (bindingMask) {
407 addUpdateIcuUpdate(updateOpCodes, bindingMask, anchorIdx);
408 }
409}
410/**
411 * Parses text containing an ICU expression and produces a JSON object for it.
412 * Original code from closure library, modified for Angular.
413 *
414 * @param pattern Text containing an ICU expression that needs to be parsed.
415 *
416 */
417export function parseICUBlock(pattern) {
418 const cases = [];
419 const values = [];
420 let icuType = 1 /* plural */;
421 let mainBinding = 0;
422 pattern = pattern.replace(ICU_BLOCK_REGEXP, function (str, binding, type) {
423 if (type === 'select') {
424 icuType = 0 /* select */;
425 }
426 else {
427 icuType = 1 /* plural */;
428 }
429 mainBinding = parseInt(binding.substr(1), 10);
430 return '';
431 });
432 const parts = i18nParseTextIntoPartsAndICU(pattern);
433 // Looking for (key block)+ sequence. One of the keys has to be "other".
434 for (let pos = 0; pos < parts.length;) {
435 let key = parts[pos++].trim();
436 if (icuType === 1 /* plural */) {
437 // Key can be "=x", we just want "x"
438 key = key.replace(/\s*(?:=)?(\w+)\s*/, '$1');
439 }
440 if (key.length) {
441 cases.push(key);
442 }
443 const blocks = i18nParseTextIntoPartsAndICU(parts[pos++]);
444 if (cases.length > values.length) {
445 values.push(blocks);
446 }
447 }
448 // TODO(ocombe): support ICU expressions in attributes, see #21615
449 return { type: icuType, mainBinding: mainBinding, cases, values };
450}
451/**
452 * Breaks pattern into strings and top level {...} blocks.
453 * Can be used to break a message into text and ICU expressions, or to break an ICU expression
454 * into keys and cases. Original code from closure library, modified for Angular.
455 *
456 * @param pattern (sub)Pattern to be broken.
457 * @returns An `Array<string|IcuExpression>` where:
458 * - odd positions: `string` => text between ICU expressions
459 * - even positions: `ICUExpression` => ICU expression parsed into `ICUExpression` record.
460 */
461export function i18nParseTextIntoPartsAndICU(pattern) {
462 if (!pattern) {
463 return [];
464 }
465 let prevPos = 0;
466 const braceStack = [];
467 const results = [];
468 const braces = /[{}]/g;
469 // lastIndex doesn't get set to 0 so we have to.
470 braces.lastIndex = 0;
471 let match;
472 while (match = braces.exec(pattern)) {
473 const pos = match.index;
474 if (match[0] == '}') {
475 braceStack.pop();
476 if (braceStack.length == 0) {
477 // End of the block.
478 const block = pattern.substring(prevPos, pos);
479 if (ICU_BLOCK_REGEXP.test(block)) {
480 results.push(parseICUBlock(block));
481 }
482 else {
483 results.push(block);
484 }
485 prevPos = pos + 1;
486 }
487 }
488 else {
489 if (braceStack.length == 0) {
490 const substring = pattern.substring(prevPos, pos);
491 results.push(substring);
492 prevPos = pos + 1;
493 }
494 braceStack.push('{');
495 }
496 }
497 const substring = pattern.substring(prevPos);
498 results.push(substring);
499 return results;
500}
501/**
502 * Parses a node, its children and its siblings, and generates the mutate & update OpCodes.
503 *
504 */
505export function parseIcuCase(tView, tIcu, lView, updateOpCodes, parentIdx, caseName, unsafeCaseHtml, nestedIcus) {
506 const create = [];
507 const remove = [];
508 const update = [];
509 if (ngDevMode) {
510 attachDebugGetter(create, icuCreateOpCodesToString);
511 attachDebugGetter(remove, i18nRemoveOpCodesToString);
512 attachDebugGetter(update, i18nUpdateOpCodesToString);
513 }
514 tIcu.cases.push(caseName);
515 tIcu.create.push(create);
516 tIcu.remove.push(remove);
517 tIcu.update.push(update);
518 const inertBodyHelper = getInertBodyHelper(getDocument());
519 const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeCaseHtml);
520 ngDevMode && assertDefined(inertBodyElement, 'Unable to generate inert body element');
521 const inertRootNode = getTemplateContent(inertBodyElement) || inertBodyElement;
522 if (inertRootNode) {
523 return walkIcuTree(tView, tIcu, lView, updateOpCodes, create, remove, update, inertRootNode, parentIdx, nestedIcus, 0);
524 }
525 else {
526 return 0;
527 }
528}
529function walkIcuTree(tView, tIcu, lView, sharedUpdateOpCodes, create, remove, update, parentNode, parentIdx, nestedIcus, depth) {
530 let bindingMask = 0;
531 let currentNode = parentNode.firstChild;
532 while (currentNode) {
533 const newIndex = allocExpando(tView, lView, 1, null);
534 switch (currentNode.nodeType) {
535 case Node.ELEMENT_NODE:
536 const element = currentNode;
537 const tagName = element.tagName.toLowerCase();
538 if (VALID_ELEMENTS.hasOwnProperty(tagName)) {
539 addCreateNodeAndAppend(create, ELEMENT_MARKER, tagName, parentIdx, newIndex);
540 tView.data[newIndex] = tagName;
541 const elAttrs = element.attributes;
542 for (let i = 0; i < elAttrs.length; i++) {
543 const attr = elAttrs.item(i);
544 const lowerAttrName = attr.name.toLowerCase();
545 const hasBinding = !!attr.value.match(BINDING_REGEXP);
546 // we assume the input string is safe, unless it's using a binding
547 if (hasBinding) {
548 if (VALID_ATTRS.hasOwnProperty(lowerAttrName)) {
549 if (URI_ATTRS[lowerAttrName]) {
550 generateBindingUpdateOpCodes(update, attr.value, newIndex, attr.name, 0, _sanitizeUrl);
551 }
552 else if (SRCSET_ATTRS[lowerAttrName]) {
553 generateBindingUpdateOpCodes(update, attr.value, newIndex, attr.name, 0, sanitizeSrcset);
554 }
555 else {
556 generateBindingUpdateOpCodes(update, attr.value, newIndex, attr.name, 0, null);
557 }
558 }
559 else {
560 ngDevMode &&
561 console.warn(`WARNING: ignoring unsafe attribute value ` +
562 `${lowerAttrName} on element ${tagName} ` +
563 `(see https://g.co/ng/security#xss)`);
564 }
565 }
566 else {
567 addCreateAttribute(create, newIndex, attr);
568 }
569 }
570 // Parse the children of this node (if any)
571 bindingMask = walkIcuTree(tView, tIcu, lView, sharedUpdateOpCodes, create, remove, update, currentNode, newIndex, nestedIcus, depth + 1) |
572 bindingMask;
573 addRemoveNode(remove, newIndex, depth);
574 }
575 break;
576 case Node.TEXT_NODE:
577 const value = currentNode.textContent || '';
578 const hasBinding = value.match(BINDING_REGEXP);
579 addCreateNodeAndAppend(create, null, hasBinding ? '' : value, parentIdx, newIndex);
580 addRemoveNode(remove, newIndex, depth);
581 if (hasBinding) {
582 bindingMask =
583 generateBindingUpdateOpCodes(update, value, newIndex, null, 0, null) | bindingMask;
584 }
585 break;
586 case Node.COMMENT_NODE:
587 // Check if the comment node is a placeholder for a nested ICU
588 const isNestedIcu = NESTED_ICU.exec(currentNode.textContent || '');
589 if (isNestedIcu) {
590 const nestedIcuIndex = parseInt(isNestedIcu[1], 10);
591 const icuExpression = nestedIcus[nestedIcuIndex];
592 // Create the comment node that will anchor the ICU expression
593 addCreateNodeAndAppend(create, ICU_MARKER, ngDevMode ? `nested ICU ${nestedIcuIndex}` : '', parentIdx, newIndex);
594 icuStart(tView, lView, sharedUpdateOpCodes, parentIdx, icuExpression, newIndex);
595 addRemoveNestedIcu(remove, newIndex, depth);
596 }
597 break;
598 }
599 currentNode = currentNode.nextSibling;
600 }
601 return bindingMask;
602}
603function addRemoveNode(remove, index, depth) {
604 if (depth === 0) {
605 remove.push(index);
606 }
607}
608function addRemoveNestedIcu(remove, index, depth) {
609 if (depth === 0) {
610 remove.push(~index); // remove ICU at `index`
611 remove.push(index); // remove ICU comment at `index`
612 }
613}
614function addUpdateIcuSwitch(update, icuExpression, index) {
615 update.push(toMaskBit(icuExpression.mainBinding), 2, -1 - icuExpression.mainBinding, index << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */);
616}
617function addUpdateIcuUpdate(update, bindingMask, index) {
618 update.push(bindingMask, 1, index << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */);
619}
620function addCreateNodeAndAppend(create, marker, text, appendToParentIdx, createAtIdx) {
621 if (marker !== null) {
622 create.push(marker);
623 }
624 create.push(text, createAtIdx, icuCreateOpCode(0 /* AppendChild */, appendToParentIdx, createAtIdx));
625}
626function addCreateAttribute(create, newIndex, attr) {
627 create.push(newIndex << 1 /* SHIFT_REF */ | 1 /* Attr */, attr.name, attr.value);
628}
629//# 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 {assertDefined, assertEqual, assertGreaterThanOrEqual, assertOneOf, assertString} from '../../util/assert';\nimport {CharCode} from '../../util/char_code';\nimport {loadIcuContainerVisitor} from '../instructions/i18n_icu_container_visitor';\nimport {allocExpando, createTNodeAtIndex} from '../instructions/shared';\nimport {getDocument} from '../interfaces/document';\nimport {ELEMENT_MARKER, I18nCreateOpCode, I18nCreateOpCodes, I18nRemoveOpCodes, I18nUpdateOpCode, I18nUpdateOpCodes, ICU_MARKER, IcuCreateOpCode, IcuCreateOpCodes, IcuExpression, IcuType, TI18n, TIcu} from '../interfaces/i18n';\nimport {TNode, TNodeType} from '../interfaces/node';\nimport {SanitizerFn} from '../interfaces/sanitization';\nimport {HEADER_OFFSET, LView, TView} from '../interfaces/view';\nimport {getCurrentParentTNode, getCurrentTNode, setCurrentTNode} from '../state';\nimport {attachDebugGetter} from '../util/debug_utils';\n\nimport {i18nCreateOpCodesToString, i18nRemoveOpCodesToString, i18nUpdateOpCodesToString, icuCreateOpCodesToString} from './i18n_debug';\nimport {addTNodeAndUpdateInsertBeforeIndex} from './i18n_insert_before_index';\nimport {ensureIcuContainerVisitorLoaded} from './i18n_tree_shaking';\nimport {createTNodePlaceholder, icuCreateOpCode, setTIcu, setTNodeInsertBeforeIndex} from './i18n_util';\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\nconst MARKER = `�`;\nconst SUBTEMPLATE_REGEXP = /�\\/?\\*(\\d+:\\d+)�/gi;\nconst PH_REGEXP = /�(\\/?[#*]\\d+):?\\d*�/gi;\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 * Create dynamic nodes from i18n translation block.\n *\n * - Text nodes are created synchronously\n * - TNodes are linked into tree lazily\n *\n * @param tView Current `TView`\n * @parentTNodeIndex index to the parent TNode of this i18n block\n * @param lView Current `LView`\n * @param index Index of `ɵɵi18nStart` instruction.\n * @param message Message to translate.\n * @param subTemplateIndex Index into the sub template of message translation. (ie in case of\n *     `ngIf`) (-1 otherwise)\n */\nexport function i18nStartFirstCreatePass(\n    tView: TView, parentTNodeIndex: number, lView: LView, index: number, message: string,\n    subTemplateIndex: number) {\n  const rootTNode = getCurrentParentTNode();\n  const createOpCodes: I18nCreateOpCodes = [] as any;\n  const updateOpCodes: I18nUpdateOpCodes = [] as any;\n  const existingTNodeStack: TNode[][] = [[]];\n  if (ngDevMode) {\n    attachDebugGetter(createOpCodes, i18nCreateOpCodesToString);\n    attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);\n  }\n\n  message = getTranslationForTemplate(message, subTemplateIndex);\n  const msgParts = replaceNgsp(message).split(PH_REGEXP);\n  for (let i = 0; i < msgParts.length; i++) {\n    let value = msgParts[i];\n    if ((i & 1) === 0) {\n      // Even indexes are text (including bindings & ICU expressions)\n      const parts = i18nParseTextIntoPartsAndICU(value);\n      for (let j = 0; j < parts.length; j++) {\n        let part = parts[j];\n        if ((j & 1) === 0) {\n          // `j` is odd therefore `part` is string\n          const text = part as string;\n          ngDevMode && assertString(text, 'Parsed ICU part should be string');\n          if (text !== '') {\n            i18nStartFirstCreatePassProcessTextNode(\n                tView, rootTNode, existingTNodeStack[0], createOpCodes, updateOpCodes, lView, text);\n          }\n        } else {\n          // `j` is Even therefor `part` is an `ICUExpression`\n          const icuExpression: IcuExpression = part as IcuExpression;\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          // Note: we intentionally retain the error here by not using `ngDevMode`, because\n          // the value can change based on the locale and users aren't guaranteed to hit\n          // an invalid string while they're developing.\n          if (typeof icuExpression !== 'object') {\n            throw new Error(`Unable to parse ICU expression in \"${message}\" message.`);\n          }\n          const icuContainerTNode = createTNodeAndAddOpCode(\n              tView, rootTNode, existingTNodeStack[0], lView, createOpCodes,\n              ngDevMode ? `ICU ${index}:${icuExpression.mainBinding}` : '', true);\n          const icuNodeIndex = icuContainerTNode.index;\n          ngDevMode &&\n              assertGreaterThanOrEqual(\n                  icuNodeIndex, HEADER_OFFSET, 'Index must be in absolute LView offset');\n          icuStart(tView, lView, updateOpCodes, parentTNodeIndex, icuExpression, icuNodeIndex);\n        }\n      }\n    } else {\n      // Odd indexes are placeholders (elements and sub-templates)\n      // At this point value is something like: '/#1:2' (originally coming from '�/#1:2�')\n      const isClosing = value.charCodeAt(0) === CharCode.SLASH;\n      const type = value.charCodeAt(isClosing ? 1 : 0);\n      ngDevMode && assertOneOf(type, CharCode.STAR, CharCode.HASH);\n      const index = HEADER_OFFSET + Number.parseInt(value.substring((isClosing ? 2 : 1)));\n      if (isClosing) {\n        existingTNodeStack.shift();\n        setCurrentTNode(getCurrentParentTNode()!, false);\n      } else {\n        const tNode = createTNodePlaceholder(tView, existingTNodeStack[0], index);\n        existingTNodeStack.unshift([]);\n        setCurrentTNode(tNode, true);\n      }\n    }\n  }\n\n  tView.data[index] = <TI18n>{\n    create: createOpCodes,\n    update: updateOpCodes,\n  };\n}\n\n/**\n * Allocate space in i18n Range add create OpCode instruction to crete a text or comment node.\n *\n * @param tView Current `TView` needed to allocate space in i18n range.\n * @param rootTNode Root `TNode` of the i18n block. This node determines if the new TNode will be\n *     added as part of the `i18nStart` instruction or as part of the `TNode.insertBeforeIndex`.\n * @param existingTNodes internal state for `addTNodeAndUpdateInsertBeforeIndex`.\n * @param lView Current `LView` needed to allocate space in i18n range.\n * @param createOpCodes Array storing `I18nCreateOpCodes` where new opCodes will be added.\n * @param text Text to be added when the `Text` or `Comment` node will be created.\n * @param isICU true if a `Comment` node for ICU (instead of `Text`) node should be created.\n */\nfunction createTNodeAndAddOpCode(\n    tView: TView, rootTNode: TNode|null, existingTNodes: TNode[], lView: LView,\n    createOpCodes: I18nCreateOpCodes, text: string|null, isICU: boolean): TNode {\n  const i18nNodeIdx = allocExpando(tView, lView, 1, null);\n  let opCode = i18nNodeIdx << I18nCreateOpCode.SHIFT;\n  let parentTNode = getCurrentParentTNode();\n\n  if (rootTNode === parentTNode) {\n    // FIXME(misko): A null `parentTNode` should represent when we fall of the `LView` boundary.\n    // (there is no parent), but in some circumstances (because we are inconsistent about how we set\n    // `previousOrParentTNode`) it could point to `rootTNode` So this is a work around.\n    parentTNode = null;\n  }\n  if (parentTNode === null) {\n    // If we don't have a parent that means that we can eagerly add nodes.\n    // If we have a parent than these nodes can't be added now (as the parent has not been created\n    // yet) and instead the `parentTNode` is responsible for adding it. See\n    // `TNode.insertBeforeIndex`\n    opCode |= I18nCreateOpCode.APPEND_EAGERLY;\n  }\n  if (isICU) {\n    opCode |= I18nCreateOpCode.COMMENT;\n    ensureIcuContainerVisitorLoaded(loadIcuContainerVisitor);\n  }\n  createOpCodes.push(opCode, text === null ? '' : text);\n  // We store `{{?}}` so that when looking at debug `TNodeType.template` we can see where the\n  // bindings are.\n  const tNode = createTNodeAtIndex(\n      tView, i18nNodeIdx, isICU ? TNodeType.Icu : TNodeType.Text,\n      text === null ? (ngDevMode ? '{{?}}' : '') : text, null);\n  addTNodeAndUpdateInsertBeforeIndex(existingTNodes, tNode);\n  const tNodeIdx = tNode.index;\n  setCurrentTNode(tNode, false /* Text nodes are self closing */);\n  if (parentTNode !== null && rootTNode !== parentTNode) {\n    // We are a child of deeper node (rather than a direct child of `i18nStart` instruction.)\n    // We have to make sure to add ourselves to the parent.\n    setTNodeInsertBeforeIndex(parentTNode, tNodeIdx);\n  }\n  return tNode;\n}\n\n/**\n * Processes text node in i18n block.\n *\n * Text nodes can have:\n * - Create instruction in `createOpCodes` for creating the text node.\n * - Allocate spec for text node in i18n range of `LView`\n * - If contains binding:\n *    - bindings => allocate space in i18n range of `LView` to store the binding value.\n *    - populate `updateOpCodes` with update instructions.\n *\n * @param tView Current `TView`\n * @param rootTNode Root `TNode` of the i18n block. This node determines if the new TNode will\n *     be added as part of the `i18nStart` instruction or as part of the\n *     `TNode.insertBeforeIndex`.\n * @param existingTNodes internal state for `addTNodeAndUpdateInsertBeforeIndex`.\n * @param createOpCodes Location where the creation OpCodes will be stored.\n * @param lView Current `LView`\n * @param text The translated text (which may contain binding)\n */\nfunction i18nStartFirstCreatePassProcessTextNode(\n    tView: TView, rootTNode: TNode|null, existingTNodes: TNode[], createOpCodes: I18nCreateOpCodes,\n    updateOpCodes: I18nUpdateOpCodes, lView: LView, text: string): void {\n  const hasBinding = text.match(BINDING_REGEXP);\n  const tNode = createTNodeAndAddOpCode(\n      tView, rootTNode, existingTNodes, lView, createOpCodes, hasBinding ? null : text, false);\n  if (hasBinding) {\n    generateBindingUpdateOpCodes(updateOpCodes, text, tNode.index, null, 0, null);\n  }\n}\n\n/**\n * See `i18nAttributes` above.\n */\nexport function i18nAttributesFirstPass(tView: TView, index: number, values: string[]) {\n  const previousElement = getCurrentTNode()!;\n  const previousElementIndex = previousElement.index;\n  const updateOpCodes: I18nUpdateOpCodes = [] as any;\n  if (ngDevMode) {\n    attachDebugGetter(updateOpCodes, i18nUpdateOpCodesToString);\n  }\n  if (tView.firstCreatePass && tView.data[index] === null) {\n    for (let i = 0; i < values.length; i += 2) {\n      const attrName = values[i];\n      const message = values[i + 1];\n\n      if (message !== '') {\n        // Check if attribute value contains an ICU and throw an error if that's the case.\n        // ICUs in element attributes are not supported.\n        // Note: we intentionally retain the error here by not using `ngDevMode`, because\n        // the `value` can change based on the locale and users aren't guaranteed to hit\n        // an invalid string while they're developing.\n        if (ICU_REGEXP.test(message)) {\n          throw new Error(\n              `ICU expressions are not supported in attributes. Message: \"${message}\".`);\n        }\n\n        // i18n attributes that hit this code path are guaranteed to have bindings, because\n        // the compiler treats static i18n attributes as regular attribute bindings.\n        // Since this may not be the first i18n attribute on this element we need to pass in how\n        // many previous bindings there have already been.\n        generateBindingUpdateOpCodes(\n            updateOpCodes, message, previousElementIndex, attrName, countBindings(updateOpCodes),\n            null);\n      }\n    }\n    tView.data[index] = updateOpCodes;\n  }\n}\n\n\n/**\n * Generate the OpCodes to update the bindings of a string.\n *\n * @param updateOpCodes Place where the update opcodes will be stored.\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 * @param bindingStart The lView index of the next expression that can be bound via an opCode.\n * @returns The mask value for these bindings\n */\nfunction generateBindingUpdateOpCodes(\n    updateOpCodes: I18nUpdateOpCodes, str: string, destinationNode: number, attrName: string|null,\n    bindingStart: number, sanitizeFn: SanitizerFn|null): number {\n  ngDevMode &&\n      assertGreaterThanOrEqual(\n          destinationNode, HEADER_OFFSET, 'Index must be in absolute LView offset');\n  const maskIndex = updateOpCodes.length;  // Location of mask\n  const sizeIndex = maskIndex + 1;         // location of size for skipping\n  updateOpCodes.push(null, null);          // Alloc space for mask and size\n  const startIndex = maskIndex + 2;        // location of first allocation.\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 = bindingStart + 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[maskIndex] = mask;\n  updateOpCodes[sizeIndex] = updateOpCodes.length - startIndex;\n  return mask;\n}\n\n/**\n * Count the number of bindings in the given `opCodes`.\n *\n * It could be possible to speed this up, by passing the number of bindings found back from\n * `generateBindingUpdateOpCodes()` to `i18nAttributesFirstPass()` but this would then require more\n * complexity in the code and/or transient objects to be created.\n *\n * Since this function is only called once when the template is instantiated, is trivial in the\n * first instance (since `opCodes` will be an empty array), and it is not common for elements to\n * contain multiple i18n bound attributes, it seems like this is a reasonable compromise.\n */\nfunction countBindings(opCodes: I18nUpdateOpCodes): number {\n  let count = 0;\n  for (let i = 0; i < opCodes.length; i++) {\n    const opCode = opCodes[i];\n    // Bindings are negative numbers.\n    if (typeof opCode === 'number' && opCode < 0) {\n      count++;\n    }\n  }\n  return count;\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\n * have more than 32 bindings this will be hit very rarely. The downside of hitting this corner\n * case is 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): subTemplateIndex is - 1 {\n  return subTemplateIndex === -1;\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\n * translated 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 icuExpression\n * @param index Index where the anchor is stored and an optional `TIcuContainerNode`\n *   - `lView[anchorIdx]` points to a `Comment` node representing the anchor for the ICU.\n *   - `tView.data[anchorIdx]` points to the `TIcuContainerNode` if ICU is root (`null` otherwise)\n */\nexport function icuStart(\n    tView: TView, lView: LView, updateOpCodes: I18nUpdateOpCodes, parentIdx: number,\n    icuExpression: IcuExpression, anchorIdx: number) {\n  ngDevMode && assertDefined(icuExpression, 'ICU expression must be defined');\n  let bindingMask = 0;\n  const tIcu: TIcu = {\n    type: icuExpression.type,\n    currentCaseLViewIndex: allocExpando(tView, lView, 1, null),\n    anchorIdx,\n    cases: [],\n    create: [],\n    remove: [],\n    update: []\n  };\n  addUpdateIcuSwitch(updateOpCodes, icuExpression, anchorIdx);\n  setTIcu(tView, anchorIdx, tIcu);\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    bindingMask = parseIcuCase(\n                      tView, tIcu, lView, updateOpCodes, parentIdx, icuExpression.cases[i],\n                      valueArr.join(''), nestedIcus) |\n        bindingMask;\n  }\n  if (bindingMask) {\n    addUpdateIcuUpdate(updateOpCodes, bindingMask, anchorIdx);\n  }\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 = i18nParseTextIntoPartsAndICU(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 = i18nParseTextIntoPartsAndICU(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 * 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\n * into keys and cases. Original code from closure library, modified for Angular.\n *\n * @param pattern (sub)Pattern to be broken.\n * @returns An `Array<string|IcuExpression>` where:\n *   - odd positions: `string` => text between ICU expressions\n *   - even positions: `ICUExpression` => ICU expression parsed into `ICUExpression` record.\n */\nexport function i18nParseTextIntoPartsAndICU(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 */\nexport function parseIcuCase(\n    tView: TView, tIcu: TIcu, lView: LView, updateOpCodes: I18nUpdateOpCodes, parentIdx: number,\n    caseName: string, unsafeCaseHtml: string, nestedIcus: IcuExpression[]): number {\n  const create: IcuCreateOpCodes = [] as any;\n  const remove: I18nRemoveOpCodes = [] as any;\n  const update: I18nUpdateOpCodes = [] as any;\n  if (ngDevMode) {\n    attachDebugGetter(create, icuCreateOpCodesToString);\n    attachDebugGetter(remove, i18nRemoveOpCodesToString);\n    attachDebugGetter(update, i18nUpdateOpCodesToString);\n  }\n  tIcu.cases.push(caseName);\n  tIcu.create.push(create);\n  tIcu.remove.push(remove);\n  tIcu.update.push(update);\n\n  const inertBodyHelper = getInertBodyHelper(getDocument());\n  const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeCaseHtml);\n  ngDevMode && assertDefined(inertBodyElement, 'Unable to generate inert body element');\n  const inertRootNode = getTemplateContent(inertBodyElement!) as Element || inertBodyElement;\n  if (inertRootNode) {\n    return walkIcuTree(\n        tView, tIcu, lView, updateOpCodes, create, remove, update, inertRootNode, parentIdx,\n        nestedIcus, 0);\n  } else {\n    return 0;\n  }\n}\n\nfunction walkIcuTree(\n    tView: TView, tIcu: TIcu, lView: LView, sharedUpdateOpCodes: I18nUpdateOpCodes,\n    create: IcuCreateOpCodes, remove: I18nRemoveOpCodes, update: I18nUpdateOpCodes,\n    parentNode: Element, parentIdx: number, nestedIcus: IcuExpression[], depth: number): number {\n  let bindingMask = 0;\n  let currentNode = parentNode.firstChild;\n  while (currentNode) {\n    const newIndex = allocExpando(tView, lView, 1, null);\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          addCreateNodeAndAppend(create, ELEMENT_MARKER, tagName, parentIdx, newIndex);\n          tView.data[newIndex] = tagName;\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                  generateBindingUpdateOpCodes(\n                      update, attr.value, newIndex, attr.name, 0, _sanitizeUrl);\n                } else if (SRCSET_ATTRS[lowerAttrName]) {\n                  generateBindingUpdateOpCodes(\n                      update, attr.value, newIndex, attr.name, 0, sanitizeSrcset);\n                } else {\n                  generateBindingUpdateOpCodes(update, attr.value, newIndex, attr.name, 0, null);\n                }\n              } else {\n                ngDevMode &&\n                    console.warn(\n                        `WARNING: ignoring unsafe attribute value ` +\n                        `${lowerAttrName} on element ${tagName} ` +\n                        `(see https://g.co/ng/security#xss)`);\n              }\n            } else {\n              addCreateAttribute(create, newIndex, attr);\n            }\n          }\n          // Parse the children of this node (if any)\n          bindingMask = walkIcuTree(\n                            tView, tIcu, lView, sharedUpdateOpCodes, create, remove, update,\n                            currentNode as Element, newIndex, nestedIcus, depth + 1) |\n              bindingMask;\n          addRemoveNode(remove, newIndex, depth);\n        }\n        break;\n      case Node.TEXT_NODE:\n        const value = currentNode.textContent || '';\n        const hasBinding = value.match(BINDING_REGEXP);\n        addCreateNodeAndAppend(create, null, hasBinding ? '' : value, parentIdx, newIndex);\n        addRemoveNode(remove, newIndex, depth);\n        if (hasBinding) {\n          bindingMask =\n              generateBindingUpdateOpCodes(update, value, newIndex, null, 0, null) | bindingMask;\n        }\n        break;\n      case Node.COMMENT_NODE:\n        // Check if the comment node is a placeholder for a nested ICU\n        const isNestedIcu = NESTED_ICU.exec(currentNode.textContent || '');\n        if (isNestedIcu) {\n          const nestedIcuIndex = parseInt(isNestedIcu[1], 10);\n          const icuExpression: IcuExpression = nestedIcus[nestedIcuIndex];\n          // Create the comment node that will anchor the ICU expression\n          addCreateNodeAndAppend(\n              create, ICU_MARKER, ngDevMode ? `nested ICU ${nestedIcuIndex}` : '', parentIdx,\n              newIndex);\n          icuStart(tView, lView, sharedUpdateOpCodes, parentIdx, icuExpression, newIndex);\n          addRemoveNestedIcu(remove, newIndex, depth);\n        }\n        break;\n    }\n    currentNode = currentNode.nextSibling;\n  }\n  return bindingMask;\n}\n\nfunction addRemoveNode(remove: I18nRemoveOpCodes, index: number, depth: number) {\n  if (depth === 0) {\n    remove.push(index);\n  }\n}\n\nfunction addRemoveNestedIcu(remove: I18nRemoveOpCodes, index: number, depth: number) {\n  if (depth === 0) {\n    remove.push(~index);  // remove ICU at `index`\n    remove.push(index);   // remove ICU comment at `index`\n  }\n}\n\nfunction addUpdateIcuSwitch(\n    update: I18nUpdateOpCodes, icuExpression: IcuExpression, index: number) {\n  update.push(\n      toMaskBit(icuExpression.mainBinding), 2, -1 - icuExpression.mainBinding,\n      index << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuSwitch);\n}\n\nfunction addUpdateIcuUpdate(update: I18nUpdateOpCodes, bindingMask: number, index: number) {\n  update.push(bindingMask, 1, index << I18nUpdateOpCode.SHIFT_REF | I18nUpdateOpCode.IcuUpdate);\n}\n\nfunction addCreateNodeAndAppend(\n    create: IcuCreateOpCodes, marker: null|ICU_MARKER|ELEMENT_MARKER, text: string,\n    appendToParentIdx: number, createAtIdx: number) {\n  if (marker !== null) {\n    create.push(marker);\n  }\n  create.push(\n      text, createAtIdx,\n      icuCreateOpCode(IcuCreateOpCode.AppendChild, appendToParentIdx, createAtIdx));\n}\n\nfunction addCreateAttribute(create: IcuCreateOpCodes, newIndex: number, attr: Attr) {\n  create.push(newIndex << IcuCreateOpCode.SHIFT_REF | IcuCreateOpCode.Attr, attr.name, attr.value);\n}\n"]}
Note: See TracBrowser for help on using the repository browser.