@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
|
@@ -0,0 +1,445 @@
|
|
|
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 { getPluralCase } from '../../i18n/localization';
|
|
9
|
+
import { assertDefined, assertEqual, assertIndexInRange } from '../../util/assert';
|
|
10
|
+
import { attachPatchData } from '../context_discovery';
|
|
11
|
+
import { elementAttributeInternal, elementPropertyInternal, getOrCreateTNode, textBindingInternal } from '../instructions/shared';
|
|
12
|
+
import { NATIVE } from '../interfaces/container';
|
|
13
|
+
import { COMMENT_MARKER, ELEMENT_MARKER } from '../interfaces/i18n';
|
|
14
|
+
import { isLContainer } from '../interfaces/type_checks';
|
|
15
|
+
import { HEADER_OFFSET, RENDERER, T_HOST } from '../interfaces/view';
|
|
16
|
+
import { appendChild, applyProjection, createTextNode, nativeRemoveNode } from '../node_manipulation';
|
|
17
|
+
import { getBindingIndex, getPreviousOrParentTNode, setIsNotParent, setPreviousOrParentTNode } from '../state';
|
|
18
|
+
import { renderStringify } from '../util/misc_utils';
|
|
19
|
+
import { getNativeByIndex, getNativeByTNode, getTNode, load } from '../util/view_utils';
|
|
20
|
+
import { getLocaleId } from './i18n_locale_id';
|
|
21
|
+
const i18nIndexStack = [];
|
|
22
|
+
let i18nIndexStackPointer = -1;
|
|
23
|
+
function popI18nIndex() {
|
|
24
|
+
return i18nIndexStack[i18nIndexStackPointer--];
|
|
25
|
+
}
|
|
26
|
+
export function pushI18nIndex(index) {
|
|
27
|
+
i18nIndexStack[++i18nIndexStackPointer] = index;
|
|
28
|
+
}
|
|
29
|
+
let changeMask = 0b0;
|
|
30
|
+
let shiftsCounter = 0;
|
|
31
|
+
export function setMaskBit(bit) {
|
|
32
|
+
if (bit) {
|
|
33
|
+
changeMask = changeMask | (1 << shiftsCounter);
|
|
34
|
+
}
|
|
35
|
+
shiftsCounter++;
|
|
36
|
+
}
|
|
37
|
+
export function applyI18n(tView, lView, index) {
|
|
38
|
+
if (shiftsCounter > 0) {
|
|
39
|
+
ngDevMode && assertDefined(tView, `tView should be defined`);
|
|
40
|
+
const tI18n = tView.data[index + HEADER_OFFSET];
|
|
41
|
+
let updateOpCodes;
|
|
42
|
+
let tIcus = null;
|
|
43
|
+
if (Array.isArray(tI18n)) {
|
|
44
|
+
updateOpCodes = tI18n;
|
|
45
|
+
}
|
|
46
|
+
else {
|
|
47
|
+
updateOpCodes = tI18n.update;
|
|
48
|
+
tIcus = tI18n.icus;
|
|
49
|
+
}
|
|
50
|
+
const bindingsStartIndex = getBindingIndex() - shiftsCounter - 1;
|
|
51
|
+
applyUpdateOpCodes(tView, tIcus, lView, updateOpCodes, bindingsStartIndex, changeMask);
|
|
52
|
+
// Reset changeMask & maskBit to default for the next update cycle
|
|
53
|
+
changeMask = 0b0;
|
|
54
|
+
shiftsCounter = 0;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Apply `I18nMutateOpCodes` OpCodes.
|
|
59
|
+
*
|
|
60
|
+
* @param tView Current `TView`
|
|
61
|
+
* @param rootIndex Pointer to the root (parent) tNode for the i18n.
|
|
62
|
+
* @param createOpCodes OpCodes to process
|
|
63
|
+
* @param lView Current `LView`
|
|
64
|
+
*/
|
|
65
|
+
export function applyCreateOpCodes(tView, rootindex, createOpCodes, lView) {
|
|
66
|
+
const renderer = lView[RENDERER];
|
|
67
|
+
let currentTNode = null;
|
|
68
|
+
let previousTNode = null;
|
|
69
|
+
const visitedNodes = [];
|
|
70
|
+
for (let i = 0; i < createOpCodes.length; i++) {
|
|
71
|
+
const opCode = createOpCodes[i];
|
|
72
|
+
if (typeof opCode == 'string') {
|
|
73
|
+
const textRNode = createTextNode(opCode, renderer);
|
|
74
|
+
const textNodeIndex = createOpCodes[++i];
|
|
75
|
+
ngDevMode && ngDevMode.rendererCreateTextNode++;
|
|
76
|
+
previousTNode = currentTNode;
|
|
77
|
+
currentTNode =
|
|
78
|
+
createDynamicNodeAtIndex(tView, lView, textNodeIndex, 3 /* Element */, textRNode, null);
|
|
79
|
+
visitedNodes.push(textNodeIndex);
|
|
80
|
+
setIsNotParent();
|
|
81
|
+
}
|
|
82
|
+
else if (typeof opCode == 'number') {
|
|
83
|
+
switch (opCode & 7 /* MASK_INSTRUCTION */) {
|
|
84
|
+
case 1 /* AppendChild */:
|
|
85
|
+
const destinationNodeIndex = opCode >>> 17 /* SHIFT_PARENT */;
|
|
86
|
+
let destinationTNode;
|
|
87
|
+
if (destinationNodeIndex === rootindex) {
|
|
88
|
+
// If the destination node is `i18nStart`, we don't have a
|
|
89
|
+
// top-level node and we should use the host node instead
|
|
90
|
+
destinationTNode = lView[T_HOST];
|
|
91
|
+
}
|
|
92
|
+
else {
|
|
93
|
+
destinationTNode = getTNode(tView, destinationNodeIndex);
|
|
94
|
+
}
|
|
95
|
+
ngDevMode &&
|
|
96
|
+
assertDefined(currentTNode, `You need to create or select a node before you can insert it into the DOM`);
|
|
97
|
+
previousTNode =
|
|
98
|
+
appendI18nNode(tView, currentTNode, destinationTNode, previousTNode, lView);
|
|
99
|
+
break;
|
|
100
|
+
case 0 /* Select */:
|
|
101
|
+
// Negative indices indicate that a given TNode is a sibling node, not a parent node
|
|
102
|
+
// (see `i18nStartFirstPass` for additional information).
|
|
103
|
+
const isParent = opCode >= 0;
|
|
104
|
+
// FIXME(misko): This SHIFT_REF looks suspect as it does not have mask.
|
|
105
|
+
const nodeIndex = (isParent ? opCode : ~opCode) >>> 3 /* SHIFT_REF */;
|
|
106
|
+
visitedNodes.push(nodeIndex);
|
|
107
|
+
previousTNode = currentTNode;
|
|
108
|
+
currentTNode = getTNode(tView, nodeIndex);
|
|
109
|
+
if (currentTNode) {
|
|
110
|
+
setPreviousOrParentTNode(currentTNode, isParent);
|
|
111
|
+
}
|
|
112
|
+
break;
|
|
113
|
+
case 5 /* ElementEnd */:
|
|
114
|
+
const elementIndex = opCode >>> 3 /* SHIFT_REF */;
|
|
115
|
+
previousTNode = currentTNode = getTNode(tView, elementIndex);
|
|
116
|
+
setPreviousOrParentTNode(currentTNode, false);
|
|
117
|
+
break;
|
|
118
|
+
case 4 /* Attr */:
|
|
119
|
+
const elementNodeIndex = opCode >>> 3 /* SHIFT_REF */;
|
|
120
|
+
const attrName = createOpCodes[++i];
|
|
121
|
+
const attrValue = createOpCodes[++i];
|
|
122
|
+
// This code is used for ICU expressions only, since we don't support
|
|
123
|
+
// directives/components in ICUs, we don't need to worry about inputs here
|
|
124
|
+
elementAttributeInternal(getTNode(tView, elementNodeIndex), lView, attrName, attrValue, null, null);
|
|
125
|
+
break;
|
|
126
|
+
default:
|
|
127
|
+
throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
else {
|
|
131
|
+
switch (opCode) {
|
|
132
|
+
case COMMENT_MARKER:
|
|
133
|
+
const commentValue = createOpCodes[++i];
|
|
134
|
+
const commentNodeIndex = createOpCodes[++i];
|
|
135
|
+
ngDevMode &&
|
|
136
|
+
assertEqual(typeof commentValue, 'string', `Expected "${commentValue}" to be a comment node value`);
|
|
137
|
+
const commentRNode = renderer.createComment(commentValue);
|
|
138
|
+
ngDevMode && ngDevMode.rendererCreateComment++;
|
|
139
|
+
previousTNode = currentTNode;
|
|
140
|
+
currentTNode = createDynamicNodeAtIndex(tView, lView, commentNodeIndex, 5 /* IcuContainer */, commentRNode, null);
|
|
141
|
+
visitedNodes.push(commentNodeIndex);
|
|
142
|
+
attachPatchData(commentRNode, lView);
|
|
143
|
+
// We will add the case nodes later, during the update phase
|
|
144
|
+
setIsNotParent();
|
|
145
|
+
break;
|
|
146
|
+
case ELEMENT_MARKER:
|
|
147
|
+
const tagNameValue = createOpCodes[++i];
|
|
148
|
+
const elementNodeIndex = createOpCodes[++i];
|
|
149
|
+
ngDevMode &&
|
|
150
|
+
assertEqual(typeof tagNameValue, 'string', `Expected "${tagNameValue}" to be an element node tag name`);
|
|
151
|
+
const elementRNode = renderer.createElement(tagNameValue);
|
|
152
|
+
ngDevMode && ngDevMode.rendererCreateElement++;
|
|
153
|
+
previousTNode = currentTNode;
|
|
154
|
+
currentTNode = createDynamicNodeAtIndex(tView, lView, elementNodeIndex, 3 /* Element */, elementRNode, tagNameValue);
|
|
155
|
+
visitedNodes.push(elementNodeIndex);
|
|
156
|
+
break;
|
|
157
|
+
default:
|
|
158
|
+
throw new Error(`Unable to determine the type of mutate operation for "${opCode}"`);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
setIsNotParent();
|
|
163
|
+
return visitedNodes;
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Apply `I18nUpdateOpCodes` OpCodes
|
|
167
|
+
*
|
|
168
|
+
* @param tView Current `TView`
|
|
169
|
+
* @param tIcus If ICUs present than this contains them.
|
|
170
|
+
* @param lView Current `LView`
|
|
171
|
+
* @param updateOpCodes OpCodes to process
|
|
172
|
+
* @param bindingsStartIndex Location of the first `ɵɵi18nApply`
|
|
173
|
+
* @param changeMask Each bit corresponds to a `ɵɵi18nExp` (Counting backwards from
|
|
174
|
+
* `bindingsStartIndex`)
|
|
175
|
+
*/
|
|
176
|
+
export function applyUpdateOpCodes(tView, tIcus, lView, updateOpCodes, bindingsStartIndex, changeMask) {
|
|
177
|
+
let caseCreated = false;
|
|
178
|
+
for (let i = 0; i < updateOpCodes.length; i++) {
|
|
179
|
+
// bit code to check if we should apply the next update
|
|
180
|
+
const checkBit = updateOpCodes[i];
|
|
181
|
+
// Number of opCodes to skip until next set of update codes
|
|
182
|
+
const skipCodes = updateOpCodes[++i];
|
|
183
|
+
if (checkBit & changeMask) {
|
|
184
|
+
// The value has been updated since last checked
|
|
185
|
+
let value = '';
|
|
186
|
+
for (let j = i + 1; j <= (i + skipCodes); j++) {
|
|
187
|
+
const opCode = updateOpCodes[j];
|
|
188
|
+
if (typeof opCode == 'string') {
|
|
189
|
+
value += opCode;
|
|
190
|
+
}
|
|
191
|
+
else if (typeof opCode == 'number') {
|
|
192
|
+
if (opCode < 0) {
|
|
193
|
+
// Negative opCode represent `i18nExp` values offset.
|
|
194
|
+
value += renderStringify(lView[bindingsStartIndex - opCode]);
|
|
195
|
+
}
|
|
196
|
+
else {
|
|
197
|
+
const nodeIndex = opCode >>> 2 /* SHIFT_REF */;
|
|
198
|
+
switch (opCode & 3 /* MASK_OPCODE */) {
|
|
199
|
+
case 1 /* Attr */:
|
|
200
|
+
const propName = updateOpCodes[++j];
|
|
201
|
+
const sanitizeFn = updateOpCodes[++j];
|
|
202
|
+
elementPropertyInternal(tView, getTNode(tView, nodeIndex), lView, propName, value, lView[RENDERER], sanitizeFn, false);
|
|
203
|
+
break;
|
|
204
|
+
case 0 /* Text */:
|
|
205
|
+
textBindingInternal(lView, nodeIndex, value);
|
|
206
|
+
break;
|
|
207
|
+
case 2 /* IcuSwitch */:
|
|
208
|
+
caseCreated =
|
|
209
|
+
applyIcuSwitchCase(tView, tIcus, updateOpCodes[++j], lView, value);
|
|
210
|
+
break;
|
|
211
|
+
case 3 /* IcuUpdate */:
|
|
212
|
+
applyIcuUpdateCase(tView, tIcus, updateOpCodes[++j], bindingsStartIndex, lView, caseCreated);
|
|
213
|
+
break;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
i += skipCodes;
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* Apply OpCodes associated with updating an existing ICU.
|
|
224
|
+
*
|
|
225
|
+
* @param tView Current `TView`
|
|
226
|
+
* @param tIcus ICUs active at this location.
|
|
227
|
+
* @param tIcuIndex Index into `tIcus` to process.
|
|
228
|
+
* @param bindingsStartIndex Location of the first `ɵɵi18nApply`
|
|
229
|
+
* @param lView Current `LView`
|
|
230
|
+
* @param changeMask Each bit corresponds to a `ɵɵi18nExp` (Counting backwards from
|
|
231
|
+
* `bindingsStartIndex`)
|
|
232
|
+
*/
|
|
233
|
+
function applyIcuUpdateCase(tView, tIcus, tIcuIndex, bindingsStartIndex, lView, caseCreated) {
|
|
234
|
+
ngDevMode && assertIndexInRange(tIcus, tIcuIndex);
|
|
235
|
+
const tIcu = tIcus[tIcuIndex];
|
|
236
|
+
ngDevMode && assertIndexInRange(lView, tIcu.currentCaseLViewIndex);
|
|
237
|
+
const activeCaseIndex = lView[tIcu.currentCaseLViewIndex];
|
|
238
|
+
if (activeCaseIndex !== null) {
|
|
239
|
+
const mask = caseCreated ?
|
|
240
|
+
-1 : // -1 is same as all bits on, which simulates creation since it marks all bits dirty
|
|
241
|
+
changeMask;
|
|
242
|
+
applyUpdateOpCodes(tView, tIcus, lView, tIcu.update[activeCaseIndex], bindingsStartIndex, mask);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* Apply OpCodes associated with switching a case on ICU.
|
|
247
|
+
*
|
|
248
|
+
* This involves tearing down existing case and than building up a new case.
|
|
249
|
+
*
|
|
250
|
+
* @param tView Current `TView`
|
|
251
|
+
* @param tIcus ICUs active at this location.
|
|
252
|
+
* @param tICuIndex Index into `tIcus` to process.
|
|
253
|
+
* @param lView Current `LView`
|
|
254
|
+
* @param value Value of the case to update to.
|
|
255
|
+
* @returns true if a new case was created (needed so that the update executes regardless of the
|
|
256
|
+
* bitmask)
|
|
257
|
+
*/
|
|
258
|
+
function applyIcuSwitchCase(tView, tIcus, tICuIndex, lView, value) {
|
|
259
|
+
applyIcuSwitchCaseRemove(tView, tIcus, tICuIndex, lView);
|
|
260
|
+
// Rebuild a new case for this ICU
|
|
261
|
+
let caseCreated = false;
|
|
262
|
+
const tIcu = tIcus[tICuIndex];
|
|
263
|
+
const caseIndex = getCaseIndex(tIcu, value);
|
|
264
|
+
lView[tIcu.currentCaseLViewIndex] = caseIndex !== -1 ? caseIndex : null;
|
|
265
|
+
if (caseIndex > -1) {
|
|
266
|
+
// Add the nodes for the new case
|
|
267
|
+
applyCreateOpCodes(tView, -1, // -1 means we don't have parent node
|
|
268
|
+
tIcu.create[caseIndex], lView);
|
|
269
|
+
caseCreated = true;
|
|
270
|
+
}
|
|
271
|
+
return caseCreated;
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* Apply OpCodes associated with tearing down of DOM.
|
|
275
|
+
*
|
|
276
|
+
* This involves tearing down existing case and than building up a new case.
|
|
277
|
+
*
|
|
278
|
+
* @param tView Current `TView`
|
|
279
|
+
* @param tIcus ICUs active at this location.
|
|
280
|
+
* @param tIcuIndex Index into `tIcus` to process.
|
|
281
|
+
* @param lView Current `LView`
|
|
282
|
+
* @returns true if a new case was created (needed so that the update executes regardless of the
|
|
283
|
+
* bitmask)
|
|
284
|
+
*/
|
|
285
|
+
function applyIcuSwitchCaseRemove(tView, tIcus, tIcuIndex, lView) {
|
|
286
|
+
ngDevMode && assertIndexInRange(tIcus, tIcuIndex);
|
|
287
|
+
const tIcu = tIcus[tIcuIndex];
|
|
288
|
+
const activeCaseIndex = lView[tIcu.currentCaseLViewIndex];
|
|
289
|
+
if (activeCaseIndex !== null) {
|
|
290
|
+
const removeCodes = tIcu.remove[activeCaseIndex];
|
|
291
|
+
for (let k = 0; k < removeCodes.length; k++) {
|
|
292
|
+
const removeOpCode = removeCodes[k];
|
|
293
|
+
const nodeOrIcuIndex = removeOpCode >>> 3 /* SHIFT_REF */;
|
|
294
|
+
switch (removeOpCode & 7 /* MASK_INSTRUCTION */) {
|
|
295
|
+
case 3 /* Remove */:
|
|
296
|
+
// FIXME(misko): this comment is wrong!
|
|
297
|
+
// Remove DOM element, but do *not* mark TNode as detached, since we are
|
|
298
|
+
// just switching ICU cases (while keeping the same TNode), so a DOM element
|
|
299
|
+
// representing a new ICU case will be re-created.
|
|
300
|
+
removeNode(tView, lView, nodeOrIcuIndex, /* markAsDetached */ false);
|
|
301
|
+
break;
|
|
302
|
+
case 6 /* RemoveNestedIcu */:
|
|
303
|
+
applyIcuSwitchCaseRemove(tView, tIcus, nodeOrIcuIndex, lView);
|
|
304
|
+
break;
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
function appendI18nNode(tView, tNode, parentTNode, previousTNode, lView) {
|
|
310
|
+
ngDevMode && ngDevMode.rendererMoveNode++;
|
|
311
|
+
const nextNode = tNode.next;
|
|
312
|
+
if (!previousTNode) {
|
|
313
|
+
previousTNode = parentTNode;
|
|
314
|
+
}
|
|
315
|
+
// Re-organize node tree to put this node in the correct position.
|
|
316
|
+
if (previousTNode === parentTNode && tNode !== parentTNode.child) {
|
|
317
|
+
tNode.next = parentTNode.child;
|
|
318
|
+
parentTNode.child = tNode;
|
|
319
|
+
}
|
|
320
|
+
else if (previousTNode !== parentTNode && tNode !== previousTNode.next) {
|
|
321
|
+
tNode.next = previousTNode.next;
|
|
322
|
+
previousTNode.next = tNode;
|
|
323
|
+
}
|
|
324
|
+
else {
|
|
325
|
+
tNode.next = null;
|
|
326
|
+
}
|
|
327
|
+
if (parentTNode !== lView[T_HOST]) {
|
|
328
|
+
tNode.parent = parentTNode;
|
|
329
|
+
}
|
|
330
|
+
// If tNode was moved around, we might need to fix a broken link.
|
|
331
|
+
let cursor = tNode.next;
|
|
332
|
+
while (cursor) {
|
|
333
|
+
if (cursor.next === tNode) {
|
|
334
|
+
cursor.next = nextNode;
|
|
335
|
+
}
|
|
336
|
+
cursor = cursor.next;
|
|
337
|
+
}
|
|
338
|
+
// If the placeholder to append is a projection, we need to move the projected nodes instead
|
|
339
|
+
if (tNode.type === 1 /* Projection */) {
|
|
340
|
+
applyProjection(tView, lView, tNode);
|
|
341
|
+
return tNode;
|
|
342
|
+
}
|
|
343
|
+
appendChild(tView, lView, getNativeByTNode(tNode, lView), tNode);
|
|
344
|
+
const slotValue = lView[tNode.index];
|
|
345
|
+
if (tNode.type !== 0 /* Container */ && isLContainer(slotValue)) {
|
|
346
|
+
// Nodes that inject ViewContainerRef also have a comment node that should be moved
|
|
347
|
+
appendChild(tView, lView, slotValue[NATIVE], tNode);
|
|
348
|
+
}
|
|
349
|
+
return tNode;
|
|
350
|
+
}
|
|
351
|
+
/**
|
|
352
|
+
* See `i18nEnd` above.
|
|
353
|
+
*/
|
|
354
|
+
export function i18nEndFirstPass(tView, lView) {
|
|
355
|
+
ngDevMode &&
|
|
356
|
+
assertEqual(getBindingIndex(), tView.bindingStartIndex, 'i18nEnd should be called before any binding');
|
|
357
|
+
const rootIndex = popI18nIndex();
|
|
358
|
+
const tI18n = tView.data[rootIndex + HEADER_OFFSET];
|
|
359
|
+
ngDevMode && assertDefined(tI18n, `You should call i18nStart before i18nEnd`);
|
|
360
|
+
// Find the last node that was added before `i18nEnd`
|
|
361
|
+
const lastCreatedNode = getPreviousOrParentTNode();
|
|
362
|
+
// Read the instructions to insert/move/remove DOM elements
|
|
363
|
+
const visitedNodes = applyCreateOpCodes(tView, rootIndex, tI18n.create, lView);
|
|
364
|
+
// Remove deleted nodes
|
|
365
|
+
let index = rootIndex + 1;
|
|
366
|
+
while (index <= lastCreatedNode.index - HEADER_OFFSET) {
|
|
367
|
+
if (visitedNodes.indexOf(index) === -1) {
|
|
368
|
+
removeNode(tView, lView, index, /* markAsDetached */ true);
|
|
369
|
+
}
|
|
370
|
+
// Check if an element has any local refs and skip them
|
|
371
|
+
const tNode = getTNode(tView, index);
|
|
372
|
+
if (tNode &&
|
|
373
|
+
(tNode.type === 0 /* Container */ || tNode.type === 3 /* Element */ ||
|
|
374
|
+
tNode.type === 4 /* ElementContainer */) &&
|
|
375
|
+
tNode.localNames !== null) {
|
|
376
|
+
// Divide by 2 to get the number of local refs,
|
|
377
|
+
// since they are stored as an array that also includes directive indexes,
|
|
378
|
+
// i.e. ["localRef", directiveIndex, ...]
|
|
379
|
+
index += tNode.localNames.length >> 1;
|
|
380
|
+
}
|
|
381
|
+
index++;
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
function removeNode(tView, lView, index, markAsDetached) {
|
|
385
|
+
const removedPhTNode = getTNode(tView, index);
|
|
386
|
+
const removedPhRNode = getNativeByIndex(index, lView);
|
|
387
|
+
if (removedPhRNode) {
|
|
388
|
+
nativeRemoveNode(lView[RENDERER], removedPhRNode);
|
|
389
|
+
}
|
|
390
|
+
const slotValue = load(lView, index);
|
|
391
|
+
if (isLContainer(slotValue)) {
|
|
392
|
+
const lContainer = slotValue;
|
|
393
|
+
if (removedPhTNode.type !== 0 /* Container */) {
|
|
394
|
+
nativeRemoveNode(lView[RENDERER], lContainer[NATIVE]);
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
if (markAsDetached) {
|
|
398
|
+
// Define this node as detached to avoid projecting it later
|
|
399
|
+
removedPhTNode.flags |= 64 /* isDetached */;
|
|
400
|
+
}
|
|
401
|
+
ngDevMode && ngDevMode.rendererRemoveNode++;
|
|
402
|
+
}
|
|
403
|
+
/**
|
|
404
|
+
* Creates and stores the dynamic TNode, and unhooks it from the tree for now.
|
|
405
|
+
*/
|
|
406
|
+
function createDynamicNodeAtIndex(tView, lView, index, type, native, name) {
|
|
407
|
+
const previousOrParentTNode = getPreviousOrParentTNode();
|
|
408
|
+
ngDevMode && assertIndexInRange(lView, index + HEADER_OFFSET);
|
|
409
|
+
lView[index + HEADER_OFFSET] = native;
|
|
410
|
+
// FIXME(misko): Why does this create A TNode??? I would not expect this to be here.
|
|
411
|
+
const tNode = getOrCreateTNode(tView, lView[T_HOST], index, type, name, null);
|
|
412
|
+
// We are creating a dynamic node, the previous tNode might not be pointing at this node.
|
|
413
|
+
// We will link ourselves into the tree later with `appendI18nNode`.
|
|
414
|
+
if (previousOrParentTNode && previousOrParentTNode.next === tNode) {
|
|
415
|
+
previousOrParentTNode.next = null;
|
|
416
|
+
}
|
|
417
|
+
return tNode;
|
|
418
|
+
}
|
|
419
|
+
/**
|
|
420
|
+
* Returns the index of the current case of an ICU expression depending on the main binding value
|
|
421
|
+
*
|
|
422
|
+
* @param icuExpression
|
|
423
|
+
* @param bindingValue The value of the main binding used by this ICU expression
|
|
424
|
+
*/
|
|
425
|
+
function getCaseIndex(icuExpression, bindingValue) {
|
|
426
|
+
let index = icuExpression.cases.indexOf(bindingValue);
|
|
427
|
+
if (index === -1) {
|
|
428
|
+
switch (icuExpression.type) {
|
|
429
|
+
case 1 /* plural */: {
|
|
430
|
+
const resolvedCase = getPluralCase(bindingValue, getLocaleId());
|
|
431
|
+
index = icuExpression.cases.indexOf(resolvedCase);
|
|
432
|
+
if (index === -1 && resolvedCase !== 'other') {
|
|
433
|
+
index = icuExpression.cases.indexOf('other');
|
|
434
|
+
}
|
|
435
|
+
break;
|
|
436
|
+
}
|
|
437
|
+
case 0 /* select */: {
|
|
438
|
+
index = icuExpression.cases.indexOf('other');
|
|
439
|
+
break;
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
return index;
|
|
444
|
+
}
|
|
445
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n_apply.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/i18n/i18n_apply.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {getPluralCase} from '../../i18n/localization';\nimport {assertDefined, assertEqual, assertIndexInRange} from '../../util/assert';\nimport {attachPatchData} from '../context_discovery';\nimport {elementAttributeInternal, elementPropertyInternal, getOrCreateTNode, textBindingInternal} from '../instructions/shared';\nimport {LContainer, NATIVE} from '../interfaces/container';\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} from '../interfaces/view';\nimport {appendChild, applyProjection, createTextNode, nativeRemoveNode} from '../node_manipulation';\nimport {getBindingIndex, getLView, getPreviousOrParentTNode, getTView, setIsNotParent, setPreviousOrParentTNode} from '../state';\nimport {renderStringify} from '../util/misc_utils';\nimport {getNativeByIndex, getNativeByTNode, getTNode, load} from '../util/view_utils';\nimport {getLocaleId} from './i18n_locale_id';\n\n\nconst i18nIndexStack: number[] = [];\nlet i18nIndexStackPointer = -1;\n\nfunction popI18nIndex() {\n  return i18nIndexStack[i18nIndexStackPointer--];\n}\n\nexport function pushI18nIndex(index: number) {\n  i18nIndexStack[++i18nIndexStackPointer] = index;\n}\n\nlet changeMask = 0b0;\nlet shiftsCounter = 0;\n\nexport function setMaskBit(bit: boolean) {\n  if (bit) {\n    changeMask = changeMask | (1 << shiftsCounter);\n  }\n  shiftsCounter++;\n}\n\nexport function applyI18n(tView: TView, lView: LView, index: number) {\n  if (shiftsCounter > 0) {\n    ngDevMode && assertDefined(tView, `tView should be defined`);\n    const tI18n = tView.data[index + HEADER_OFFSET] as TI18n | I18nUpdateOpCodes;\n    let updateOpCodes: I18nUpdateOpCodes;\n    let tIcus: TIcu[]|null = null;\n    if (Array.isArray(tI18n)) {\n      updateOpCodes = tI18n as I18nUpdateOpCodes;\n    } else {\n      updateOpCodes = (tI18n as TI18n).update;\n      tIcus = (tI18n as TI18n).icus;\n    }\n    const bindingsStartIndex = getBindingIndex() - shiftsCounter - 1;\n    applyUpdateOpCodes(tView, tIcus, lView, updateOpCodes, bindingsStartIndex, changeMask);\n\n    // Reset changeMask & maskBit to default for the next update cycle\n    changeMask = 0b0;\n    shiftsCounter = 0;\n  }\n}\n\n/**\n * Apply `I18nMutateOpCodes` OpCodes.\n *\n * @param tView Current `TView`\n * @param rootIndex Pointer to the root (parent) tNode for the i18n.\n * @param createOpCodes OpCodes to process\n * @param lView Current `LView`\n */\nexport function applyCreateOpCodes(\n    tView: TView, rootindex: number, createOpCodes: I18nMutateOpCodes, 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_INSTRUCTION) {\n        case I18nMutateOpCode.AppendChild:\n          const destinationNodeIndex = opCode >>> I18nMutateOpCode.SHIFT_PARENT;\n          let destinationTNode: TNode;\n          if (destinationNodeIndex === rootindex) {\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 indices 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          // FIXME(misko): This SHIFT_REF looks suspect as it does not have mask.\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          // 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\n\n/**\n * Apply `I18nUpdateOpCodes` OpCodes\n *\n * @param tView Current `TView`\n * @param tIcus If ICUs present than this contains them.\n * @param lView Current `LView`\n * @param updateOpCodes OpCodes to process\n * @param bindingsStartIndex Location of the first `ɵɵi18nApply`\n * @param changeMask Each bit corresponds to a `ɵɵi18nExp` (Counting backwards from\n *     `bindingsStartIndex`)\n */\nexport function applyUpdateOpCodes(\n    tView: TView, tIcus: TIcu[]|null, lView: LView, updateOpCodes: I18nUpdateOpCodes,\n    bindingsStartIndex: number, changeMask: number) {\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 (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            // Negative opCode represent `i18nExp` values offset.\n            value += renderStringify(lView[bindingsStartIndex - opCode]);\n          } else {\n            const nodeIndex = opCode >>> I18nUpdateOpCode.SHIFT_REF;\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                caseCreated =\n                    applyIcuSwitchCase(tView, tIcus!, updateOpCodes[++j] as number, lView, value);\n                break;\n              case I18nUpdateOpCode.IcuUpdate:\n                applyIcuUpdateCase(\n                    tView, tIcus!, updateOpCodes[++j] as number, bindingsStartIndex, lView,\n                    caseCreated);\n                break;\n            }\n          }\n        }\n      }\n    }\n    i += skipCodes;\n  }\n}\n\n/**\n * Apply OpCodes associated with updating an existing ICU.\n *\n * @param tView Current `TView`\n * @param tIcus ICUs active at this location.\n * @param tIcuIndex Index into `tIcus` to process.\n * @param bindingsStartIndex Location of the first `ɵɵi18nApply`\n * @param lView Current `LView`\n * @param changeMask Each bit corresponds to a `ɵɵi18nExp` (Counting backwards from\n *     `bindingsStartIndex`)\n */\nfunction applyIcuUpdateCase(\n    tView: TView, tIcus: TIcu[], tIcuIndex: number, bindingsStartIndex: number, lView: LView,\n    caseCreated: boolean) {\n  ngDevMode && assertIndexInRange(tIcus, tIcuIndex);\n  const tIcu = tIcus[tIcuIndex];\n  ngDevMode && assertIndexInRange(lView, tIcu.currentCaseLViewIndex);\n  const activeCaseIndex = lView[tIcu.currentCaseLViewIndex];\n  if (activeCaseIndex !== null) {\n    const mask = caseCreated ?\n        -1 :  // -1 is same as all bits on, which simulates creation since it marks all bits dirty\n        changeMask;\n    applyUpdateOpCodes(tView, tIcus, lView, tIcu.update[activeCaseIndex], bindingsStartIndex, mask);\n  }\n}\n\n/**\n * Apply OpCodes associated with switching a case on ICU.\n *\n * This involves tearing down existing case and than building up a new case.\n *\n * @param tView Current `TView`\n * @param tIcus ICUs active at this location.\n * @param tICuIndex Index into `tIcus` to process.\n * @param lView Current `LView`\n * @param value Value of the case to update to.\n * @returns true if a new case was created (needed so that the update executes regardless of the\n *     bitmask)\n */\nfunction applyIcuSwitchCase(\n    tView: TView, tIcus: TIcu[], tICuIndex: number, lView: LView, value: string): boolean {\n  applyIcuSwitchCaseRemove(tView, tIcus, tICuIndex, lView);\n\n  // Rebuild a new case for this ICU\n  let caseCreated = false;\n  const tIcu = tIcus[tICuIndex];\n  const caseIndex = getCaseIndex(tIcu, value);\n  lView[tIcu.currentCaseLViewIndex] = caseIndex !== -1 ? caseIndex : null;\n  if (caseIndex > -1) {\n    // Add the nodes for the new case\n    applyCreateOpCodes(\n        tView, -1,  // -1 means we don't have parent node\n        tIcu.create[caseIndex], lView);\n    caseCreated = true;\n  }\n  return caseCreated;\n}\n\n/**\n * Apply OpCodes associated with tearing down of DOM.\n *\n * This involves tearing down existing case and than building up a new case.\n *\n * @param tView Current `TView`\n * @param tIcus ICUs active at this location.\n * @param tIcuIndex Index into `tIcus` to process.\n * @param lView Current `LView`\n * @returns true if a new case was created (needed so that the update executes regardless of the\n *     bitmask)\n */\nfunction applyIcuSwitchCaseRemove(tView: TView, tIcus: TIcu[], tIcuIndex: number, lView: LView) {\n  ngDevMode && assertIndexInRange(tIcus, tIcuIndex);\n  const tIcu = tIcus[tIcuIndex];\n  const activeCaseIndex = lView[tIcu.currentCaseLViewIndex];\n  if (activeCaseIndex !== null) {\n    const removeCodes = tIcu.remove[activeCaseIndex];\n    for (let k = 0; k < removeCodes.length; k++) {\n      const removeOpCode = removeCodes[k] as number;\n      const nodeOrIcuIndex = removeOpCode >>> I18nMutateOpCode.SHIFT_REF;\n      switch (removeOpCode & I18nMutateOpCode.MASK_INSTRUCTION) {\n        case I18nMutateOpCode.Remove:\n          // FIXME(misko): this comment is wrong!\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, nodeOrIcuIndex, /* markAsDetached */ false);\n          break;\n        case I18nMutateOpCode.RemoveNestedIcu:\n          applyIcuSwitchCaseRemove(tView, tIcus, nodeOrIcuIndex, lView);\n          break;\n      }\n    }\n  }\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\n/**\n * See `i18nEnd` above.\n */\nexport function 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 = popI18nIndex();\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 = applyCreateOpCodes(tView, rootIndex, tI18n.create, 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\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 * 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 && assertIndexInRange(lView, index + HEADER_OFFSET);\n  lView[index + HEADER_OFFSET] = native;\n  // FIXME(misko): Why does this create A TNode??? I would not expect this to be here.\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\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"]}
|