@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.
- package/bundles/core-testing.umd.js +1 -1
- package/bundles/core-testing.umd.min.js +1 -1
- package/bundles/core-testing.umd.min.js.map +1 -1
- package/bundles/core.umd.js +1678 -1387
- package/bundles/core.umd.js.map +1 -1
- package/bundles/core.umd.min.js +152 -124
- package/bundles/core.umd.min.js.map +1 -1
- package/core.d.ts +258 -58
- package/core.metadata.json +1 -1
- package/esm2015/core.js +3 -3
- package/esm2015/src/application_module.js +2 -2
- package/esm2015/src/application_ref.js +2 -2
- package/esm2015/src/core_render3_private_export.js +2 -1
- package/esm2015/src/metadata/di.js +1 -1
- package/esm2015/src/r3_symbols.js +2 -1
- package/esm2015/src/reflection/reflection_capabilities.js +38 -8
- package/esm2015/src/render3/bindings.js +3 -3
- package/esm2015/src/render3/component.js +3 -3
- package/esm2015/src/render3/di.js +1 -1
- package/esm2015/src/render3/i18n/i18n_apply.js +445 -0
- package/esm2015/src/render3/i18n/i18n_debug.js +170 -0
- package/esm2015/src/render3/i18n/i18n_locale_id.js +37 -0
- package/esm2015/src/render3/i18n/i18n_parse.js +635 -0
- package/esm2015/src/render3/i18n/i18n_postprocess.js +121 -0
- package/esm2015/src/render3/index.js +3 -2
- package/esm2015/src/render3/instructions/advance.js +3 -3
- package/esm2015/src/render3/instructions/element.js +3 -3
- package/esm2015/src/render3/instructions/element_container.js +3 -3
- package/esm2015/src/render3/instructions/i18n.js +164 -0
- package/esm2015/src/render3/instructions/listener.js +3 -3
- package/esm2015/src/render3/instructions/lview_debug.js +54 -213
- package/esm2015/src/render3/instructions/shared.js +54 -38
- package/esm2015/src/render3/instructions/text.js +3 -3
- package/esm2015/src/render3/interfaces/i18n.js +12 -3
- package/esm2015/src/render3/interfaces/node.js +13 -1
- package/esm2015/src/render3/interfaces/view.js +1 -1
- package/esm2015/src/render3/jit/directive.js +33 -10
- package/esm2015/src/render3/ng_module_ref.js +2 -2
- package/esm2015/src/render3/node_manipulation.js +1 -11
- package/esm2015/src/render3/pure_function.js +3 -3
- package/esm2015/src/render3/query.js +14 -12
- package/esm2015/src/render3/styling/style_binding_list.js +3 -3
- package/esm2015/src/render3/styling/styling_parser.js +3 -2
- package/esm2015/src/render3/util/debug_utils.js +31 -2
- package/esm2015/src/render3/util/discovery_utils.js +1 -1
- package/esm2015/src/render3/util/view_utils.js +5 -5
- package/esm2015/src/render3/view_engine_compatibility.js +13 -4
- package/esm2015/src/util/assert.js +2 -2
- package/esm2015/src/util/char_code.js +1 -1
- package/esm2015/src/version.js +1 -1
- package/fesm2015/core.js +1652 -1364
- package/fesm2015/core.js.map +1 -1
- package/fesm2015/testing.js +1 -1
- package/package.json +1 -1
- package/src/r3_symbols.d.ts +13 -1
- package/testing/testing.d.ts +1 -1
- package/testing.d.ts +1 -1
- package/esm2015/src/render3/i18n.js +0 -1225
|
@@ -1,1225 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright Google LLC All Rights Reserved.
|
|
4
|
-
*
|
|
5
|
-
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
-
* found in the LICENSE file at https://angular.io/license
|
|
7
|
-
*/
|
|
8
|
-
import '../util/ng_i18n_closure_mode';
|
|
9
|
-
import { DEFAULT_LOCALE_ID, getPluralCase } from '../i18n/localization';
|
|
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 { assertDataInRange, assertDefined, assertEqual } from '../util/assert';
|
|
15
|
-
import { bindingUpdated } from './bindings';
|
|
16
|
-
import { attachPatchData } from './context_discovery';
|
|
17
|
-
import { setDelayProjection } from './instructions/all';
|
|
18
|
-
import { attachI18nOpCodesDebug } from './instructions/lview_debug';
|
|
19
|
-
import { allocExpando, elementAttributeInternal, elementPropertyInternal, getOrCreateTNode, setInputsForProperty, setNgReflectProperties, textBindingInternal } from './instructions/shared';
|
|
20
|
-
import { NATIVE } from './interfaces/container';
|
|
21
|
-
import { getDocument } from './interfaces/document';
|
|
22
|
-
import { COMMENT_MARKER, ELEMENT_MARKER } from './interfaces/i18n';
|
|
23
|
-
import { isLContainer } from './interfaces/type_checks';
|
|
24
|
-
import { HEADER_OFFSET, RENDERER, T_HOST } from './interfaces/view';
|
|
25
|
-
import { appendChild, applyProjection, createTextNode, nativeRemoveNode } from './node_manipulation';
|
|
26
|
-
import { getBindingIndex, getIsParent, getLView, getPreviousOrParentTNode, getTView, nextBindingIndex, setIsNotParent, setPreviousOrParentTNode } from './state';
|
|
27
|
-
import { renderStringify } from './util/misc_utils';
|
|
28
|
-
import { getNativeByIndex, getNativeByTNode, getTNode, load } from './util/view_utils';
|
|
29
|
-
const MARKER = `�`;
|
|
30
|
-
const ICU_BLOCK_REGEXP = /^\s*(�\d+:?\d*�)\s*,\s*(select|plural)\s*,/;
|
|
31
|
-
const SUBTEMPLATE_REGEXP = /�\/?\*(\d+:\d+)�/gi;
|
|
32
|
-
const PH_REGEXP = /�(\/?[#*!]\d+):?\d*�/gi;
|
|
33
|
-
const BINDING_REGEXP = /�(\d+):?\d*�/gi;
|
|
34
|
-
const ICU_REGEXP = /({\s*�\d+:?\d*�\s*,\s*\S{6}\s*,[\s\S]*})/gi;
|
|
35
|
-
// i18nPostprocess consts
|
|
36
|
-
const ROOT_TEMPLATE_ID = 0;
|
|
37
|
-
const PP_MULTI_VALUE_PLACEHOLDERS_REGEXP = /\[(�.+?�?)\]/;
|
|
38
|
-
const PP_PLACEHOLDERS_REGEXP = /\[(�.+?�?)\]|(�\/?\*\d+:\d+�)/g;
|
|
39
|
-
const PP_ICU_VARS_REGEXP = /({\s*)(VAR_(PLURAL|SELECT)(_\d+)?)(\s*,)/g;
|
|
40
|
-
const PP_ICU_PLACEHOLDERS_REGEXP = /{([A-Z0-9_]+)}/g;
|
|
41
|
-
const PP_ICUS_REGEXP = /�I18N_EXP_(ICU(_\d+)?)�/g;
|
|
42
|
-
const PP_CLOSE_TEMPLATE_REGEXP = /\/\*/;
|
|
43
|
-
const PP_TEMPLATE_ID_REGEXP = /\d+\:(\d+)/;
|
|
44
|
-
/**
|
|
45
|
-
* Breaks pattern into strings and top level {...} blocks.
|
|
46
|
-
* Can be used to break a message into text and ICU expressions, or to break an ICU expression into
|
|
47
|
-
* keys and cases.
|
|
48
|
-
* Original code from closure library, modified for Angular.
|
|
49
|
-
*
|
|
50
|
-
* @param pattern (sub)Pattern to be broken.
|
|
51
|
-
*
|
|
52
|
-
*/
|
|
53
|
-
function extractParts(pattern) {
|
|
54
|
-
if (!pattern) {
|
|
55
|
-
return [];
|
|
56
|
-
}
|
|
57
|
-
let prevPos = 0;
|
|
58
|
-
const braceStack = [];
|
|
59
|
-
const results = [];
|
|
60
|
-
const braces = /[{}]/g;
|
|
61
|
-
// lastIndex doesn't get set to 0 so we have to.
|
|
62
|
-
braces.lastIndex = 0;
|
|
63
|
-
let match;
|
|
64
|
-
while (match = braces.exec(pattern)) {
|
|
65
|
-
const pos = match.index;
|
|
66
|
-
if (match[0] == '}') {
|
|
67
|
-
braceStack.pop();
|
|
68
|
-
if (braceStack.length == 0) {
|
|
69
|
-
// End of the block.
|
|
70
|
-
const block = pattern.substring(prevPos, pos);
|
|
71
|
-
if (ICU_BLOCK_REGEXP.test(block)) {
|
|
72
|
-
results.push(parseICUBlock(block));
|
|
73
|
-
}
|
|
74
|
-
else {
|
|
75
|
-
results.push(block);
|
|
76
|
-
}
|
|
77
|
-
prevPos = pos + 1;
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
else {
|
|
81
|
-
if (braceStack.length == 0) {
|
|
82
|
-
const substring = pattern.substring(prevPos, pos);
|
|
83
|
-
results.push(substring);
|
|
84
|
-
prevPos = pos + 1;
|
|
85
|
-
}
|
|
86
|
-
braceStack.push('{');
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
const substring = pattern.substring(prevPos);
|
|
90
|
-
results.push(substring);
|
|
91
|
-
return results;
|
|
92
|
-
}
|
|
93
|
-
/**
|
|
94
|
-
* Parses text containing an ICU expression and produces a JSON object for it.
|
|
95
|
-
* Original code from closure library, modified for Angular.
|
|
96
|
-
*
|
|
97
|
-
* @param pattern Text containing an ICU expression that needs to be parsed.
|
|
98
|
-
*
|
|
99
|
-
*/
|
|
100
|
-
function parseICUBlock(pattern) {
|
|
101
|
-
const cases = [];
|
|
102
|
-
const values = [];
|
|
103
|
-
let icuType = 1 /* plural */;
|
|
104
|
-
let mainBinding = 0;
|
|
105
|
-
pattern = pattern.replace(ICU_BLOCK_REGEXP, function (str, binding, type) {
|
|
106
|
-
if (type === 'select') {
|
|
107
|
-
icuType = 0 /* select */;
|
|
108
|
-
}
|
|
109
|
-
else {
|
|
110
|
-
icuType = 1 /* plural */;
|
|
111
|
-
}
|
|
112
|
-
mainBinding = parseInt(binding.substr(1), 10);
|
|
113
|
-
return '';
|
|
114
|
-
});
|
|
115
|
-
const parts = extractParts(pattern);
|
|
116
|
-
// Looking for (key block)+ sequence. One of the keys has to be "other".
|
|
117
|
-
for (let pos = 0; pos < parts.length;) {
|
|
118
|
-
let key = parts[pos++].trim();
|
|
119
|
-
if (icuType === 1 /* plural */) {
|
|
120
|
-
// Key can be "=x", we just want "x"
|
|
121
|
-
key = key.replace(/\s*(?:=)?(\w+)\s*/, '$1');
|
|
122
|
-
}
|
|
123
|
-
if (key.length) {
|
|
124
|
-
cases.push(key);
|
|
125
|
-
}
|
|
126
|
-
const blocks = extractParts(parts[pos++]);
|
|
127
|
-
if (cases.length > values.length) {
|
|
128
|
-
values.push(blocks);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
// TODO(ocombe): support ICU expressions in attributes, see #21615
|
|
132
|
-
return { type: icuType, mainBinding: mainBinding, cases, values };
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Removes everything inside the sub-templates of a message.
|
|
136
|
-
*/
|
|
137
|
-
function removeInnerTemplateTranslation(message) {
|
|
138
|
-
let match;
|
|
139
|
-
let res = '';
|
|
140
|
-
let index = 0;
|
|
141
|
-
let inTemplate = false;
|
|
142
|
-
let tagMatched;
|
|
143
|
-
while ((match = SUBTEMPLATE_REGEXP.exec(message)) !== null) {
|
|
144
|
-
if (!inTemplate) {
|
|
145
|
-
res += message.substring(index, match.index + match[0].length);
|
|
146
|
-
tagMatched = match[1];
|
|
147
|
-
inTemplate = true;
|
|
148
|
-
}
|
|
149
|
-
else {
|
|
150
|
-
if (match[0] === `${MARKER}/*${tagMatched}${MARKER}`) {
|
|
151
|
-
index = match.index;
|
|
152
|
-
inTemplate = false;
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
ngDevMode &&
|
|
157
|
-
assertEqual(inTemplate, false, `Tag mismatch: unable to find the end of the sub-template in the translation "${message}"`);
|
|
158
|
-
res += message.substr(index);
|
|
159
|
-
return res;
|
|
160
|
-
}
|
|
161
|
-
/**
|
|
162
|
-
* Extracts a part of a message and removes the rest.
|
|
163
|
-
*
|
|
164
|
-
* This method is used for extracting a part of the message associated with a template. A translated
|
|
165
|
-
* message can span multiple templates.
|
|
166
|
-
*
|
|
167
|
-
* Example:
|
|
168
|
-
* ```
|
|
169
|
-
* <div i18n>Translate <span *ngIf>me</span>!</div>
|
|
170
|
-
* ```
|
|
171
|
-
*
|
|
172
|
-
* @param message The message to crop
|
|
173
|
-
* @param subTemplateIndex Index of the sub-template to extract. If undefined it returns the
|
|
174
|
-
* external template and removes all sub-templates.
|
|
175
|
-
*/
|
|
176
|
-
export function getTranslationForTemplate(message, subTemplateIndex) {
|
|
177
|
-
if (isRootTemplateMessage(subTemplateIndex)) {
|
|
178
|
-
// We want the root template message, ignore all sub-templates
|
|
179
|
-
return removeInnerTemplateTranslation(message);
|
|
180
|
-
}
|
|
181
|
-
else {
|
|
182
|
-
// We want a specific sub-template
|
|
183
|
-
const start = message.indexOf(`:${subTemplateIndex}${MARKER}`) + 2 + subTemplateIndex.toString().length;
|
|
184
|
-
const end = message.search(new RegExp(`${MARKER}\\/\\*\\d+:${subTemplateIndex}${MARKER}`));
|
|
185
|
-
return removeInnerTemplateTranslation(message.substring(start, end));
|
|
186
|
-
}
|
|
187
|
-
}
|
|
188
|
-
/**
|
|
189
|
-
* Generate the OpCodes to update the bindings of a string.
|
|
190
|
-
*
|
|
191
|
-
* @param str The string containing the bindings.
|
|
192
|
-
* @param destinationNode Index of the destination node which will receive the binding.
|
|
193
|
-
* @param attrName Name of the attribute, if the string belongs to an attribute.
|
|
194
|
-
* @param sanitizeFn Sanitization function used to sanitize the string after update, if necessary.
|
|
195
|
-
*/
|
|
196
|
-
function generateBindingUpdateOpCodes(str, destinationNode, attrName, sanitizeFn = null) {
|
|
197
|
-
const updateOpCodes = [null, null]; // Alloc space for mask and size
|
|
198
|
-
const textParts = str.split(BINDING_REGEXP);
|
|
199
|
-
let mask = 0;
|
|
200
|
-
for (let j = 0; j < textParts.length; j++) {
|
|
201
|
-
const textValue = textParts[j];
|
|
202
|
-
if (j & 1) {
|
|
203
|
-
// Odd indexes are bindings
|
|
204
|
-
const bindingIndex = parseInt(textValue, 10);
|
|
205
|
-
updateOpCodes.push(-1 - bindingIndex);
|
|
206
|
-
mask = mask | toMaskBit(bindingIndex);
|
|
207
|
-
}
|
|
208
|
-
else if (textValue !== '') {
|
|
209
|
-
// Even indexes are text
|
|
210
|
-
updateOpCodes.push(textValue);
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
updateOpCodes.push(destinationNode << 2 /* SHIFT_REF */ |
|
|
214
|
-
(attrName ? 1 /* Attr */ : 0 /* Text */));
|
|
215
|
-
if (attrName) {
|
|
216
|
-
updateOpCodes.push(attrName, sanitizeFn);
|
|
217
|
-
}
|
|
218
|
-
updateOpCodes[0] = mask;
|
|
219
|
-
updateOpCodes[1] = updateOpCodes.length - 2;
|
|
220
|
-
return updateOpCodes;
|
|
221
|
-
}
|
|
222
|
-
function getBindingMask(icuExpression, mask = 0) {
|
|
223
|
-
mask = mask | toMaskBit(icuExpression.mainBinding);
|
|
224
|
-
let match;
|
|
225
|
-
for (let i = 0; i < icuExpression.values.length; i++) {
|
|
226
|
-
const valueArr = icuExpression.values[i];
|
|
227
|
-
for (let j = 0; j < valueArr.length; j++) {
|
|
228
|
-
const value = valueArr[j];
|
|
229
|
-
if (typeof value === 'string') {
|
|
230
|
-
while (match = BINDING_REGEXP.exec(value)) {
|
|
231
|
-
mask = mask | toMaskBit(parseInt(match[1], 10));
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
else {
|
|
235
|
-
mask = getBindingMask(value, mask);
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
return mask;
|
|
240
|
-
}
|
|
241
|
-
const i18nIndexStack = [];
|
|
242
|
-
let i18nIndexStackPointer = -1;
|
|
243
|
-
/**
|
|
244
|
-
* Convert binding index to mask bit.
|
|
245
|
-
*
|
|
246
|
-
* Each index represents a single bit on the bit-mask. Because bit-mask only has 32 bits, we make
|
|
247
|
-
* the 32nd bit share all masks for all bindings higher than 32. Since it is extremely rare to have
|
|
248
|
-
* more than 32 bindings this will be hit very rarely. The downside of hitting this corner case is
|
|
249
|
-
* that we will execute binding code more often than necessary. (penalty of performance)
|
|
250
|
-
*/
|
|
251
|
-
function toMaskBit(bindingIndex) {
|
|
252
|
-
return 1 << Math.min(bindingIndex, 31);
|
|
253
|
-
}
|
|
254
|
-
const parentIndexStack = [];
|
|
255
|
-
/**
|
|
256
|
-
* Marks a block of text as translatable.
|
|
257
|
-
*
|
|
258
|
-
* The instructions `i18nStart` and `i18nEnd` mark the translation block in the template.
|
|
259
|
-
* The translation `message` is the value which is locale specific. The translation string may
|
|
260
|
-
* contain placeholders which associate inner elements and sub-templates within the translation.
|
|
261
|
-
*
|
|
262
|
-
* The translation `message` placeholders are:
|
|
263
|
-
* - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be
|
|
264
|
-
* interpolated into. The placeholder `index` points to the expression binding index. An optional
|
|
265
|
-
* `block` that matches the sub-template in which it was declared.
|
|
266
|
-
* - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*: Marks the beginning
|
|
267
|
-
* and end of DOM element that were embedded in the original translation block. The placeholder
|
|
268
|
-
* `index` points to the element index in the template instructions set. An optional `block` that
|
|
269
|
-
* matches the sub-template in which it was declared.
|
|
270
|
-
* - `�!{index}(:{block})�`/`�/!{index}(:{block})�`: *Projection Placeholder*: Marks the
|
|
271
|
-
* beginning and end of <ng-content> that was embedded in the original translation block.
|
|
272
|
-
* The placeholder `index` points to the element index in the template instructions set.
|
|
273
|
-
* An optional `block` that matches the sub-template in which it was declared.
|
|
274
|
-
* - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be
|
|
275
|
-
* split up and translated separately in each angular template function. The `index` points to the
|
|
276
|
-
* `template` instruction index. A `block` that matches the sub-template in which it was declared.
|
|
277
|
-
*
|
|
278
|
-
* @param index A unique index of the translation in the static block.
|
|
279
|
-
* @param message The translation message.
|
|
280
|
-
* @param subTemplateIndex Optional sub-template index in the `message`.
|
|
281
|
-
*
|
|
282
|
-
* @codeGenApi
|
|
283
|
-
*/
|
|
284
|
-
export function ɵɵi18nStart(index, message, subTemplateIndex) {
|
|
285
|
-
const tView = getTView();
|
|
286
|
-
ngDevMode && assertDefined(tView, `tView should be defined`);
|
|
287
|
-
i18nIndexStack[++i18nIndexStackPointer] = index;
|
|
288
|
-
// We need to delay projections until `i18nEnd`
|
|
289
|
-
setDelayProjection(true);
|
|
290
|
-
if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
|
|
291
|
-
i18nStartFirstPass(getLView(), tView, index, message, subTemplateIndex);
|
|
292
|
-
}
|
|
293
|
-
}
|
|
294
|
-
// Count for the number of vars that will be allocated for each i18n block.
|
|
295
|
-
// It is global because this is used in multiple functions that include loops and recursive calls.
|
|
296
|
-
// This is reset to 0 when `i18nStartFirstPass` is called.
|
|
297
|
-
let i18nVarsCount;
|
|
298
|
-
function allocNodeIndex(startIndex) {
|
|
299
|
-
return startIndex + i18nVarsCount++;
|
|
300
|
-
}
|
|
301
|
-
/**
|
|
302
|
-
* See `i18nStart` above.
|
|
303
|
-
*/
|
|
304
|
-
function i18nStartFirstPass(lView, tView, index, message, subTemplateIndex) {
|
|
305
|
-
const startIndex = tView.blueprint.length - HEADER_OFFSET;
|
|
306
|
-
i18nVarsCount = 0;
|
|
307
|
-
const previousOrParentTNode = getPreviousOrParentTNode();
|
|
308
|
-
const parentTNode = getIsParent() ? previousOrParentTNode : previousOrParentTNode && previousOrParentTNode.parent;
|
|
309
|
-
let parentIndex = parentTNode && parentTNode !== lView[T_HOST] ? parentTNode.index - HEADER_OFFSET : index;
|
|
310
|
-
let parentIndexPointer = 0;
|
|
311
|
-
parentIndexStack[parentIndexPointer] = parentIndex;
|
|
312
|
-
const createOpCodes = [];
|
|
313
|
-
// If the previous node wasn't the direct parent then we have a translation without top level
|
|
314
|
-
// element and we need to keep a reference of the previous element if there is one. We should also
|
|
315
|
-
// keep track whether an element was a parent node or not, so that the logic that consumes
|
|
316
|
-
// the generated `I18nMutateOpCode`s can leverage this information to properly set TNode state
|
|
317
|
-
// (whether it's a parent or sibling).
|
|
318
|
-
if (index > 0 && previousOrParentTNode !== parentTNode) {
|
|
319
|
-
let previousTNodeIndex = previousOrParentTNode.index - HEADER_OFFSET;
|
|
320
|
-
// If current TNode is a sibling node, encode it using a negative index. This information is
|
|
321
|
-
// required when the `Select` action is processed (see the `readCreateOpCodes` function).
|
|
322
|
-
if (!getIsParent()) {
|
|
323
|
-
previousTNodeIndex = ~previousTNodeIndex;
|
|
324
|
-
}
|
|
325
|
-
// Create an OpCode to select the previous TNode
|
|
326
|
-
createOpCodes.push(previousTNodeIndex << 3 /* SHIFT_REF */ | 0 /* Select */);
|
|
327
|
-
}
|
|
328
|
-
const updateOpCodes = [];
|
|
329
|
-
const icuExpressions = [];
|
|
330
|
-
if (message === '' && isRootTemplateMessage(subTemplateIndex)) {
|
|
331
|
-
// If top level translation is an empty string, do not invoke additional processing
|
|
332
|
-
// and just create op codes for empty text node instead.
|
|
333
|
-
createOpCodes.push(message, allocNodeIndex(startIndex), parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
334
|
-
}
|
|
335
|
-
else {
|
|
336
|
-
const templateTranslation = getTranslationForTemplate(message, subTemplateIndex);
|
|
337
|
-
const msgParts = replaceNgsp(templateTranslation).split(PH_REGEXP);
|
|
338
|
-
for (let i = 0; i < msgParts.length; i++) {
|
|
339
|
-
let value = msgParts[i];
|
|
340
|
-
if (i & 1) {
|
|
341
|
-
// Odd indexes are placeholders (elements and sub-templates)
|
|
342
|
-
if (value.charAt(0) === '/') {
|
|
343
|
-
// It is a closing tag
|
|
344
|
-
if (value.charAt(1) === "#" /* ELEMENT */) {
|
|
345
|
-
const phIndex = parseInt(value.substr(2), 10);
|
|
346
|
-
parentIndex = parentIndexStack[--parentIndexPointer];
|
|
347
|
-
createOpCodes.push(phIndex << 3 /* SHIFT_REF */ | 5 /* ElementEnd */);
|
|
348
|
-
}
|
|
349
|
-
}
|
|
350
|
-
else {
|
|
351
|
-
const phIndex = parseInt(value.substr(1), 10);
|
|
352
|
-
const isElement = value.charAt(0) === "#" /* ELEMENT */;
|
|
353
|
-
// The value represents a placeholder that we move to the designated index.
|
|
354
|
-
// Note: positive indicies indicate that a TNode with a given index should also be marked
|
|
355
|
-
// as parent while executing `Select` instruction.
|
|
356
|
-
createOpCodes.push((isElement ? phIndex : ~phIndex) << 3 /* SHIFT_REF */ |
|
|
357
|
-
0 /* Select */, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
358
|
-
if (isElement) {
|
|
359
|
-
parentIndexStack[++parentIndexPointer] = parentIndex = phIndex;
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
}
|
|
363
|
-
else {
|
|
364
|
-
// Even indexes are text (including bindings & ICU expressions)
|
|
365
|
-
const parts = extractParts(value);
|
|
366
|
-
for (let j = 0; j < parts.length; j++) {
|
|
367
|
-
if (j & 1) {
|
|
368
|
-
// Odd indexes are ICU expressions
|
|
369
|
-
const icuExpression = parts[j];
|
|
370
|
-
// Verify that ICU expression has the right shape. Translations might contain invalid
|
|
371
|
-
// constructions (while original messages were correct), so ICU parsing at runtime may
|
|
372
|
-
// not succeed (thus `icuExpression` remains a string).
|
|
373
|
-
if (typeof icuExpression !== 'object') {
|
|
374
|
-
throw new Error(`Unable to parse ICU expression in "${templateTranslation}" message.`);
|
|
375
|
-
}
|
|
376
|
-
// Create the comment node that will anchor the ICU expression
|
|
377
|
-
const icuNodeIndex = allocNodeIndex(startIndex);
|
|
378
|
-
createOpCodes.push(COMMENT_MARKER, ngDevMode ? `ICU ${icuNodeIndex}` : '', icuNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
379
|
-
// Update codes for the ICU expression
|
|
380
|
-
const mask = getBindingMask(icuExpression);
|
|
381
|
-
icuStart(icuExpressions, icuExpression, icuNodeIndex, icuNodeIndex);
|
|
382
|
-
// Since this is recursive, the last TIcu that was pushed is the one we want
|
|
383
|
-
const tIcuIndex = icuExpressions.length - 1;
|
|
384
|
-
updateOpCodes.push(toMaskBit(icuExpression.mainBinding), // mask of the main binding
|
|
385
|
-
3, // skip 3 opCodes if not changed
|
|
386
|
-
-1 - icuExpression.mainBinding, icuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */, tIcuIndex, mask, // mask of all the bindings of this ICU expression
|
|
387
|
-
2, // skip 2 opCodes if not changed
|
|
388
|
-
icuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, tIcuIndex);
|
|
389
|
-
}
|
|
390
|
-
else if (parts[j] !== '') {
|
|
391
|
-
const text = parts[j];
|
|
392
|
-
// Even indexes are text (including bindings)
|
|
393
|
-
const hasBinding = text.match(BINDING_REGEXP);
|
|
394
|
-
// Create text nodes
|
|
395
|
-
const textNodeIndex = allocNodeIndex(startIndex);
|
|
396
|
-
createOpCodes.push(
|
|
397
|
-
// If there is a binding, the value will be set during update
|
|
398
|
-
hasBinding ? '' : text, textNodeIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
399
|
-
if (hasBinding) {
|
|
400
|
-
addAllToArray(generateBindingUpdateOpCodes(text, textNodeIndex), updateOpCodes);
|
|
401
|
-
}
|
|
402
|
-
}
|
|
403
|
-
}
|
|
404
|
-
}
|
|
405
|
-
}
|
|
406
|
-
}
|
|
407
|
-
if (i18nVarsCount > 0) {
|
|
408
|
-
allocExpando(tView, lView, i18nVarsCount);
|
|
409
|
-
}
|
|
410
|
-
ngDevMode &&
|
|
411
|
-
attachI18nOpCodesDebug(createOpCodes, updateOpCodes, icuExpressions.length ? icuExpressions : null, lView);
|
|
412
|
-
// NOTE: local var needed to properly assert the type of `TI18n`.
|
|
413
|
-
const tI18n = {
|
|
414
|
-
vars: i18nVarsCount,
|
|
415
|
-
create: createOpCodes,
|
|
416
|
-
update: updateOpCodes,
|
|
417
|
-
icus: icuExpressions.length ? icuExpressions : null,
|
|
418
|
-
};
|
|
419
|
-
tView.data[index + HEADER_OFFSET] = tI18n;
|
|
420
|
-
}
|
|
421
|
-
function appendI18nNode(tView, tNode, parentTNode, previousTNode, lView) {
|
|
422
|
-
ngDevMode && ngDevMode.rendererMoveNode++;
|
|
423
|
-
const nextNode = tNode.next;
|
|
424
|
-
if (!previousTNode) {
|
|
425
|
-
previousTNode = parentTNode;
|
|
426
|
-
}
|
|
427
|
-
// Re-organize node tree to put this node in the correct position.
|
|
428
|
-
if (previousTNode === parentTNode && tNode !== parentTNode.child) {
|
|
429
|
-
tNode.next = parentTNode.child;
|
|
430
|
-
parentTNode.child = tNode;
|
|
431
|
-
}
|
|
432
|
-
else if (previousTNode !== parentTNode && tNode !== previousTNode.next) {
|
|
433
|
-
tNode.next = previousTNode.next;
|
|
434
|
-
previousTNode.next = tNode;
|
|
435
|
-
}
|
|
436
|
-
else {
|
|
437
|
-
tNode.next = null;
|
|
438
|
-
}
|
|
439
|
-
if (parentTNode !== lView[T_HOST]) {
|
|
440
|
-
tNode.parent = parentTNode;
|
|
441
|
-
}
|
|
442
|
-
// If tNode was moved around, we might need to fix a broken link.
|
|
443
|
-
let cursor = tNode.next;
|
|
444
|
-
while (cursor) {
|
|
445
|
-
if (cursor.next === tNode) {
|
|
446
|
-
cursor.next = nextNode;
|
|
447
|
-
}
|
|
448
|
-
cursor = cursor.next;
|
|
449
|
-
}
|
|
450
|
-
// If the placeholder to append is a projection, we need to move the projected nodes instead
|
|
451
|
-
if (tNode.type === 1 /* Projection */) {
|
|
452
|
-
applyProjection(tView, lView, tNode);
|
|
453
|
-
return tNode;
|
|
454
|
-
}
|
|
455
|
-
appendChild(tView, lView, getNativeByTNode(tNode, lView), tNode);
|
|
456
|
-
const slotValue = lView[tNode.index];
|
|
457
|
-
if (tNode.type !== 0 /* Container */ && isLContainer(slotValue)) {
|
|
458
|
-
// Nodes that inject ViewContainerRef also have a comment node that should be moved
|
|
459
|
-
appendChild(tView, lView, slotValue[NATIVE], tNode);
|
|
460
|
-
}
|
|
461
|
-
return tNode;
|
|
462
|
-
}
|
|
463
|
-
function isRootTemplateMessage(subTemplateIndex) {
|
|
464
|
-
return subTemplateIndex === undefined;
|
|
465
|
-
}
|
|
466
|
-
/**
|
|
467
|
-
* Handles message string post-processing for internationalization.
|
|
468
|
-
*
|
|
469
|
-
* Handles message string post-processing by transforming it from intermediate
|
|
470
|
-
* format (that might contain some markers that we need to replace) to the final
|
|
471
|
-
* form, consumable by i18nStart instruction. Post processing steps include:
|
|
472
|
-
*
|
|
473
|
-
* 1. Resolve all multi-value cases (like [�*1:1��#2:1�|�#4:1�|�5�])
|
|
474
|
-
* 2. Replace all ICU vars (like "VAR_PLURAL")
|
|
475
|
-
* 3. Replace all placeholders used inside ICUs in a form of {PLACEHOLDER}
|
|
476
|
-
* 4. Replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�)
|
|
477
|
-
* in case multiple ICUs have the same placeholder name
|
|
478
|
-
*
|
|
479
|
-
* @param message Raw translation string for post processing
|
|
480
|
-
* @param replacements Set of replacements that should be applied
|
|
481
|
-
*
|
|
482
|
-
* @returns Transformed string that can be consumed by i18nStart instruction
|
|
483
|
-
*
|
|
484
|
-
* @codeGenApi
|
|
485
|
-
*/
|
|
486
|
-
export function ɵɵi18nPostprocess(message, replacements = {}) {
|
|
487
|
-
/**
|
|
488
|
-
* Step 1: resolve all multi-value placeholders like [�#5�|�*1:1��#2:1�|�#4:1�]
|
|
489
|
-
*
|
|
490
|
-
* Note: due to the way we process nested templates (BFS), multi-value placeholders are typically
|
|
491
|
-
* grouped by templates, for example: [�#5�|�#6�|�#1:1�|�#3:2�] where �#5� and �#6� belong to root
|
|
492
|
-
* template, �#1:1� belong to nested template with index 1 and �#1:2� - nested template with index
|
|
493
|
-
* 3. However in real templates the order might be different: i.e. �#1:1� and/or �#3:2� may go in
|
|
494
|
-
* front of �#6�. The post processing step restores the right order by keeping track of the
|
|
495
|
-
* template id stack and looks for placeholders that belong to the currently active template.
|
|
496
|
-
*/
|
|
497
|
-
let result = message;
|
|
498
|
-
if (PP_MULTI_VALUE_PLACEHOLDERS_REGEXP.test(message)) {
|
|
499
|
-
const matches = {};
|
|
500
|
-
const templateIdsStack = [ROOT_TEMPLATE_ID];
|
|
501
|
-
result = result.replace(PP_PLACEHOLDERS_REGEXP, (m, phs, tmpl) => {
|
|
502
|
-
const content = phs || tmpl;
|
|
503
|
-
const placeholders = matches[content] || [];
|
|
504
|
-
if (!placeholders.length) {
|
|
505
|
-
content.split('|').forEach((placeholder) => {
|
|
506
|
-
const match = placeholder.match(PP_TEMPLATE_ID_REGEXP);
|
|
507
|
-
const templateId = match ? parseInt(match[1], 10) : ROOT_TEMPLATE_ID;
|
|
508
|
-
const isCloseTemplateTag = PP_CLOSE_TEMPLATE_REGEXP.test(placeholder);
|
|
509
|
-
placeholders.push([templateId, isCloseTemplateTag, placeholder]);
|
|
510
|
-
});
|
|
511
|
-
matches[content] = placeholders;
|
|
512
|
-
}
|
|
513
|
-
if (!placeholders.length) {
|
|
514
|
-
throw new Error(`i18n postprocess: unmatched placeholder - ${content}`);
|
|
515
|
-
}
|
|
516
|
-
const currentTemplateId = templateIdsStack[templateIdsStack.length - 1];
|
|
517
|
-
let idx = 0;
|
|
518
|
-
// find placeholder index that matches current template id
|
|
519
|
-
for (let i = 0; i < placeholders.length; i++) {
|
|
520
|
-
if (placeholders[i][0] === currentTemplateId) {
|
|
521
|
-
idx = i;
|
|
522
|
-
break;
|
|
523
|
-
}
|
|
524
|
-
}
|
|
525
|
-
// update template id stack based on the current tag extracted
|
|
526
|
-
const [templateId, isCloseTemplateTag, placeholder] = placeholders[idx];
|
|
527
|
-
if (isCloseTemplateTag) {
|
|
528
|
-
templateIdsStack.pop();
|
|
529
|
-
}
|
|
530
|
-
else if (currentTemplateId !== templateId) {
|
|
531
|
-
templateIdsStack.push(templateId);
|
|
532
|
-
}
|
|
533
|
-
// remove processed tag from the list
|
|
534
|
-
placeholders.splice(idx, 1);
|
|
535
|
-
return placeholder;
|
|
536
|
-
});
|
|
537
|
-
}
|
|
538
|
-
// return current result if no replacements specified
|
|
539
|
-
if (!Object.keys(replacements).length) {
|
|
540
|
-
return result;
|
|
541
|
-
}
|
|
542
|
-
/**
|
|
543
|
-
* Step 2: replace all ICU vars (like "VAR_PLURAL")
|
|
544
|
-
*/
|
|
545
|
-
result = result.replace(PP_ICU_VARS_REGEXP, (match, start, key, _type, _idx, end) => {
|
|
546
|
-
return replacements.hasOwnProperty(key) ? `${start}${replacements[key]}${end}` : match;
|
|
547
|
-
});
|
|
548
|
-
/**
|
|
549
|
-
* Step 3: replace all placeholders used inside ICUs in a form of {PLACEHOLDER}
|
|
550
|
-
*/
|
|
551
|
-
result = result.replace(PP_ICU_PLACEHOLDERS_REGEXP, (match, key) => {
|
|
552
|
-
return replacements.hasOwnProperty(key) ? replacements[key] : match;
|
|
553
|
-
});
|
|
554
|
-
/**
|
|
555
|
-
* Step 4: replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�) in case
|
|
556
|
-
* multiple ICUs have the same placeholder name
|
|
557
|
-
*/
|
|
558
|
-
result = result.replace(PP_ICUS_REGEXP, (match, key) => {
|
|
559
|
-
if (replacements.hasOwnProperty(key)) {
|
|
560
|
-
const list = replacements[key];
|
|
561
|
-
if (!list.length) {
|
|
562
|
-
throw new Error(`i18n postprocess: unmatched ICU - ${match} with key: ${key}`);
|
|
563
|
-
}
|
|
564
|
-
return list.shift();
|
|
565
|
-
}
|
|
566
|
-
return match;
|
|
567
|
-
});
|
|
568
|
-
return result;
|
|
569
|
-
}
|
|
570
|
-
/**
|
|
571
|
-
* Translates a translation block marked by `i18nStart` and `i18nEnd`. It inserts the text/ICU nodes
|
|
572
|
-
* into the render tree, moves the placeholder nodes and removes the deleted nodes.
|
|
573
|
-
*
|
|
574
|
-
* @codeGenApi
|
|
575
|
-
*/
|
|
576
|
-
export function ɵɵi18nEnd() {
|
|
577
|
-
const lView = getLView();
|
|
578
|
-
const tView = getTView();
|
|
579
|
-
ngDevMode && assertDefined(tView, `tView should be defined`);
|
|
580
|
-
i18nEndFirstPass(tView, lView);
|
|
581
|
-
// Stop delaying projections
|
|
582
|
-
setDelayProjection(false);
|
|
583
|
-
}
|
|
584
|
-
/**
|
|
585
|
-
* See `i18nEnd` above.
|
|
586
|
-
*/
|
|
587
|
-
function i18nEndFirstPass(tView, lView) {
|
|
588
|
-
ngDevMode &&
|
|
589
|
-
assertEqual(getBindingIndex(), tView.bindingStartIndex, 'i18nEnd should be called before any binding');
|
|
590
|
-
const rootIndex = i18nIndexStack[i18nIndexStackPointer--];
|
|
591
|
-
const tI18n = tView.data[rootIndex + HEADER_OFFSET];
|
|
592
|
-
ngDevMode && assertDefined(tI18n, `You should call i18nStart before i18nEnd`);
|
|
593
|
-
// Find the last node that was added before `i18nEnd`
|
|
594
|
-
const lastCreatedNode = getPreviousOrParentTNode();
|
|
595
|
-
// Read the instructions to insert/move/remove DOM elements
|
|
596
|
-
const visitedNodes = readCreateOpCodes(rootIndex, tI18n.create, tView, lView);
|
|
597
|
-
// Remove deleted nodes
|
|
598
|
-
let index = rootIndex + 1;
|
|
599
|
-
while (index <= lastCreatedNode.index - HEADER_OFFSET) {
|
|
600
|
-
if (visitedNodes.indexOf(index) === -1) {
|
|
601
|
-
removeNode(tView, lView, index, /* markAsDetached */ true);
|
|
602
|
-
}
|
|
603
|
-
// Check if an element has any local refs and skip them
|
|
604
|
-
const tNode = getTNode(tView, index);
|
|
605
|
-
if (tNode &&
|
|
606
|
-
(tNode.type === 0 /* Container */ || tNode.type === 3 /* Element */ ||
|
|
607
|
-
tNode.type === 4 /* ElementContainer */) &&
|
|
608
|
-
tNode.localNames !== null) {
|
|
609
|
-
// Divide by 2 to get the number of local refs,
|
|
610
|
-
// since they are stored as an array that also includes directive indexes,
|
|
611
|
-
// i.e. ["localRef", directiveIndex, ...]
|
|
612
|
-
index += tNode.localNames.length >> 1;
|
|
613
|
-
}
|
|
614
|
-
index++;
|
|
615
|
-
}
|
|
616
|
-
}
|
|
617
|
-
/**
|
|
618
|
-
* Creates and stores the dynamic TNode, and unhooks it from the tree for now.
|
|
619
|
-
*/
|
|
620
|
-
function createDynamicNodeAtIndex(tView, lView, index, type, native, name) {
|
|
621
|
-
const previousOrParentTNode = getPreviousOrParentTNode();
|
|
622
|
-
ngDevMode && assertDataInRange(lView, index + HEADER_OFFSET);
|
|
623
|
-
lView[index + HEADER_OFFSET] = native;
|
|
624
|
-
const tNode = getOrCreateTNode(tView, lView[T_HOST], index, type, name, null);
|
|
625
|
-
// We are creating a dynamic node, the previous tNode might not be pointing at this node.
|
|
626
|
-
// We will link ourselves into the tree later with `appendI18nNode`.
|
|
627
|
-
if (previousOrParentTNode && previousOrParentTNode.next === tNode) {
|
|
628
|
-
previousOrParentTNode.next = null;
|
|
629
|
-
}
|
|
630
|
-
return tNode;
|
|
631
|
-
}
|
|
632
|
-
function readCreateOpCodes(index, createOpCodes, tView, lView) {
|
|
633
|
-
const renderer = lView[RENDERER];
|
|
634
|
-
let currentTNode = null;
|
|
635
|
-
let previousTNode = null;
|
|
636
|
-
const visitedNodes = [];
|
|
637
|
-
for (let i = 0; i < createOpCodes.length; i++) {
|
|
638
|
-
const opCode = createOpCodes[i];
|
|
639
|
-
if (typeof opCode == 'string') {
|
|
640
|
-
const textRNode = createTextNode(opCode, renderer);
|
|
641
|
-
const textNodeIndex = createOpCodes[++i];
|
|
642
|
-
ngDevMode && ngDevMode.rendererCreateTextNode++;
|
|
643
|
-
previousTNode = currentTNode;
|
|
644
|
-
currentTNode =
|
|
645
|
-
createDynamicNodeAtIndex(tView, lView, textNodeIndex, 3 /* Element */, textRNode, null);
|
|
646
|
-
visitedNodes.push(textNodeIndex);
|
|
647
|
-
setIsNotParent();
|
|
648
|
-
}
|
|
649
|
-
else if (typeof opCode == 'number') {
|
|
650
|
-
switch (opCode & 7 /* MASK_OPCODE */) {
|
|
651
|
-
case 1 /* AppendChild */:
|
|
652
|
-
const destinationNodeIndex = opCode >>> 17 /* SHIFT_PARENT */;
|
|
653
|
-
let destinationTNode;
|
|
654
|
-
if (destinationNodeIndex === index) {
|
|
655
|
-
// If the destination node is `i18nStart`, we don't have a
|
|
656
|
-
// top-level node and we should use the host node instead
|
|
657
|
-
destinationTNode = lView[T_HOST];
|
|
658
|
-
}
|
|
659
|
-
else {
|
|
660
|
-
destinationTNode = getTNode(tView, destinationNodeIndex);
|
|
661
|
-
}
|
|
662
|
-
ngDevMode &&
|
|
663
|
-
assertDefined(currentTNode, `You need to create or select a node before you can insert it into the DOM`);
|
|
664
|
-
previousTNode =
|
|
665
|
-
appendI18nNode(tView, currentTNode, destinationTNode, previousTNode, lView);
|
|
666
|
-
break;
|
|
667
|
-
case 0 /* Select */:
|
|
668
|
-
// Negative indicies indicate that a given TNode is a sibling node, not a parent node
|
|
669
|
-
// (see `i18nStartFirstPass` for additional information).
|
|
670
|
-
const isParent = opCode >= 0;
|
|
671
|
-
const nodeIndex = (isParent ? opCode : ~opCode) >>> 3 /* SHIFT_REF */;
|
|
672
|
-
visitedNodes.push(nodeIndex);
|
|
673
|
-
previousTNode = currentTNode;
|
|
674
|
-
currentTNode = getTNode(tView, nodeIndex);
|
|
675
|
-
if (currentTNode) {
|
|
676
|
-
setPreviousOrParentTNode(currentTNode, isParent);
|
|
677
|
-
}
|
|
678
|
-
break;
|
|
679
|
-
case 5 /* ElementEnd */:
|
|
680
|
-
const elementIndex = opCode >>> 3 /* SHIFT_REF */;
|
|
681
|
-
previousTNode = currentTNode = getTNode(tView, elementIndex);
|
|
682
|
-
setPreviousOrParentTNode(currentTNode, false);
|
|
683
|
-
break;
|
|
684
|
-
case 4 /* Attr */:
|
|
685
|
-
const elementNodeIndex = opCode >>> 3 /* SHIFT_REF */;
|
|
686
|
-
const attrName = createOpCodes[++i];
|
|
687
|
-
const attrValue = createOpCodes[++i];
|
|
688
|
-
// This code is used for ICU expressions only, since we don't support
|
|
689
|
-
// directives/components in ICUs, we don't need to worry about inputs here
|
|
690
|
-
elementAttributeInternal(getTNode(tView, elementNodeIndex), lView, attrName, attrValue, null, null);
|
|
691
|
-
break;
|
|
692
|
-
default:
|
|
693
|
-
throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
|
|
694
|
-
}
|
|
695
|
-
}
|
|
696
|
-
else {
|
|
697
|
-
switch (opCode) {
|
|
698
|
-
case COMMENT_MARKER:
|
|
699
|
-
const commentValue = createOpCodes[++i];
|
|
700
|
-
const commentNodeIndex = createOpCodes[++i];
|
|
701
|
-
ngDevMode &&
|
|
702
|
-
assertEqual(typeof commentValue, 'string', `Expected "${commentValue}" to be a comment node value`);
|
|
703
|
-
const commentRNode = renderer.createComment(commentValue);
|
|
704
|
-
ngDevMode && ngDevMode.rendererCreateComment++;
|
|
705
|
-
previousTNode = currentTNode;
|
|
706
|
-
currentTNode = createDynamicNodeAtIndex(tView, lView, commentNodeIndex, 5 /* IcuContainer */, commentRNode, null);
|
|
707
|
-
visitedNodes.push(commentNodeIndex);
|
|
708
|
-
attachPatchData(commentRNode, lView);
|
|
709
|
-
currentTNode.activeCaseIndex = null;
|
|
710
|
-
// We will add the case nodes later, during the update phase
|
|
711
|
-
setIsNotParent();
|
|
712
|
-
break;
|
|
713
|
-
case ELEMENT_MARKER:
|
|
714
|
-
const tagNameValue = createOpCodes[++i];
|
|
715
|
-
const elementNodeIndex = createOpCodes[++i];
|
|
716
|
-
ngDevMode &&
|
|
717
|
-
assertEqual(typeof tagNameValue, 'string', `Expected "${tagNameValue}" to be an element node tag name`);
|
|
718
|
-
const elementRNode = renderer.createElement(tagNameValue);
|
|
719
|
-
ngDevMode && ngDevMode.rendererCreateElement++;
|
|
720
|
-
previousTNode = currentTNode;
|
|
721
|
-
currentTNode = createDynamicNodeAtIndex(tView, lView, elementNodeIndex, 3 /* Element */, elementRNode, tagNameValue);
|
|
722
|
-
visitedNodes.push(elementNodeIndex);
|
|
723
|
-
break;
|
|
724
|
-
default:
|
|
725
|
-
throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
|
|
726
|
-
}
|
|
727
|
-
}
|
|
728
|
-
}
|
|
729
|
-
setIsNotParent();
|
|
730
|
-
return visitedNodes;
|
|
731
|
-
}
|
|
732
|
-
function readUpdateOpCodes(updateOpCodes, icus, bindingsStartIndex, changeMask, tView, lView, bypassCheckBit = false) {
|
|
733
|
-
let caseCreated = false;
|
|
734
|
-
for (let i = 0; i < updateOpCodes.length; i++) {
|
|
735
|
-
// bit code to check if we should apply the next update
|
|
736
|
-
const checkBit = updateOpCodes[i];
|
|
737
|
-
// Number of opCodes to skip until next set of update codes
|
|
738
|
-
const skipCodes = updateOpCodes[++i];
|
|
739
|
-
if (bypassCheckBit || (checkBit & changeMask)) {
|
|
740
|
-
// The value has been updated since last checked
|
|
741
|
-
let value = '';
|
|
742
|
-
for (let j = i + 1; j <= (i + skipCodes); j++) {
|
|
743
|
-
const opCode = updateOpCodes[j];
|
|
744
|
-
if (typeof opCode == 'string') {
|
|
745
|
-
value += opCode;
|
|
746
|
-
}
|
|
747
|
-
else if (typeof opCode == 'number') {
|
|
748
|
-
if (opCode < 0) {
|
|
749
|
-
// It's a binding index whose value is negative
|
|
750
|
-
value += renderStringify(lView[bindingsStartIndex - opCode]);
|
|
751
|
-
}
|
|
752
|
-
else {
|
|
753
|
-
const nodeIndex = opCode >>> 2 /* SHIFT_REF */;
|
|
754
|
-
let tIcuIndex;
|
|
755
|
-
let tIcu;
|
|
756
|
-
let icuTNode;
|
|
757
|
-
switch (opCode & 3 /* MASK_OPCODE */) {
|
|
758
|
-
case 1 /* Attr */:
|
|
759
|
-
const propName = updateOpCodes[++j];
|
|
760
|
-
const sanitizeFn = updateOpCodes[++j];
|
|
761
|
-
elementPropertyInternal(tView, getTNode(tView, nodeIndex), lView, propName, value, lView[RENDERER], sanitizeFn, false);
|
|
762
|
-
break;
|
|
763
|
-
case 0 /* Text */:
|
|
764
|
-
textBindingInternal(lView, nodeIndex, value);
|
|
765
|
-
break;
|
|
766
|
-
case 2 /* IcuSwitch */:
|
|
767
|
-
tIcuIndex = updateOpCodes[++j];
|
|
768
|
-
tIcu = icus[tIcuIndex];
|
|
769
|
-
icuTNode = getTNode(tView, nodeIndex);
|
|
770
|
-
// If there is an active case, delete the old nodes
|
|
771
|
-
if (icuTNode.activeCaseIndex !== null) {
|
|
772
|
-
const removeCodes = tIcu.remove[icuTNode.activeCaseIndex];
|
|
773
|
-
for (let k = 0; k < removeCodes.length; k++) {
|
|
774
|
-
const removeOpCode = removeCodes[k];
|
|
775
|
-
switch (removeOpCode & 7 /* MASK_OPCODE */) {
|
|
776
|
-
case 3 /* Remove */:
|
|
777
|
-
const nodeIndex = removeOpCode >>> 3 /* SHIFT_REF */;
|
|
778
|
-
// Remove DOM element, but do *not* mark TNode as detached, since we are
|
|
779
|
-
// just switching ICU cases (while keeping the same TNode), so a DOM element
|
|
780
|
-
// representing a new ICU case will be re-created.
|
|
781
|
-
removeNode(tView, lView, nodeIndex, /* markAsDetached */ false);
|
|
782
|
-
break;
|
|
783
|
-
case 6 /* RemoveNestedIcu */:
|
|
784
|
-
const nestedIcuNodeIndex = removeCodes[k + 1] >>> 3 /* SHIFT_REF */;
|
|
785
|
-
const nestedIcuTNode = getTNode(tView, nestedIcuNodeIndex);
|
|
786
|
-
const activeIndex = nestedIcuTNode.activeCaseIndex;
|
|
787
|
-
if (activeIndex !== null) {
|
|
788
|
-
const nestedIcuTIndex = removeOpCode >>> 3 /* SHIFT_REF */;
|
|
789
|
-
const nestedTIcu = icus[nestedIcuTIndex];
|
|
790
|
-
addAllToArray(nestedTIcu.remove[activeIndex], removeCodes);
|
|
791
|
-
}
|
|
792
|
-
break;
|
|
793
|
-
}
|
|
794
|
-
}
|
|
795
|
-
}
|
|
796
|
-
// Update the active caseIndex
|
|
797
|
-
const caseIndex = getCaseIndex(tIcu, value);
|
|
798
|
-
icuTNode.activeCaseIndex = caseIndex !== -1 ? caseIndex : null;
|
|
799
|
-
if (caseIndex > -1) {
|
|
800
|
-
// Add the nodes for the new case
|
|
801
|
-
readCreateOpCodes(-1, tIcu.create[caseIndex], tView, lView);
|
|
802
|
-
caseCreated = true;
|
|
803
|
-
}
|
|
804
|
-
break;
|
|
805
|
-
case 3 /* IcuUpdate */:
|
|
806
|
-
tIcuIndex = updateOpCodes[++j];
|
|
807
|
-
tIcu = icus[tIcuIndex];
|
|
808
|
-
icuTNode = getTNode(tView, nodeIndex);
|
|
809
|
-
if (icuTNode.activeCaseIndex !== null) {
|
|
810
|
-
readUpdateOpCodes(tIcu.update[icuTNode.activeCaseIndex], icus, bindingsStartIndex, changeMask, tView, lView, caseCreated);
|
|
811
|
-
}
|
|
812
|
-
break;
|
|
813
|
-
}
|
|
814
|
-
}
|
|
815
|
-
}
|
|
816
|
-
}
|
|
817
|
-
}
|
|
818
|
-
i += skipCodes;
|
|
819
|
-
}
|
|
820
|
-
}
|
|
821
|
-
function removeNode(tView, lView, index, markAsDetached) {
|
|
822
|
-
const removedPhTNode = getTNode(tView, index);
|
|
823
|
-
const removedPhRNode = getNativeByIndex(index, lView);
|
|
824
|
-
if (removedPhRNode) {
|
|
825
|
-
nativeRemoveNode(lView[RENDERER], removedPhRNode);
|
|
826
|
-
}
|
|
827
|
-
const slotValue = load(lView, index);
|
|
828
|
-
if (isLContainer(slotValue)) {
|
|
829
|
-
const lContainer = slotValue;
|
|
830
|
-
if (removedPhTNode.type !== 0 /* Container */) {
|
|
831
|
-
nativeRemoveNode(lView[RENDERER], lContainer[NATIVE]);
|
|
832
|
-
}
|
|
833
|
-
}
|
|
834
|
-
if (markAsDetached) {
|
|
835
|
-
// Define this node as detached to avoid projecting it later
|
|
836
|
-
removedPhTNode.flags |= 64 /* isDetached */;
|
|
837
|
-
}
|
|
838
|
-
ngDevMode && ngDevMode.rendererRemoveNode++;
|
|
839
|
-
}
|
|
840
|
-
/**
|
|
841
|
-
*
|
|
842
|
-
* Use this instruction to create a translation block that doesn't contain any placeholder.
|
|
843
|
-
* It calls both {@link i18nStart} and {@link i18nEnd} in one instruction.
|
|
844
|
-
*
|
|
845
|
-
* The translation `message` is the value which is locale specific. The translation string may
|
|
846
|
-
* contain placeholders which associate inner elements and sub-templates within the translation.
|
|
847
|
-
*
|
|
848
|
-
* The translation `message` placeholders are:
|
|
849
|
-
* - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be
|
|
850
|
-
* interpolated into. The placeholder `index` points to the expression binding index. An optional
|
|
851
|
-
* `block` that matches the sub-template in which it was declared.
|
|
852
|
-
* - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*: Marks the beginning
|
|
853
|
-
* and end of DOM element that were embedded in the original translation block. The placeholder
|
|
854
|
-
* `index` points to the element index in the template instructions set. An optional `block` that
|
|
855
|
-
* matches the sub-template in which it was declared.
|
|
856
|
-
* - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be
|
|
857
|
-
* split up and translated separately in each angular template function. The `index` points to the
|
|
858
|
-
* `template` instruction index. A `block` that matches the sub-template in which it was declared.
|
|
859
|
-
*
|
|
860
|
-
* @param index A unique index of the translation in the static block.
|
|
861
|
-
* @param message The translation message.
|
|
862
|
-
* @param subTemplateIndex Optional sub-template index in the `message`.
|
|
863
|
-
*
|
|
864
|
-
* @codeGenApi
|
|
865
|
-
*/
|
|
866
|
-
export function ɵɵi18n(index, message, subTemplateIndex) {
|
|
867
|
-
ɵɵi18nStart(index, message, subTemplateIndex);
|
|
868
|
-
ɵɵi18nEnd();
|
|
869
|
-
}
|
|
870
|
-
/**
|
|
871
|
-
* Marks a list of attributes as translatable.
|
|
872
|
-
*
|
|
873
|
-
* @param index A unique index in the static block
|
|
874
|
-
* @param values
|
|
875
|
-
*
|
|
876
|
-
* @codeGenApi
|
|
877
|
-
*/
|
|
878
|
-
export function ɵɵi18nAttributes(index, values) {
|
|
879
|
-
const lView = getLView();
|
|
880
|
-
const tView = getTView();
|
|
881
|
-
ngDevMode && assertDefined(tView, `tView should be defined`);
|
|
882
|
-
i18nAttributesFirstPass(lView, tView, index, values);
|
|
883
|
-
}
|
|
884
|
-
/**
|
|
885
|
-
* See `i18nAttributes` above.
|
|
886
|
-
*/
|
|
887
|
-
function i18nAttributesFirstPass(lView, tView, index, values) {
|
|
888
|
-
const previousElement = getPreviousOrParentTNode();
|
|
889
|
-
const previousElementIndex = previousElement.index - HEADER_OFFSET;
|
|
890
|
-
const updateOpCodes = [];
|
|
891
|
-
for (let i = 0; i < values.length; i += 2) {
|
|
892
|
-
const attrName = values[i];
|
|
893
|
-
const message = values[i + 1];
|
|
894
|
-
const parts = message.split(ICU_REGEXP);
|
|
895
|
-
for (let j = 0; j < parts.length; j++) {
|
|
896
|
-
const value = parts[j];
|
|
897
|
-
if (j & 1) {
|
|
898
|
-
// Odd indexes are ICU expressions
|
|
899
|
-
// TODO(ocombe): support ICU expressions in attributes
|
|
900
|
-
throw new Error('ICU expressions are not yet supported in attributes');
|
|
901
|
-
}
|
|
902
|
-
else if (value !== '') {
|
|
903
|
-
// Even indexes are text (including bindings)
|
|
904
|
-
const hasBinding = !!value.match(BINDING_REGEXP);
|
|
905
|
-
if (hasBinding) {
|
|
906
|
-
if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
|
|
907
|
-
addAllToArray(generateBindingUpdateOpCodes(value, previousElementIndex, attrName), updateOpCodes);
|
|
908
|
-
}
|
|
909
|
-
}
|
|
910
|
-
else {
|
|
911
|
-
const tNode = getTNode(tView, previousElementIndex);
|
|
912
|
-
// Set attributes for Elements only, for other types (like ElementContainer),
|
|
913
|
-
// only set inputs below
|
|
914
|
-
if (tNode.type === 3 /* Element */) {
|
|
915
|
-
elementAttributeInternal(tNode, lView, attrName, value, null, null);
|
|
916
|
-
}
|
|
917
|
-
// Check if that attribute is a directive input
|
|
918
|
-
const dataValue = tNode.inputs !== null && tNode.inputs[attrName];
|
|
919
|
-
if (dataValue) {
|
|
920
|
-
setInputsForProperty(tView, lView, dataValue, attrName, value);
|
|
921
|
-
if (ngDevMode) {
|
|
922
|
-
const element = getNativeByIndex(previousElementIndex, lView);
|
|
923
|
-
setNgReflectProperties(lView, element, tNode.type, dataValue, value);
|
|
924
|
-
}
|
|
925
|
-
}
|
|
926
|
-
}
|
|
927
|
-
}
|
|
928
|
-
}
|
|
929
|
-
}
|
|
930
|
-
if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {
|
|
931
|
-
tView.data[index + HEADER_OFFSET] = updateOpCodes;
|
|
932
|
-
}
|
|
933
|
-
}
|
|
934
|
-
let changeMask = 0b0;
|
|
935
|
-
let shiftsCounter = 0;
|
|
936
|
-
/**
|
|
937
|
-
* Stores the values of the bindings during each update cycle in order to determine if we need to
|
|
938
|
-
* update the translated nodes.
|
|
939
|
-
*
|
|
940
|
-
* @param value The binding's value
|
|
941
|
-
* @returns This function returns itself so that it may be chained
|
|
942
|
-
* (e.g. `i18nExp(ctx.name)(ctx.title)`)
|
|
943
|
-
*
|
|
944
|
-
* @codeGenApi
|
|
945
|
-
*/
|
|
946
|
-
export function ɵɵi18nExp(value) {
|
|
947
|
-
const lView = getLView();
|
|
948
|
-
if (bindingUpdated(lView, nextBindingIndex(), value)) {
|
|
949
|
-
changeMask = changeMask | (1 << shiftsCounter);
|
|
950
|
-
}
|
|
951
|
-
shiftsCounter++;
|
|
952
|
-
return ɵɵi18nExp;
|
|
953
|
-
}
|
|
954
|
-
/**
|
|
955
|
-
* Updates a translation block or an i18n attribute when the bindings have changed.
|
|
956
|
-
*
|
|
957
|
-
* @param index Index of either {@link i18nStart} (translation block) or {@link i18nAttributes}
|
|
958
|
-
* (i18n attribute) on which it should update the content.
|
|
959
|
-
*
|
|
960
|
-
* @codeGenApi
|
|
961
|
-
*/
|
|
962
|
-
export function ɵɵi18nApply(index) {
|
|
963
|
-
if (shiftsCounter) {
|
|
964
|
-
const tView = getTView();
|
|
965
|
-
ngDevMode && assertDefined(tView, `tView should be defined`);
|
|
966
|
-
const tI18n = tView.data[index + HEADER_OFFSET];
|
|
967
|
-
let updateOpCodes;
|
|
968
|
-
let icus = null;
|
|
969
|
-
if (Array.isArray(tI18n)) {
|
|
970
|
-
updateOpCodes = tI18n;
|
|
971
|
-
}
|
|
972
|
-
else {
|
|
973
|
-
updateOpCodes = tI18n.update;
|
|
974
|
-
icus = tI18n.icus;
|
|
975
|
-
}
|
|
976
|
-
const bindingsStartIndex = getBindingIndex() - shiftsCounter - 1;
|
|
977
|
-
const lView = getLView();
|
|
978
|
-
readUpdateOpCodes(updateOpCodes, icus, bindingsStartIndex, changeMask, tView, lView);
|
|
979
|
-
// Reset changeMask & maskBit to default for the next update cycle
|
|
980
|
-
changeMask = 0b0;
|
|
981
|
-
shiftsCounter = 0;
|
|
982
|
-
}
|
|
983
|
-
}
|
|
984
|
-
/**
|
|
985
|
-
* Returns the index of the current case of an ICU expression depending on the main binding value
|
|
986
|
-
*
|
|
987
|
-
* @param icuExpression
|
|
988
|
-
* @param bindingValue The value of the main binding used by this ICU expression
|
|
989
|
-
*/
|
|
990
|
-
function getCaseIndex(icuExpression, bindingValue) {
|
|
991
|
-
let index = icuExpression.cases.indexOf(bindingValue);
|
|
992
|
-
if (index === -1) {
|
|
993
|
-
switch (icuExpression.type) {
|
|
994
|
-
case 1 /* plural */: {
|
|
995
|
-
const resolvedCase = getPluralCase(bindingValue, getLocaleId());
|
|
996
|
-
index = icuExpression.cases.indexOf(resolvedCase);
|
|
997
|
-
if (index === -1 && resolvedCase !== 'other') {
|
|
998
|
-
index = icuExpression.cases.indexOf('other');
|
|
999
|
-
}
|
|
1000
|
-
break;
|
|
1001
|
-
}
|
|
1002
|
-
case 0 /* select */: {
|
|
1003
|
-
index = icuExpression.cases.indexOf('other');
|
|
1004
|
-
break;
|
|
1005
|
-
}
|
|
1006
|
-
}
|
|
1007
|
-
}
|
|
1008
|
-
return index;
|
|
1009
|
-
}
|
|
1010
|
-
/**
|
|
1011
|
-
* Generate the OpCodes for ICU expressions.
|
|
1012
|
-
*
|
|
1013
|
-
* @param tIcus
|
|
1014
|
-
* @param icuExpression
|
|
1015
|
-
* @param startIndex
|
|
1016
|
-
* @param expandoStartIndex
|
|
1017
|
-
*/
|
|
1018
|
-
function icuStart(tIcus, icuExpression, startIndex, expandoStartIndex) {
|
|
1019
|
-
const createCodes = [];
|
|
1020
|
-
const removeCodes = [];
|
|
1021
|
-
const updateCodes = [];
|
|
1022
|
-
const vars = [];
|
|
1023
|
-
const childIcus = [];
|
|
1024
|
-
for (let i = 0; i < icuExpression.values.length; i++) {
|
|
1025
|
-
// Each value is an array of strings & other ICU expressions
|
|
1026
|
-
const valueArr = icuExpression.values[i];
|
|
1027
|
-
const nestedIcus = [];
|
|
1028
|
-
for (let j = 0; j < valueArr.length; j++) {
|
|
1029
|
-
const value = valueArr[j];
|
|
1030
|
-
if (typeof value !== 'string') {
|
|
1031
|
-
// It is an nested ICU expression
|
|
1032
|
-
const icuIndex = nestedIcus.push(value) - 1;
|
|
1033
|
-
// Replace nested ICU expression by a comment node
|
|
1034
|
-
valueArr[j] = `<!--�${icuIndex}�-->`;
|
|
1035
|
-
}
|
|
1036
|
-
}
|
|
1037
|
-
const icuCase = parseIcuCase(valueArr.join(''), startIndex, nestedIcus, tIcus, expandoStartIndex);
|
|
1038
|
-
createCodes.push(icuCase.create);
|
|
1039
|
-
removeCodes.push(icuCase.remove);
|
|
1040
|
-
updateCodes.push(icuCase.update);
|
|
1041
|
-
vars.push(icuCase.vars);
|
|
1042
|
-
childIcus.push(icuCase.childIcus);
|
|
1043
|
-
}
|
|
1044
|
-
const tIcu = {
|
|
1045
|
-
type: icuExpression.type,
|
|
1046
|
-
vars,
|
|
1047
|
-
childIcus,
|
|
1048
|
-
cases: icuExpression.cases,
|
|
1049
|
-
create: createCodes,
|
|
1050
|
-
remove: removeCodes,
|
|
1051
|
-
update: updateCodes
|
|
1052
|
-
};
|
|
1053
|
-
tIcus.push(tIcu);
|
|
1054
|
-
// Adding the maximum possible of vars needed (based on the cases with the most vars)
|
|
1055
|
-
i18nVarsCount += Math.max(...vars);
|
|
1056
|
-
}
|
|
1057
|
-
/**
|
|
1058
|
-
* Transforms a string template into an HTML template and a list of instructions used to update
|
|
1059
|
-
* attributes or nodes that contain bindings.
|
|
1060
|
-
*
|
|
1061
|
-
* @param unsafeHtml The string to parse
|
|
1062
|
-
* @param parentIndex
|
|
1063
|
-
* @param nestedIcus
|
|
1064
|
-
* @param tIcus
|
|
1065
|
-
* @param expandoStartIndex
|
|
1066
|
-
*/
|
|
1067
|
-
function parseIcuCase(unsafeHtml, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
|
|
1068
|
-
const inertBodyHelper = getInertBodyHelper(getDocument());
|
|
1069
|
-
const inertBodyElement = inertBodyHelper.getInertBodyElement(unsafeHtml);
|
|
1070
|
-
if (!inertBodyElement) {
|
|
1071
|
-
throw new Error('Unable to generate inert body element');
|
|
1072
|
-
}
|
|
1073
|
-
const wrapper = getTemplateContent(inertBodyElement) || inertBodyElement;
|
|
1074
|
-
const opCodes = { vars: 0, childIcus: [], create: [], remove: [], update: [] };
|
|
1075
|
-
parseNodes(wrapper.firstChild, opCodes, parentIndex, nestedIcus, tIcus, expandoStartIndex);
|
|
1076
|
-
return opCodes;
|
|
1077
|
-
}
|
|
1078
|
-
const NESTED_ICU = /�(\d+)�/;
|
|
1079
|
-
/**
|
|
1080
|
-
* Parses a node, its children and its siblings, and generates the mutate & update OpCodes.
|
|
1081
|
-
*
|
|
1082
|
-
* @param currentNode The first node to parse
|
|
1083
|
-
* @param icuCase The data for the ICU expression case that contains those nodes
|
|
1084
|
-
* @param parentIndex Index of the current node's parent
|
|
1085
|
-
* @param nestedIcus Data for the nested ICU expressions that this case contains
|
|
1086
|
-
* @param tIcus Data for all ICU expressions of the current message
|
|
1087
|
-
* @param expandoStartIndex Expando start index for the current ICU expression
|
|
1088
|
-
*/
|
|
1089
|
-
function parseNodes(currentNode, icuCase, parentIndex, nestedIcus, tIcus, expandoStartIndex) {
|
|
1090
|
-
if (currentNode) {
|
|
1091
|
-
const nestedIcusToCreate = [];
|
|
1092
|
-
while (currentNode) {
|
|
1093
|
-
const nextNode = currentNode.nextSibling;
|
|
1094
|
-
const newIndex = expandoStartIndex + ++icuCase.vars;
|
|
1095
|
-
switch (currentNode.nodeType) {
|
|
1096
|
-
case Node.ELEMENT_NODE:
|
|
1097
|
-
const element = currentNode;
|
|
1098
|
-
const tagName = element.tagName.toLowerCase();
|
|
1099
|
-
if (!VALID_ELEMENTS.hasOwnProperty(tagName)) {
|
|
1100
|
-
// This isn't a valid element, we won't create an element for it
|
|
1101
|
-
icuCase.vars--;
|
|
1102
|
-
}
|
|
1103
|
-
else {
|
|
1104
|
-
icuCase.create.push(ELEMENT_MARKER, tagName, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
1105
|
-
const elAttrs = element.attributes;
|
|
1106
|
-
for (let i = 0; i < elAttrs.length; i++) {
|
|
1107
|
-
const attr = elAttrs.item(i);
|
|
1108
|
-
const lowerAttrName = attr.name.toLowerCase();
|
|
1109
|
-
const hasBinding = !!attr.value.match(BINDING_REGEXP);
|
|
1110
|
-
// we assume the input string is safe, unless it's using a binding
|
|
1111
|
-
if (hasBinding) {
|
|
1112
|
-
if (VALID_ATTRS.hasOwnProperty(lowerAttrName)) {
|
|
1113
|
-
if (URI_ATTRS[lowerAttrName]) {
|
|
1114
|
-
addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, _sanitizeUrl), icuCase.update);
|
|
1115
|
-
}
|
|
1116
|
-
else if (SRCSET_ATTRS[lowerAttrName]) {
|
|
1117
|
-
addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name, sanitizeSrcset), icuCase.update);
|
|
1118
|
-
}
|
|
1119
|
-
else {
|
|
1120
|
-
addAllToArray(generateBindingUpdateOpCodes(attr.value, newIndex, attr.name), icuCase.update);
|
|
1121
|
-
}
|
|
1122
|
-
}
|
|
1123
|
-
else {
|
|
1124
|
-
ngDevMode &&
|
|
1125
|
-
console.warn(`WARNING: ignoring unsafe attribute value ${lowerAttrName} on element ${tagName} (see http://g.co/ng/security#xss)`);
|
|
1126
|
-
}
|
|
1127
|
-
}
|
|
1128
|
-
else {
|
|
1129
|
-
icuCase.create.push(newIndex << 3 /* SHIFT_REF */ | 4 /* Attr */, attr.name, attr.value);
|
|
1130
|
-
}
|
|
1131
|
-
}
|
|
1132
|
-
// Parse the children of this node (if any)
|
|
1133
|
-
parseNodes(currentNode.firstChild, icuCase, newIndex, nestedIcus, tIcus, expandoStartIndex);
|
|
1134
|
-
// Remove the parent node after the children
|
|
1135
|
-
icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
|
|
1136
|
-
}
|
|
1137
|
-
break;
|
|
1138
|
-
case Node.TEXT_NODE:
|
|
1139
|
-
const value = currentNode.textContent || '';
|
|
1140
|
-
const hasBinding = value.match(BINDING_REGEXP);
|
|
1141
|
-
icuCase.create.push(hasBinding ? '' : value, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
1142
|
-
icuCase.remove.push(newIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
|
|
1143
|
-
if (hasBinding) {
|
|
1144
|
-
addAllToArray(generateBindingUpdateOpCodes(value, newIndex), icuCase.update);
|
|
1145
|
-
}
|
|
1146
|
-
break;
|
|
1147
|
-
case Node.COMMENT_NODE:
|
|
1148
|
-
// Check if the comment node is a placeholder for a nested ICU
|
|
1149
|
-
const match = NESTED_ICU.exec(currentNode.textContent || '');
|
|
1150
|
-
if (match) {
|
|
1151
|
-
const nestedIcuIndex = parseInt(match[1], 10);
|
|
1152
|
-
const newLocal = ngDevMode ? `nested ICU ${nestedIcuIndex}` : '';
|
|
1153
|
-
// Create the comment node that will anchor the ICU expression
|
|
1154
|
-
icuCase.create.push(COMMENT_MARKER, newLocal, newIndex, parentIndex << 17 /* SHIFT_PARENT */ | 1 /* AppendChild */);
|
|
1155
|
-
const nestedIcu = nestedIcus[nestedIcuIndex];
|
|
1156
|
-
nestedIcusToCreate.push([nestedIcu, newIndex]);
|
|
1157
|
-
}
|
|
1158
|
-
else {
|
|
1159
|
-
// We do not handle any other type of comment
|
|
1160
|
-
icuCase.vars--;
|
|
1161
|
-
}
|
|
1162
|
-
break;
|
|
1163
|
-
default:
|
|
1164
|
-
// We do not handle any other type of element
|
|
1165
|
-
icuCase.vars--;
|
|
1166
|
-
}
|
|
1167
|
-
currentNode = nextNode;
|
|
1168
|
-
}
|
|
1169
|
-
for (let i = 0; i < nestedIcusToCreate.length; i++) {
|
|
1170
|
-
const nestedIcu = nestedIcusToCreate[i][0];
|
|
1171
|
-
const nestedIcuNodeIndex = nestedIcusToCreate[i][1];
|
|
1172
|
-
icuStart(tIcus, nestedIcu, nestedIcuNodeIndex, expandoStartIndex + icuCase.vars);
|
|
1173
|
-
// Since this is recursive, the last TIcu that was pushed is the one we want
|
|
1174
|
-
const nestTIcuIndex = tIcus.length - 1;
|
|
1175
|
-
icuCase.vars += Math.max(...tIcus[nestTIcuIndex].vars);
|
|
1176
|
-
icuCase.childIcus.push(nestTIcuIndex);
|
|
1177
|
-
const mask = getBindingMask(nestedIcu);
|
|
1178
|
-
icuCase.update.push(toMaskBit(nestedIcu.mainBinding), // mask of the main binding
|
|
1179
|
-
3, // skip 3 opCodes if not changed
|
|
1180
|
-
-1 - nestedIcu.mainBinding, nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 2 /* IcuSwitch */, nestTIcuIndex, mask, // mask of all the bindings of this ICU expression
|
|
1181
|
-
2, // skip 2 opCodes if not changed
|
|
1182
|
-
nestedIcuNodeIndex << 2 /* SHIFT_REF */ | 3 /* IcuUpdate */, nestTIcuIndex);
|
|
1183
|
-
icuCase.remove.push(nestTIcuIndex << 3 /* SHIFT_REF */ | 6 /* RemoveNestedIcu */, nestedIcuNodeIndex << 3 /* SHIFT_REF */ | 3 /* Remove */);
|
|
1184
|
-
}
|
|
1185
|
-
}
|
|
1186
|
-
}
|
|
1187
|
-
/**
|
|
1188
|
-
* Angular Dart introduced &ngsp; as a placeholder for non-removable space, see:
|
|
1189
|
-
* https://github.com/dart-lang/angular/blob/0bb611387d29d65b5af7f9d2515ab571fd3fbee4/_tests/test/compiler/preserve_whitespace_test.dart#L25-L32
|
|
1190
|
-
* In Angular Dart &ngsp; is converted to the 0xE500 PUA (Private Use Areas) unicode character
|
|
1191
|
-
* and later on replaced by a space. We are re-implementing the same idea here, since translations
|
|
1192
|
-
* might contain this special character.
|
|
1193
|
-
*/
|
|
1194
|
-
const NGSP_UNICODE_REGEXP = /\uE500/g;
|
|
1195
|
-
function replaceNgsp(value) {
|
|
1196
|
-
return value.replace(NGSP_UNICODE_REGEXP, ' ');
|
|
1197
|
-
}
|
|
1198
|
-
/**
|
|
1199
|
-
* The locale id that the application is currently using (for translations and ICU expressions).
|
|
1200
|
-
* This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
|
|
1201
|
-
* but is now defined as a global value.
|
|
1202
|
-
*/
|
|
1203
|
-
let LOCALE_ID = DEFAULT_LOCALE_ID;
|
|
1204
|
-
/**
|
|
1205
|
-
* Sets the locale id that will be used for translations and ICU expressions.
|
|
1206
|
-
* This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
|
|
1207
|
-
* but is now defined as a global value.
|
|
1208
|
-
*
|
|
1209
|
-
* @param localeId
|
|
1210
|
-
*/
|
|
1211
|
-
export function setLocaleId(localeId) {
|
|
1212
|
-
assertDefined(localeId, `Expected localeId to be defined`);
|
|
1213
|
-
if (typeof localeId === 'string') {
|
|
1214
|
-
LOCALE_ID = localeId.toLowerCase().replace(/_/g, '-');
|
|
1215
|
-
}
|
|
1216
|
-
}
|
|
1217
|
-
/**
|
|
1218
|
-
* Gets the locale id that will be used for translations and ICU expressions.
|
|
1219
|
-
* This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine
|
|
1220
|
-
* but is now defined as a global value.
|
|
1221
|
-
*/
|
|
1222
|
-
export function getLocaleId() {
|
|
1223
|
-
return LOCALE_ID;
|
|
1224
|
-
}
|
|
1225
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/render3/i18n.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_i18n_closure_mode';\n\nimport {DEFAULT_LOCALE_ID, getPluralCase} from '../i18n/localization';\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 {assertDataInRange, assertDefined, assertEqual} from '../util/assert';\n\nimport {bindingUpdated} from './bindings';\nimport {attachPatchData} from './context_discovery';\nimport {setDelayProjection} from './instructions/all';\nimport {attachI18nOpCodesDebug} from './instructions/lview_debug';\nimport {allocExpando, elementAttributeInternal, elementPropertyInternal, getOrCreateTNode, setInputsForProperty, setNgReflectProperties, textBindingInternal} from './instructions/shared';\nimport {LContainer, NATIVE} from './interfaces/container';\nimport {getDocument} from './interfaces/document';\nimport {COMMENT_MARKER, ELEMENT_MARKER, I18nMutateOpCode, I18nMutateOpCodes, I18nUpdateOpCode, I18nUpdateOpCodes, IcuType, TI18n, TIcu} from './interfaces/i18n';\nimport {TElementNode, TIcuContainerNode, TNode, TNodeFlags, TNodeType, TProjectionNode} from './interfaces/node';\nimport {RComment, RElement, RText} from './interfaces/renderer';\nimport {SanitizerFn} from './interfaces/sanitization';\nimport {isLContainer} from './interfaces/type_checks';\nimport {HEADER_OFFSET, LView, RENDERER, T_HOST, TVIEW, TView} from './interfaces/view';\nimport {appendChild, applyProjection, createTextNode, nativeRemoveNode} from './node_manipulation';\nimport {getBindingIndex, getIsParent, getLView, getPreviousOrParentTNode, getTView, nextBindingIndex, setIsNotParent, setPreviousOrParentTNode} from './state';\nimport {renderStringify} from './util/misc_utils';\nimport {getNativeByIndex, getNativeByTNode, getTNode, load} from './util/view_utils';\n\n\nconst MARKER = `�`;\nconst ICU_BLOCK_REGEXP = /^\\s*(�\\d+:?\\d*�)\\s*,\\s*(select|plural)\\s*,/;\nconst SUBTEMPLATE_REGEXP = /�\\/?\\*(\\d+:\\d+)�/gi;\nconst PH_REGEXP = /�(\\/?[#*!]\\d+):?\\d*�/gi;\nconst BINDING_REGEXP = /�(\\d+):?\\d*�/gi;\nconst ICU_REGEXP = /({\\s*�\\d+:?\\d*�\\s*,\\s*\\S{6}\\s*,[\\s\\S]*})/gi;\nconst enum TagType {\n  ELEMENT = '#',\n  TEMPLATE = '*',\n  PROJECTION = '!',\n}\n\n// i18nPostprocess consts\nconst ROOT_TEMPLATE_ID = 0;\nconst PP_MULTI_VALUE_PLACEHOLDERS_REGEXP = /\\[(�.+?�?)\\]/;\nconst PP_PLACEHOLDERS_REGEXP = /\\[(�.+?�?)\\]|(�\\/?\\*\\d+:\\d+�)/g;\nconst PP_ICU_VARS_REGEXP = /({\\s*)(VAR_(PLURAL|SELECT)(_\\d+)?)(\\s*,)/g;\nconst PP_ICU_PLACEHOLDERS_REGEXP = /{([A-Z0-9_]+)}/g;\nconst PP_ICUS_REGEXP = /�I18N_EXP_(ICU(_\\d+)?)�/g;\nconst PP_CLOSE_TEMPLATE_REGEXP = /\\/\\*/;\nconst PP_TEMPLATE_ID_REGEXP = /\\d+\\:(\\d+)/;\n\n// Parsed placeholder structure used in postprocessing (within `i18nPostprocess` function)\n// Contains the following fields: [templateId, isCloseTemplateTag, placeholder]\ntype PostprocessPlaceholder = [number, boolean, string];\n\ninterface IcuExpression {\n  type: IcuType;\n  mainBinding: number;\n  cases: string[];\n  values: (string|IcuExpression)[][];\n}\n\ninterface IcuCase {\n  /**\n   * Number of slots to allocate in expando for this case.\n   *\n   * This is the max number of DOM elements which will be created by this i18n + ICU blocks. When\n   * the DOM elements are being created they are stored in the EXPANDO, so that update OpCodes can\n   * write into them.\n   */\n  vars: number;\n\n  /**\n   * An optional array of child/sub ICUs.\n   */\n  childIcus: number[];\n\n  /**\n   * A set of OpCodes to apply in order to build up the DOM render tree for the ICU\n   */\n  create: I18nMutateOpCodes;\n\n  /**\n   * A set of OpCodes to apply in order to destroy the DOM render tree for the ICU.\n   */\n  remove: I18nMutateOpCodes;\n\n  /**\n   * A set of OpCodes to apply in order to update the DOM render tree for the ICU bindings.\n   */\n  update: I18nUpdateOpCodes;\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 * 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 */\nfunction 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 * 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 * 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 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 */\nfunction 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  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\nconst i18nIndexStack: number[] = [];\nlet i18nIndexStackPointer = -1;\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\nconst parentIndexStack: number[] = [];\n\n/**\n * Marks a block of text as translatable.\n *\n * The instructions `i18nStart` and `i18nEnd` mark the translation block in the template.\n * The translation `message` is the value which is locale specific. The translation string may\n * contain placeholders which associate inner elements and sub-templates within the translation.\n *\n * The translation `message` placeholders are:\n * - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be\n *   interpolated into. The placeholder `index` points to the expression binding index. An optional\n *   `block` that matches the sub-template in which it was declared.\n * - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*:  Marks the beginning\n *   and end of DOM element that were embedded in the original translation block. The placeholder\n *   `index` points to the element index in the template instructions set. An optional `block` that\n *   matches the sub-template in which it was declared.\n * - `�!{index}(:{block})�`/`�/!{index}(:{block})�`: *Projection Placeholder*:  Marks the\n *   beginning and end of <ng-content> that was embedded in the original translation block.\n *   The placeholder `index` points to the element index in the template instructions set.\n *   An optional `block` that matches the sub-template in which it was declared.\n * - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be\n *   split up and translated separately in each angular template function. The `index` points to the\n *   `template` instruction index. A `block` that matches the sub-template in which it was declared.\n *\n * @param index A unique index of the translation in the static block.\n * @param message The translation message.\n * @param subTemplateIndex Optional sub-template index in the `message`.\n *\n * @codeGenApi\n */\nexport function ɵɵi18nStart(index: number, message: string, subTemplateIndex?: number): void {\n  const tView = getTView();\n  ngDevMode && assertDefined(tView, `tView should be defined`);\n  i18nIndexStack[++i18nIndexStackPointer] = index;\n  // We need to delay projections until `i18nEnd`\n  setDelayProjection(true);\n  if (tView.firstCreatePass && tView.data[index + HEADER_OFFSET] === null) {\n    i18nStartFirstPass(getLView(), tView, index, message, subTemplateIndex);\n  }\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\nfunction allocNodeIndex(startIndex: number): number {\n  return startIndex + i18nVarsCount++;\n}\n\n/**\n * See `i18nStart` above.\n */\nfunction 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 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  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  ngDevMode &&\n      attachI18nOpCodesDebug(\n          createOpCodes, updateOpCodes, icuExpressions.length ? icuExpressions : null, lView);\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\nfunction appendI18nNode(\n    tView: TView, tNode: TNode, parentTNode: TNode, previousTNode: TNode|null,\n    lView: LView): TNode {\n  ngDevMode && ngDevMode.rendererMoveNode++;\n  const nextNode = tNode.next;\n  if (!previousTNode) {\n    previousTNode = parentTNode;\n  }\n\n  // Re-organize node tree to put this node in the correct position.\n  if (previousTNode === parentTNode && tNode !== parentTNode.child) {\n    tNode.next = parentTNode.child;\n    parentTNode.child = tNode;\n  } else if (previousTNode !== parentTNode && tNode !== previousTNode.next) {\n    tNode.next = previousTNode.next;\n    previousTNode.next = tNode;\n  } else {\n    tNode.next = null;\n  }\n\n  if (parentTNode !== lView[T_HOST]) {\n    tNode.parent = parentTNode as TElementNode;\n  }\n\n  // If tNode was moved around, we might need to fix a broken link.\n  let cursor: TNode|null = tNode.next;\n  while (cursor) {\n    if (cursor.next === tNode) {\n      cursor.next = nextNode;\n    }\n    cursor = cursor.next;\n  }\n\n  // If the placeholder to append is a projection, we need to move the projected nodes instead\n  if (tNode.type === TNodeType.Projection) {\n    applyProjection(tView, lView, tNode as TProjectionNode);\n    return tNode;\n  }\n\n  appendChild(tView, lView, getNativeByTNode(tNode, lView), tNode);\n\n  const slotValue = lView[tNode.index];\n  if (tNode.type !== TNodeType.Container && isLContainer(slotValue)) {\n    // Nodes that inject ViewContainerRef also have a comment node that should be moved\n    appendChild(tView, lView, slotValue[NATIVE], tNode);\n  }\n  return tNode;\n}\n\nfunction isRootTemplateMessage(subTemplateIndex: number|undefined): subTemplateIndex is undefined {\n  return subTemplateIndex === undefined;\n}\n\n/**\n * Handles message string post-processing for internationalization.\n *\n * Handles message string post-processing by transforming it from intermediate\n * format (that might contain some markers that we need to replace) to the final\n * form, consumable by i18nStart instruction. Post processing steps include:\n *\n * 1. Resolve all multi-value cases (like [�*1:1��#2:1�|�#4:1�|�5�])\n * 2. Replace all ICU vars (like \"VAR_PLURAL\")\n * 3. Replace all placeholders used inside ICUs in a form of {PLACEHOLDER}\n * 4. Replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�)\n *    in case multiple ICUs have the same placeholder name\n *\n * @param message Raw translation string for post processing\n * @param replacements Set of replacements that should be applied\n *\n * @returns Transformed string that can be consumed by i18nStart instruction\n *\n * @codeGenApi\n */\nexport function ɵɵi18nPostprocess(\n    message: string, replacements: {[key: string]: (string|string[])} = {}): string {\n  /**\n   * Step 1: resolve all multi-value placeholders like [�#5�|�*1:1��#2:1�|�#4:1�]\n   *\n   * Note: due to the way we process nested templates (BFS), multi-value placeholders are typically\n   * grouped by templates, for example: [�#5�|�#6�|�#1:1�|�#3:2�] where �#5� and �#6� belong to root\n   * template, �#1:1� belong to nested template with index 1 and �#1:2� - nested template with index\n   * 3. However in real templates the order might be different: i.e. �#1:1� and/or �#3:2� may go in\n   * front of �#6�. The post processing step restores the right order by keeping track of the\n   * template id stack and looks for placeholders that belong to the currently active template.\n   */\n  let result: string = message;\n  if (PP_MULTI_VALUE_PLACEHOLDERS_REGEXP.test(message)) {\n    const matches: {[key: string]: PostprocessPlaceholder[]} = {};\n    const templateIdsStack: number[] = [ROOT_TEMPLATE_ID];\n    result = result.replace(PP_PLACEHOLDERS_REGEXP, (m: any, phs: string, tmpl: string): string => {\n      const content = phs || tmpl;\n      const placeholders: PostprocessPlaceholder[] = matches[content] || [];\n      if (!placeholders.length) {\n        content.split('|').forEach((placeholder: string) => {\n          const match = placeholder.match(PP_TEMPLATE_ID_REGEXP);\n          const templateId = match ? parseInt(match[1], 10) : ROOT_TEMPLATE_ID;\n          const isCloseTemplateTag = PP_CLOSE_TEMPLATE_REGEXP.test(placeholder);\n          placeholders.push([templateId, isCloseTemplateTag, placeholder]);\n        });\n        matches[content] = placeholders;\n      }\n\n      if (!placeholders.length) {\n        throw new Error(`i18n postprocess: unmatched placeholder - ${content}`);\n      }\n\n      const currentTemplateId = templateIdsStack[templateIdsStack.length - 1];\n      let idx = 0;\n      // find placeholder index that matches current template id\n      for (let i = 0; i < placeholders.length; i++) {\n        if (placeholders[i][0] === currentTemplateId) {\n          idx = i;\n          break;\n        }\n      }\n      // update template id stack based on the current tag extracted\n      const [templateId, isCloseTemplateTag, placeholder] = placeholders[idx];\n      if (isCloseTemplateTag) {\n        templateIdsStack.pop();\n      } else if (currentTemplateId !== templateId) {\n        templateIdsStack.push(templateId);\n      }\n      // remove processed tag from the list\n      placeholders.splice(idx, 1);\n      return placeholder;\n    });\n  }\n\n  // return current result if no replacements specified\n  if (!Object.keys(replacements).length) {\n    return result;\n  }\n\n  /**\n   * Step 2: replace all ICU vars (like \"VAR_PLURAL\")\n   */\n  result = result.replace(PP_ICU_VARS_REGEXP, (match, start, key, _type, _idx, end): string => {\n    return replacements.hasOwnProperty(key) ? `${start}${replacements[key]}${end}` : match;\n  });\n\n  /**\n   * Step 3: replace all placeholders used inside ICUs in a form of {PLACEHOLDER}\n   */\n  result = result.replace(PP_ICU_PLACEHOLDERS_REGEXP, (match, key): string => {\n    return replacements.hasOwnProperty(key) ? replacements[key] as string : match;\n  });\n\n  /**\n   * Step 4: replace all ICU references with corresponding values (like �ICU_EXP_ICU_1�) in case\n   * multiple ICUs have the same placeholder name\n   */\n  result = result.replace(PP_ICUS_REGEXP, (match, key): string => {\n    if (replacements.hasOwnProperty(key)) {\n      const list = replacements[key] as string[];\n      if (!list.length) {\n        throw new Error(`i18n postprocess: unmatched ICU - ${match} with key: ${key}`);\n      }\n      return list.shift()!;\n    }\n    return match;\n  });\n\n  return result;\n}\n\n/**\n * Translates a translation block marked by `i18nStart` and `i18nEnd`. It inserts the text/ICU nodes\n * into the render tree, moves the placeholder nodes and removes the deleted nodes.\n *\n * @codeGenApi\n */\nexport function ɵɵi18nEnd(): void {\n  const lView = getLView();\n  const tView = getTView();\n  ngDevMode && assertDefined(tView, `tView should be defined`);\n  i18nEndFirstPass(tView, lView);\n  // Stop delaying projections\n  setDelayProjection(false);\n}\n\n/**\n * See `i18nEnd` above.\n */\nfunction i18nEndFirstPass(tView: TView, lView: LView) {\n  ngDevMode &&\n      assertEqual(\n          getBindingIndex(), tView.bindingStartIndex,\n          'i18nEnd should be called before any binding');\n\n  const rootIndex = i18nIndexStack[i18nIndexStackPointer--];\n  const tI18n = tView.data[rootIndex + HEADER_OFFSET] as TI18n;\n  ngDevMode && assertDefined(tI18n, `You should call i18nStart before i18nEnd`);\n\n  // Find the last node that was added before `i18nEnd`\n  const lastCreatedNode = getPreviousOrParentTNode();\n\n  // Read the instructions to insert/move/remove DOM elements\n  const visitedNodes = readCreateOpCodes(rootIndex, tI18n.create, tView, lView);\n\n  // Remove deleted nodes\n  let index = rootIndex + 1;\n  while (index <= lastCreatedNode.index - HEADER_OFFSET) {\n    if (visitedNodes.indexOf(index) === -1) {\n      removeNode(tView, lView, index, /* markAsDetached */ true);\n    }\n    // Check if an element has any local refs and skip them\n    const tNode = getTNode(tView, index);\n    if (tNode &&\n        (tNode.type === TNodeType.Container || tNode.type === TNodeType.Element ||\n         tNode.type === TNodeType.ElementContainer) &&\n        tNode.localNames !== null) {\n      // Divide by 2 to get the number of local refs,\n      // since they are stored as an array that also includes directive indexes,\n      // i.e. [\"localRef\", directiveIndex, ...]\n      index += tNode.localNames.length >> 1;\n    }\n    index++;\n  }\n}\n\n/**\n * Creates and stores the dynamic TNode, and unhooks it from the tree for now.\n */\nfunction createDynamicNodeAtIndex(\n    tView: TView, lView: LView, index: number, type: TNodeType, native: RElement|RText|null,\n    name: string|null): TElementNode|TIcuContainerNode {\n  const previousOrParentTNode = getPreviousOrParentTNode();\n  ngDevMode && assertDataInRange(lView, index + HEADER_OFFSET);\n  lView[index + HEADER_OFFSET] = native;\n  const tNode = getOrCreateTNode(tView, lView[T_HOST], index, type as any, name, null);\n\n  // We are creating a dynamic node, the previous tNode might not be pointing at this node.\n  // We will link ourselves into the tree later with `appendI18nNode`.\n  if (previousOrParentTNode && previousOrParentTNode.next === tNode) {\n    previousOrParentTNode.next = null;\n  }\n\n  return tNode;\n}\n\nfunction readCreateOpCodes(\n    index: number, createOpCodes: I18nMutateOpCodes, tView: TView, lView: LView): number[] {\n  const renderer = lView[RENDERER];\n  let currentTNode: TNode|null = null;\n  let previousTNode: TNode|null = null;\n  const visitedNodes: number[] = [];\n  for (let i = 0; i < createOpCodes.length; i++) {\n    const opCode = createOpCodes[i];\n    if (typeof opCode == 'string') {\n      const textRNode = createTextNode(opCode, renderer);\n      const textNodeIndex = createOpCodes[++i] as number;\n      ngDevMode && ngDevMode.rendererCreateTextNode++;\n      previousTNode = currentTNode;\n      currentTNode =\n          createDynamicNodeAtIndex(tView, lView, textNodeIndex, TNodeType.Element, textRNode, null);\n      visitedNodes.push(textNodeIndex);\n      setIsNotParent();\n    } else if (typeof opCode == 'number') {\n      switch (opCode & I18nMutateOpCode.MASK_OPCODE) {\n        case I18nMutateOpCode.AppendChild:\n          const destinationNodeIndex = opCode >>> I18nMutateOpCode.SHIFT_PARENT;\n          let destinationTNode: TNode;\n          if (destinationNodeIndex === index) {\n            // If the destination node is `i18nStart`, we don't have a\n            // top-level node and we should use the host node instead\n            destinationTNode = lView[T_HOST]!;\n          } else {\n            destinationTNode = getTNode(tView, destinationNodeIndex);\n          }\n          ngDevMode &&\n              assertDefined(\n                  currentTNode!,\n                  `You need to create or select a node before you can insert it into the DOM`);\n          previousTNode =\n              appendI18nNode(tView, currentTNode!, destinationTNode, previousTNode, lView);\n          break;\n        case I18nMutateOpCode.Select:\n          // Negative indicies indicate that a given TNode is a sibling node, not a parent node\n          // (see `i18nStartFirstPass` for additional information).\n          const isParent = opCode >= 0;\n          const nodeIndex = (isParent ? opCode : ~opCode) >>> I18nMutateOpCode.SHIFT_REF;\n          visitedNodes.push(nodeIndex);\n          previousTNode = currentTNode;\n          currentTNode = getTNode(tView, nodeIndex);\n          if (currentTNode) {\n            setPreviousOrParentTNode(currentTNode, isParent);\n          }\n          break;\n        case I18nMutateOpCode.ElementEnd:\n          const elementIndex = opCode >>> I18nMutateOpCode.SHIFT_REF;\n          previousTNode = currentTNode = getTNode(tView, elementIndex);\n          setPreviousOrParentTNode(currentTNode, false);\n          break;\n        case I18nMutateOpCode.Attr:\n          const elementNodeIndex = opCode >>> I18nMutateOpCode.SHIFT_REF;\n          const attrName = createOpCodes[++i] as string;\n          const attrValue = createOpCodes[++i] as string;\n          // This code is used for ICU expressions only, since we don't support\n          // directives/components in ICUs, we don't need to worry about inputs here\n          elementAttributeInternal(\n              getTNode(tView, elementNodeIndex), lView, attrName, attrValue, null, null);\n          break;\n        default:\n          throw new Error(`Unable to determine the type of mutate operation for \"${opCode}\"`);\n      }\n    } else {\n      switch (opCode) {\n        case COMMENT_MARKER:\n          const commentValue = createOpCodes[++i] as string;\n          const commentNodeIndex = createOpCodes[++i] as number;\n          ngDevMode &&\n              assertEqual(\n                  typeof commentValue, 'string',\n                  `Expected \"${commentValue}\" to be a comment node value`);\n          const commentRNode = renderer.createComment(commentValue);\n          ngDevMode && ngDevMode.rendererCreateComment++;\n          previousTNode = currentTNode;\n          currentTNode = createDynamicNodeAtIndex(\n              tView, lView, commentNodeIndex, TNodeType.IcuContainer, commentRNode, null);\n          visitedNodes.push(commentNodeIndex);\n          attachPatchData(commentRNode, lView);\n          (currentTNode as TIcuContainerNode).activeCaseIndex = null;\n          // We will add the case nodes later, during the update phase\n          setIsNotParent();\n          break;\n        case ELEMENT_MARKER:\n          const tagNameValue = createOpCodes[++i] as string;\n          const elementNodeIndex = createOpCodes[++i] as number;\n          ngDevMode &&\n              assertEqual(\n                  typeof tagNameValue, 'string',\n                  `Expected \"${tagNameValue}\" to be an element node tag name`);\n          const elementRNode = renderer.createElement(tagNameValue);\n          ngDevMode && ngDevMode.rendererCreateElement++;\n          previousTNode = currentTNode;\n          currentTNode = createDynamicNodeAtIndex(\n              tView, lView, elementNodeIndex, TNodeType.Element, elementRNode, tagNameValue);\n          visitedNodes.push(elementNodeIndex);\n          break;\n        default:\n          throw new Error(`Unable to determine the type of mutate operation for \"${opCode}\"`);\n      }\n    }\n  }\n\n  setIsNotParent();\n\n  return visitedNodes;\n}\n\nfunction readUpdateOpCodes(\n    updateOpCodes: I18nUpdateOpCodes, icus: TIcu[]|null, bindingsStartIndex: number,\n    changeMask: number, tView: TView, lView: LView, bypassCheckBit = false) {\n  let caseCreated = false;\n  for (let i = 0; i < updateOpCodes.length; i++) {\n    // bit code to check if we should apply the next update\n    const checkBit = updateOpCodes[i] as number;\n    // Number of opCodes to skip until next set of update codes\n    const skipCodes = updateOpCodes[++i] as number;\n    if (bypassCheckBit || (checkBit & changeMask)) {\n      // The value has been updated since last checked\n      let value = '';\n      for (let j = i + 1; j <= (i + skipCodes); j++) {\n        const opCode = updateOpCodes[j];\n        if (typeof opCode == 'string') {\n          value += opCode;\n        } else if (typeof opCode == 'number') {\n          if (opCode < 0) {\n            // It's a binding index whose value is negative\n            value += renderStringify(lView[bindingsStartIndex - opCode]);\n          } else {\n            const nodeIndex = opCode >>> I18nUpdateOpCode.SHIFT_REF;\n            let tIcuIndex: number;\n            let tIcu: TIcu;\n            let icuTNode: TIcuContainerNode;\n            switch (opCode & I18nUpdateOpCode.MASK_OPCODE) {\n              case I18nUpdateOpCode.Attr:\n                const propName = updateOpCodes[++j] as string;\n                const sanitizeFn = updateOpCodes[++j] as SanitizerFn | null;\n                elementPropertyInternal(\n                    tView, getTNode(tView, nodeIndex), lView, propName, value, lView[RENDERER],\n                    sanitizeFn, false);\n                break;\n              case I18nUpdateOpCode.Text:\n                textBindingInternal(lView, nodeIndex, value);\n                break;\n              case I18nUpdateOpCode.IcuSwitch:\n                tIcuIndex = updateOpCodes[++j] as number;\n                tIcu = icus![tIcuIndex];\n                icuTNode = getTNode(tView, nodeIndex) as TIcuContainerNode;\n                // If there is an active case, delete the old nodes\n                if (icuTNode.activeCaseIndex !== null) {\n                  const removeCodes = tIcu.remove[icuTNode.activeCaseIndex];\n                  for (let k = 0; k < removeCodes.length; k++) {\n                    const removeOpCode = removeCodes[k] as number;\n                    switch (removeOpCode & I18nMutateOpCode.MASK_OPCODE) {\n                      case I18nMutateOpCode.Remove:\n                        const nodeIndex = removeOpCode >>> I18nMutateOpCode.SHIFT_REF;\n                        // Remove DOM element, but do *not* mark TNode as detached, since we are\n                        // just switching ICU cases (while keeping the same TNode), so a DOM element\n                        // representing a new ICU case will be re-created.\n                        removeNode(tView, lView, nodeIndex, /* markAsDetached */ false);\n                        break;\n                      case I18nMutateOpCode.RemoveNestedIcu:\n                        const nestedIcuNodeIndex =\n                            removeCodes[k + 1] as number >>> I18nMutateOpCode.SHIFT_REF;\n                        const nestedIcuTNode =\n                            getTNode(tView, nestedIcuNodeIndex) as TIcuContainerNode;\n                        const activeIndex = nestedIcuTNode.activeCaseIndex;\n                        if (activeIndex !== null) {\n                          const nestedIcuTIndex = removeOpCode >>> I18nMutateOpCode.SHIFT_REF;\n                          const nestedTIcu = icus![nestedIcuTIndex];\n                          addAllToArray(nestedTIcu.remove[activeIndex], removeCodes);\n                        }\n                        break;\n                    }\n                  }\n                }\n\n                // Update the active caseIndex\n                const caseIndex = getCaseIndex(tIcu, value);\n                icuTNode.activeCaseIndex = caseIndex !== -1 ? caseIndex : null;\n                if (caseIndex > -1) {\n                  // Add the nodes for the new case\n                  readCreateOpCodes(-1, tIcu.create[caseIndex], tView, lView);\n                  caseCreated = true;\n                }\n                break;\n              case I18nUpdateOpCode.IcuUpdate:\n                tIcuIndex = updateOpCodes[++j] as number;\n                tIcu = icus![tIcuIndex];\n                icuTNode = getTNode(tView, nodeIndex) as TIcuContainerNode;\n                if (icuTNode.activeCaseIndex !== null) {\n                  readUpdateOpCodes(\n                      tIcu.update[icuTNode.activeCaseIndex], icus, bindingsStartIndex, changeMask,\n                      tView, lView, caseCreated);\n                }\n                break;\n            }\n          }\n        }\n      }\n    }\n    i += skipCodes;\n  }\n}\n\nfunction removeNode(tView: TView, lView: LView, index: number, markAsDetached: boolean) {\n  const removedPhTNode = getTNode(tView, index);\n  const removedPhRNode = getNativeByIndex(index, lView);\n  if (removedPhRNode) {\n    nativeRemoveNode(lView[RENDERER], removedPhRNode);\n  }\n\n  const slotValue = load(lView, index) as RElement | RComment | LContainer;\n  if (isLContainer(slotValue)) {\n    const lContainer = slotValue as LContainer;\n    if (removedPhTNode.type !== TNodeType.Container) {\n      nativeRemoveNode(lView[RENDERER], lContainer[NATIVE]);\n    }\n  }\n\n  if (markAsDetached) {\n    // Define this node as detached to avoid projecting it later\n    removedPhTNode.flags |= TNodeFlags.isDetached;\n  }\n  ngDevMode && ngDevMode.rendererRemoveNode++;\n}\n\n/**\n *\n * Use this instruction to create a translation block that doesn't contain any placeholder.\n * It calls both {@link i18nStart} and {@link i18nEnd} in one instruction.\n *\n * The translation `message` is the value which is locale specific. The translation string may\n * contain placeholders which associate inner elements and sub-templates within the translation.\n *\n * The translation `message` placeholders are:\n * - `�{index}(:{block})�`: *Binding Placeholder*: Marks a location where an expression will be\n *   interpolated into. The placeholder `index` points to the expression binding index. An optional\n *   `block` that matches the sub-template in which it was declared.\n * - `�#{index}(:{block})�`/`�/#{index}(:{block})�`: *Element Placeholder*:  Marks the beginning\n *   and end of DOM element that were embedded in the original translation block. The placeholder\n *   `index` points to the element index in the template instructions set. An optional `block` that\n *   matches the sub-template in which it was declared.\n * - `�*{index}:{block}�`/`�/*{index}:{block}�`: *Sub-template Placeholder*: Sub-templates must be\n *   split up and translated separately in each angular template function. The `index` points to the\n *   `template` instruction index. A `block` that matches the sub-template in which it was declared.\n *\n * @param index A unique index of the translation in the static block.\n * @param message The translation message.\n * @param subTemplateIndex Optional sub-template index in the `message`.\n *\n * @codeGenApi\n */\nexport function ɵɵi18n(index: number, message: string, subTemplateIndex?: number): void {\n  ɵɵi18nStart(index, message, subTemplateIndex);\n  ɵɵi18nEnd();\n}\n\n/**\n * Marks a list of attributes as translatable.\n *\n * @param index A unique index in the static block\n * @param values\n *\n * @codeGenApi\n */\nexport function ɵɵi18nAttributes(index: number, values: string[]): void {\n  const lView = getLView();\n  const tView = getTView();\n  ngDevMode && assertDefined(tView, `tView should be defined`);\n  i18nAttributesFirstPass(lView, tView, index, values);\n}\n\n/**\n * See `i18nAttributes` above.\n */\nfunction i18nAttributesFirstPass(lView: LView, tView: TView, index: number, values: string[]) {\n  const previousElement = getPreviousOrParentTNode();\n  const previousElementIndex = previousElement.index - HEADER_OFFSET;\n  const updateOpCodes: I18nUpdateOpCodes = [];\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\nlet changeMask = 0b0;\nlet shiftsCounter = 0;\n\n/**\n * Stores the values of the bindings during each update cycle in order to determine if we need to\n * update the translated nodes.\n *\n * @param value The binding's value\n * @returns This function returns itself so that it may be chained\n * (e.g. `i18nExp(ctx.name)(ctx.title)`)\n *\n * @codeGenApi\n */\nexport function ɵɵi18nExp<T>(value: T): typeof ɵɵi18nExp {\n  const lView = getLView();\n  if (bindingUpdated(lView, nextBindingIndex(), value)) {\n    changeMask = changeMask | (1 << shiftsCounter);\n  }\n  shiftsCounter++;\n  return ɵɵi18nExp;\n}\n\n/**\n * Updates a translation block or an i18n attribute when the bindings have changed.\n *\n * @param index Index of either {@link i18nStart} (translation block) or {@link i18nAttributes}\n * (i18n attribute) on which it should update the content.\n *\n * @codeGenApi\n */\nexport function ɵɵi18nApply(index: number) {\n  if (shiftsCounter) {\n    const tView = getTView();\n    ngDevMode && assertDefined(tView, `tView should be defined`);\n    const tI18n = tView.data[index + HEADER_OFFSET];\n    let updateOpCodes: I18nUpdateOpCodes;\n    let icus: TIcu[]|null = null;\n    if (Array.isArray(tI18n)) {\n      updateOpCodes = tI18n as I18nUpdateOpCodes;\n    } else {\n      updateOpCodes = (tI18n as TI18n).update;\n      icus = (tI18n as TI18n).icus;\n    }\n    const bindingsStartIndex = getBindingIndex() - shiftsCounter - 1;\n    const lView = getLView();\n    readUpdateOpCodes(updateOpCodes, icus, bindingsStartIndex, changeMask, tView, lView);\n\n    // Reset changeMask & maskBit to default for the next update cycle\n    changeMask = 0b0;\n    shiftsCounter = 0;\n  }\n}\n\n/**\n * Returns the index of the current case of an ICU expression depending on the main binding value\n *\n * @param icuExpression\n * @param bindingValue The value of the main binding used by this ICU expression\n */\nfunction getCaseIndex(icuExpression: TIcu, bindingValue: string): number {\n  let index = icuExpression.cases.indexOf(bindingValue);\n  if (index === -1) {\n    switch (icuExpression.type) {\n      case IcuType.plural: {\n        const resolvedCase = getPluralCase(bindingValue, getLocaleId());\n        index = icuExpression.cases.indexOf(resolvedCase);\n        if (index === -1 && resolvedCase !== 'other') {\n          index = icuExpression.cases.indexOf('other');\n        }\n        break;\n      }\n      case IcuType.select: {\n        index = icuExpression.cases.indexOf('other');\n        break;\n      }\n    }\n  }\n  return index;\n}\n\n/**\n * Generate the OpCodes for ICU expressions.\n *\n * @param tIcus\n * @param icuExpression\n * @param startIndex\n * @param expandoStartIndex\n */\nfunction icuStart(\n    tIcus: TIcu[], icuExpression: IcuExpression, startIndex: number,\n    expandoStartIndex: number): void {\n  const createCodes = [];\n  const removeCodes = [];\n  const updateCodes = [];\n  const vars = [];\n  const childIcus: number[][] = [];\n  for (let i = 0; i < icuExpression.values.length; i++) {\n    // Each value is an array of strings & other ICU expressions\n    const valueArr = icuExpression.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    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 * 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 = {vars: 0, childIcus: [], create: [], remove: [], update: []};\n  parseNodes(wrapper.firstChild, opCodes, parentIndex, nestedIcus, tIcus, expandoStartIndex);\n  return opCodes;\n}\n\nconst NESTED_ICU = /�(\\d+)�/;\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 */\nfunction 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          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          nestedIcuNodeIndex << I18nMutateOpCode.SHIFT_REF | I18nMutateOpCode.Remove);\n    }\n  }\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 * The locale id that the application is currently using (for translations and ICU expressions).\n * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine\n * but is now defined as a global value.\n */\nlet LOCALE_ID = DEFAULT_LOCALE_ID;\n\n/**\n * Sets the locale id that will be used for translations and ICU expressions.\n * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine\n * but is now defined as a global value.\n *\n * @param localeId\n */\nexport function setLocaleId(localeId: string) {\n  assertDefined(localeId, `Expected localeId to be defined`);\n  if (typeof localeId === 'string') {\n    LOCALE_ID = localeId.toLowerCase().replace(/_/g, '-');\n  }\n}\n\n/**\n * Gets the locale id that will be used for translations and ICU expressions.\n * This is the ivy version of `LOCALE_ID` that was defined as an injection token for the view engine\n * but is now defined as a global value.\n */\nexport function getLocaleId(): string {\n  return LOCALE_ID;\n}\n"]}
|