@angular/core 9.0.0-rc.9 → 9.0.0
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 +10 -9
- package/bundles/core-testing.umd.js.map +1 -1
- package/bundles/core-testing.umd.min.js +6 -6
- package/bundles/core-testing.umd.min.js.map +1 -1
- package/bundles/core.umd.js +3619 -3914
- package/bundles/core.umd.js.map +1 -1
- package/bundles/core.umd.min.js +182 -175
- package/bundles/core.umd.min.js.map +1 -1
- package/core.d.ts +483 -651
- package/core.metadata.json +1 -1
- package/esm2015/core.js +17 -17
- package/esm2015/index.js +2 -2
- package/esm2015/public_api.js +2 -2
- package/esm2015/src/application_ref.js +7 -7
- package/esm2015/src/core.js +3 -3
- package/esm2015/src/core_private_export.js +7 -7
- package/esm2015/src/core_render3_private_export.js +2 -2
- package/esm2015/src/debug/debug_node.js +44 -12
- package/esm2015/src/i18n/locale_data_api.js +1 -2
- package/esm2015/src/interface/type.js +1 -1
- package/esm2015/src/metadata/ng_module.js +1 -1
- package/esm2015/src/render/api.js +4 -1
- package/esm2015/src/render3/assert.js +9 -1
- package/esm2015/src/render3/bindings.js +16 -5
- package/esm2015/src/render3/component.js +54 -25
- package/esm2015/src/render3/component_ref.js +11 -15
- package/esm2015/src/render3/definition.js +3 -1
- package/esm2015/src/render3/di.js +3 -4
- package/esm2015/src/render3/di_setup.js +5 -7
- package/esm2015/src/render3/errors.js +3 -1
- package/esm2015/src/render3/features/inherit_definition_feature.js +41 -13
- package/esm2015/src/render3/features/ng_onchanges_feature.js +2 -2
- package/esm2015/src/render3/global_utils_api.js +3 -3
- package/esm2015/src/render3/i18n.js +60 -56
- package/esm2015/src/render3/index.js +2 -2
- package/esm2015/src/render3/instructions/advance.js +10 -11
- package/esm2015/src/render3/instructions/all.js +4 -5
- package/esm2015/src/render3/instructions/attribute.js +6 -6
- package/esm2015/src/render3/instructions/attribute_interpolation.js +40 -21
- package/esm2015/src/render3/instructions/change_detection.js +8 -23
- package/esm2015/src/render3/instructions/class_map_interpolation.js +13 -12
- package/esm2015/src/render3/instructions/container.js +15 -12
- package/esm2015/src/render3/instructions/element.js +40 -127
- package/esm2015/src/render3/instructions/element_container.js +8 -10
- package/esm2015/src/render3/instructions/embedded_view.js +7 -7
- package/esm2015/src/render3/instructions/host_property.js +10 -7
- package/esm2015/src/render3/instructions/listener.js +18 -16
- package/esm2015/src/render3/instructions/lview_debug.js +160 -23
- package/esm2015/src/render3/instructions/projection.js +7 -5
- package/esm2015/src/render3/instructions/property.js +27 -6
- package/esm2015/src/render3/instructions/property_interpolation.js +40 -22
- package/esm2015/src/render3/instructions/shared.js +267 -240
- package/esm2015/src/render3/instructions/storage.js +6 -8
- package/esm2015/src/render3/instructions/style_prop_interpolation.js +12 -12
- package/esm2015/src/render3/instructions/styling.js +731 -475
- package/esm2015/src/render3/instructions/text.js +5 -5
- package/esm2015/src/render3/interfaces/definition.js +41 -1
- package/esm2015/src/render3/interfaces/node.js +160 -115
- package/esm2015/src/render3/interfaces/styling.js +183 -375
- package/esm2015/src/render3/interfaces/view.js +10 -2
- package/esm2015/src/render3/jit/environment.js +1 -3
- package/esm2015/src/render3/node_manipulation.js +177 -57
- package/esm2015/src/render3/node_selector_matcher.js +39 -24
- package/esm2015/src/render3/node_util.js +12 -7
- package/esm2015/src/render3/pipe.js +4 -6
- package/esm2015/src/render3/query.js +32 -26
- package/esm2015/src/render3/state.js +54 -183
- package/esm2015/src/render3/styling/class_differ.js +47 -0
- package/esm2015/src/render3/styling/static_styling.js +54 -0
- package/esm2015/src/render3/styling/style_binding_list.js +437 -0
- package/esm2015/src/render3/styling/styling_parser.js +336 -0
- package/esm2015/src/render3/tokens.js +2 -2
- package/esm2015/src/render3/util/attrs_utils.js +125 -2
- package/esm2015/src/render3/util/change_detection_utils.js +33 -0
- package/esm2015/src/render3/util/discovery_utils.js +146 -119
- package/esm2015/src/render3/util/global_utils.js +5 -5
- package/esm2015/src/render3/util/view_utils.js +6 -6
- package/esm2015/src/render3/view_engine_compatibility.js +16 -17
- package/esm2015/src/render3/view_ref.js +16 -13
- package/esm2015/src/sanitization/bypass.js +1 -1
- package/esm2015/src/sanitization/sanitization.js +20 -5
- package/esm2015/src/util/array_utils.js +240 -1
- package/esm2015/src/util/assert.js +37 -21
- package/esm2015/src/util/char_code.js +8 -0
- package/esm2015/src/util/iterable.js +4 -1
- package/esm2015/src/util/ng_dev_mode.js +1 -12
- package/esm2015/src/util/stringify.js +14 -1
- package/esm2015/src/version.js +1 -1
- package/esm2015/testing/src/r3_test_bed.js +5 -1
- package/esm2015/testing/src/r3_test_bed_compiler.js +2 -12
- package/esm2015/testing/src/styling.js +103 -0
- package/esm5/core.js +17 -17
- package/esm5/src/application_ref.js +6 -6
- package/esm5/src/core_private_export.js +7 -7
- package/esm5/src/core_render3_private_export.js +2 -2
- package/esm5/src/debug/debug_node.js +29 -11
- package/esm5/src/i18n/locale_data_api.js +1 -2
- package/esm5/src/interface/type.js +1 -1
- package/esm5/src/metadata/ng_module.js +1 -1
- package/esm5/src/render/api.js +4 -1
- package/esm5/src/render3/assert.js +4 -1
- package/esm5/src/render3/bindings.js +19 -2
- package/esm5/src/render3/component.js +47 -22
- package/esm5/src/render3/component_ref.js +9 -14
- package/esm5/src/render3/definition.js +3 -1
- package/esm5/src/render3/di.js +3 -4
- package/esm5/src/render3/di_setup.js +4 -5
- package/esm5/src/render3/errors.js +3 -1
- package/esm5/src/render3/features/inherit_definition_feature.js +36 -12
- package/esm5/src/render3/features/ng_onchanges_feature.js +1 -1
- package/esm5/src/render3/global_utils_api.js +3 -3
- package/esm5/src/render3/i18n.js +51 -51
- package/esm5/src/render3/index.js +2 -2
- package/esm5/src/render3/instructions/advance.js +9 -11
- package/esm5/src/render3/instructions/all.js +1 -2
- package/esm5/src/render3/instructions/attribute.js +5 -6
- package/esm5/src/render3/instructions/attribute_interpolation.js +31 -21
- package/esm5/src/render3/instructions/change_detection.js +8 -21
- package/esm5/src/render3/instructions/class_map_interpolation.js +13 -12
- package/esm5/src/render3/instructions/container.js +13 -12
- package/esm5/src/render3/instructions/element.js +36 -108
- package/esm5/src/render3/instructions/element_container.js +8 -9
- package/esm5/src/render3/instructions/embedded_view.js +7 -7
- package/esm5/src/render3/instructions/host_property.js +8 -7
- package/esm5/src/render3/instructions/listener.js +13 -13
- package/esm5/src/render3/instructions/lview_debug.js +56 -15
- package/esm5/src/render3/instructions/projection.js +6 -5
- package/esm5/src/render3/instructions/property.js +17 -6
- package/esm5/src/render3/instructions/property_interpolation.js +31 -23
- package/esm5/src/render3/instructions/shared.js +247 -207
- package/esm5/src/render3/instructions/storage.js +4 -6
- package/esm5/src/render3/instructions/style_prop_interpolation.js +12 -12
- package/esm5/src/render3/instructions/styling.js +685 -367
- package/esm5/src/render3/instructions/text.js +5 -5
- package/esm5/src/render3/interfaces/definition.js +1 -1
- package/esm5/src/render3/interfaces/node.js +49 -1
- package/esm5/src/render3/interfaces/styling.js +57 -1
- package/esm5/src/render3/interfaces/view.js +1 -1
- package/esm5/src/render3/jit/environment.js +1 -3
- package/esm5/src/render3/node_manipulation.js +167 -54
- package/esm5/src/render3/node_selector_matcher.js +40 -20
- package/esm5/src/render3/node_util.js +12 -7
- package/esm5/src/render3/pipe.js +4 -6
- package/esm5/src/render3/query.js +25 -24
- package/esm5/src/render3/state.js +34 -131
- package/esm5/src/render3/styling/class_differ.js +39 -0
- package/esm5/src/render3/styling/static_styling.js +42 -0
- package/esm5/src/render3/styling/style_binding_list.js +411 -0
- package/esm5/src/render3/styling/styling_parser.js +265 -0
- package/esm5/src/render3/tokens.js +2 -2
- package/esm5/src/render3/util/attrs_utils.js +117 -2
- package/esm5/src/render3/util/change_detection_utils.js +23 -0
- package/esm5/src/render3/util/discovery_utils.js +115 -99
- package/esm5/src/render3/util/global_utils.js +5 -5
- package/esm5/src/render3/util/view_utils.js +5 -5
- package/esm5/src/render3/view_engine_compatibility.js +37 -39
- package/esm5/src/render3/view_ref.js +14 -13
- package/esm5/src/sanitization/bypass.js +1 -1
- package/esm5/src/sanitization/sanitization.js +16 -5
- package/esm5/src/util/array_utils.js +240 -1
- package/esm5/src/util/assert.js +37 -21
- package/esm5/src/util/char_code.js +8 -0
- package/esm5/src/util/iterable.js +4 -1
- package/esm5/src/util/ng_dev_mode.js +1 -12
- package/esm5/src/util/stringify.js +14 -1
- package/esm5/src/version.js +1 -1
- package/esm5/testing/src/r3_test_bed.js +9 -1
- package/esm5/testing/src/r3_test_bed_compiler.js +2 -9
- package/esm5/testing/src/styling.js +82 -0
- package/fesm2015/core.js +5917 -6880
- package/fesm2015/core.js.map +1 -1
- package/fesm2015/testing.js +6 -12
- package/fesm2015/testing.js.map +1 -1
- package/fesm5/core.js +3588 -3884
- package/fesm5/core.js.map +1 -1
- package/fesm5/testing.js +10 -9
- package/fesm5/testing.js.map +1 -1
- package/package.json +1 -1
- package/src/r3_symbols.d.ts +1 -1
- package/testing/testing.d.ts +1 -3
- package/testing/testing.metadata.json +1 -1
- package/testing.d.ts +1 -1
- package/esm2015/global.js +0 -7
- package/esm2015/src/render3/instructions/alloc_host_vars.js +0 -80
- package/esm2015/src/render3/styling/bindings.js +0 -1248
- package/esm2015/src/render3/styling/map_based_bindings.js +0 -384
- package/esm2015/src/render3/styling/state.js +0 -135
- package/esm2015/src/render3/styling/styling_debug.js +0 -655
- package/esm2015/src/render3/util/styling_utils.js +0 -625
- package/esm5/global.js +0 -9
- package/esm5/src/render3/instructions/alloc_host_vars.js +0 -62
- package/esm5/src/render3/styling/bindings.js +0 -949
- package/esm5/src/render3/styling/map_based_bindings.js +0 -310
- package/esm5/src/render3/styling/state.js +0 -56
- package/esm5/src/render3/styling/styling_debug.js +0 -315
- package/esm5/src/render3/util/styling_utils.js +0 -378
|
@@ -0,0 +1,411 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google Inc. 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 { keyValueArrayIndexOf } from '../../util/array_utils';
|
|
9
|
+
import { assertDataInRange, assertEqual, assertNotEqual } from '../../util/assert';
|
|
10
|
+
import { assertFirstUpdatePass } from '../assert';
|
|
11
|
+
import { getTStylingRangeNext, getTStylingRangePrev, setTStylingRangeNext, setTStylingRangeNextDuplicate, setTStylingRangePrev, setTStylingRangePrevDuplicate, toTStylingRange } from '../interfaces/styling';
|
|
12
|
+
import { getTView } from '../state';
|
|
13
|
+
/**
|
|
14
|
+
* NOTE: The word `styling` is used interchangeably as style or class styling.
|
|
15
|
+
*
|
|
16
|
+
* This file contains code to link styling instructions together so that they can be replayed in
|
|
17
|
+
* priority order. The file exists because Ivy styling instruction execution order does not match
|
|
18
|
+
* that of the priority order. The purpose of this code is to create a linked list so that the
|
|
19
|
+
* instructions can be traversed in priority order when computing the styles.
|
|
20
|
+
*
|
|
21
|
+
* Assume we are dealing with the following code:
|
|
22
|
+
* ```
|
|
23
|
+
* @Component({
|
|
24
|
+
* template: `
|
|
25
|
+
* <my-cmp [style]=" {color: '#001'} "
|
|
26
|
+
* [style.color]=" #002 "
|
|
27
|
+
* dir-style-color-1
|
|
28
|
+
* dir-style-color-2> `
|
|
29
|
+
* })
|
|
30
|
+
* class ExampleComponent {
|
|
31
|
+
* static ngComp = ... {
|
|
32
|
+
* ...
|
|
33
|
+
* // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
|
|
34
|
+
* ɵɵstyleMap({color: '#001'});
|
|
35
|
+
* ɵɵstyleProp('color', '#002');
|
|
36
|
+
* ...
|
|
37
|
+
* }
|
|
38
|
+
* }
|
|
39
|
+
*
|
|
40
|
+
* @Directive({
|
|
41
|
+
* selector: `[dir-style-color-1]',
|
|
42
|
+
* })
|
|
43
|
+
* class Style1Directive {
|
|
44
|
+
* @HostBinding('style') style = {color: '#005'};
|
|
45
|
+
* @HostBinding('style.color') color = '#006';
|
|
46
|
+
*
|
|
47
|
+
* static ngDir = ... {
|
|
48
|
+
* ...
|
|
49
|
+
* // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
|
|
50
|
+
* ɵɵstyleMap({color: '#005'});
|
|
51
|
+
* ɵɵstyleProp('color', '#006');
|
|
52
|
+
* ...
|
|
53
|
+
* }
|
|
54
|
+
* }
|
|
55
|
+
*
|
|
56
|
+
* @Directive({
|
|
57
|
+
* selector: `[dir-style-color-2]',
|
|
58
|
+
* })
|
|
59
|
+
* class Style2Directive {
|
|
60
|
+
* @HostBinding('style') style = {color: '#007'};
|
|
61
|
+
* @HostBinding('style.color') color = '#008';
|
|
62
|
+
*
|
|
63
|
+
* static ngDir = ... {
|
|
64
|
+
* ...
|
|
65
|
+
* // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
|
|
66
|
+
* ɵɵstyleMap({color: '#007'});
|
|
67
|
+
* ɵɵstyleProp('color', '#008');
|
|
68
|
+
* ...
|
|
69
|
+
* }
|
|
70
|
+
* }
|
|
71
|
+
*
|
|
72
|
+
* @Directive({
|
|
73
|
+
* selector: `my-cmp',
|
|
74
|
+
* })
|
|
75
|
+
* class MyComponent {
|
|
76
|
+
* @HostBinding('style') style = {color: '#003'};
|
|
77
|
+
* @HostBinding('style.color') color = '#004';
|
|
78
|
+
*
|
|
79
|
+
* static ngComp = ... {
|
|
80
|
+
* ...
|
|
81
|
+
* // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
|
|
82
|
+
* ɵɵstyleMap({color: '#003'});
|
|
83
|
+
* ɵɵstyleProp('color', '#004');
|
|
84
|
+
* ...
|
|
85
|
+
* }
|
|
86
|
+
* }
|
|
87
|
+
* ```
|
|
88
|
+
*
|
|
89
|
+
* The Order of instruction execution is:
|
|
90
|
+
*
|
|
91
|
+
* NOTE: the comment binding location is for illustrative purposes only.
|
|
92
|
+
*
|
|
93
|
+
* ```
|
|
94
|
+
* // Template: (ExampleComponent)
|
|
95
|
+
* ɵɵstyleMap({color: '#001'}); // Binding index: 10
|
|
96
|
+
* ɵɵstyleProp('color', '#002'); // Binding index: 12
|
|
97
|
+
* // MyComponent
|
|
98
|
+
* ɵɵstyleMap({color: '#003'}); // Binding index: 20
|
|
99
|
+
* ɵɵstyleProp('color', '#004'); // Binding index: 22
|
|
100
|
+
* // Style1Directive
|
|
101
|
+
* ɵɵstyleMap({color: '#005'}); // Binding index: 24
|
|
102
|
+
* ɵɵstyleProp('color', '#006'); // Binding index: 26
|
|
103
|
+
* // Style2Directive
|
|
104
|
+
* ɵɵstyleMap({color: '#007'}); // Binding index: 28
|
|
105
|
+
* ɵɵstyleProp('color', '#008'); // Binding index: 30
|
|
106
|
+
* ```
|
|
107
|
+
*
|
|
108
|
+
* The correct priority order of concatenation is:
|
|
109
|
+
*
|
|
110
|
+
* ```
|
|
111
|
+
* // MyComponent
|
|
112
|
+
* ɵɵstyleMap({color: '#003'}); // Binding index: 20
|
|
113
|
+
* ɵɵstyleProp('color', '#004'); // Binding index: 22
|
|
114
|
+
* // Style1Directive
|
|
115
|
+
* ɵɵstyleMap({color: '#005'}); // Binding index: 24
|
|
116
|
+
* ɵɵstyleProp('color', '#006'); // Binding index: 26
|
|
117
|
+
* // Style2Directive
|
|
118
|
+
* ɵɵstyleMap({color: '#007'}); // Binding index: 28
|
|
119
|
+
* ɵɵstyleProp('color', '#008'); // Binding index: 30
|
|
120
|
+
* // Template: (ExampleComponent)
|
|
121
|
+
* ɵɵstyleMap({color: '#001'}); // Binding index: 10
|
|
122
|
+
* ɵɵstyleProp('color', '#002'); // Binding index: 12
|
|
123
|
+
* ```
|
|
124
|
+
*
|
|
125
|
+
* What color should be rendered?
|
|
126
|
+
*
|
|
127
|
+
* Once the items are correctly sorted in the list, the answer is simply the last item in the
|
|
128
|
+
* concatenation list which is `#002`.
|
|
129
|
+
*
|
|
130
|
+
* To do so we keep a linked list of all of the bindings which pertain to this element.
|
|
131
|
+
* Notice that the bindings are inserted in the order of execution, but the `TView.data` allows
|
|
132
|
+
* us to traverse them in the order of priority.
|
|
133
|
+
*
|
|
134
|
+
* |Idx|`TView.data`|`LView` | Notes
|
|
135
|
+
* |---|------------|-----------------|--------------
|
|
136
|
+
* |...| | |
|
|
137
|
+
* |10 |`null` |`{color: '#001'}`| `ɵɵstyleMap('color', {color: '#001'})`
|
|
138
|
+
* |11 |`30 | 12` | ... |
|
|
139
|
+
* |12 |`color` |`'#002'` | `ɵɵstyleProp('color', '#002')`
|
|
140
|
+
* |13 |`10 | 0` | ... |
|
|
141
|
+
* |...| | |
|
|
142
|
+
* |20 |`null` |`{color: '#003'}`| `ɵɵstyleMap('color', {color: '#003'})`
|
|
143
|
+
* |21 |`0 | 22` | ... |
|
|
144
|
+
* |22 |`color` |`'#004'` | `ɵɵstyleProp('color', '#004')`
|
|
145
|
+
* |23 |`20 | 24` | ... |
|
|
146
|
+
* |24 |`null` |`{color: '#005'}`| `ɵɵstyleMap('color', {color: '#005'})`
|
|
147
|
+
* |25 |`22 | 26` | ... |
|
|
148
|
+
* |26 |`color` |`'#006'` | `ɵɵstyleProp('color', '#006')`
|
|
149
|
+
* |27 |`24 | 28` | ... |
|
|
150
|
+
* |28 |`null` |`{color: '#007'}`| `ɵɵstyleMap('color', {color: '#007'})`
|
|
151
|
+
* |29 |`26 | 30` | ... |
|
|
152
|
+
* |30 |`color` |`'#008'` | `ɵɵstyleProp('color', '#008')`
|
|
153
|
+
* |31 |`28 | 10` | ... |
|
|
154
|
+
*
|
|
155
|
+
* The above data structure allows us to re-concatenate the styling no matter which data binding
|
|
156
|
+
* changes.
|
|
157
|
+
*
|
|
158
|
+
* NOTE: in addition to keeping track of next/previous index the `TView.data` also stores prev/next
|
|
159
|
+
* duplicate bit. The duplicate bit if true says there either is a binding with the same name or
|
|
160
|
+
* there is a map (which may contain the name). This information is useful in knowing if other
|
|
161
|
+
* styles with higher priority need to be searched for overwrites.
|
|
162
|
+
*
|
|
163
|
+
* NOTE: See `should support example in 'tnode_linked_list.ts' documentation` in
|
|
164
|
+
* `tnode_linked_list_spec.ts` for working example.
|
|
165
|
+
*/
|
|
166
|
+
var __unused_const_as_closure_does_not_like_standalone_comment_blocks__;
|
|
167
|
+
/**
|
|
168
|
+
* Insert new `tStyleValue` at `TData` and link existing style bindings such that we maintain linked
|
|
169
|
+
* list of styles and compute the duplicate flag.
|
|
170
|
+
*
|
|
171
|
+
* Note: this function is executed during `firstUpdatePass` only to populate the `TView.data`.
|
|
172
|
+
*
|
|
173
|
+
* The function works by keeping track of `tStylingRange` which contains two pointers pointing to
|
|
174
|
+
* the head/tail of the template portion of the styles.
|
|
175
|
+
* - if `isHost === false` (we are template) then insertion is at tail of `TStylingRange`
|
|
176
|
+
* - if `isHost === true` (we are host binding) then insertion is at head of `TStylingRange`
|
|
177
|
+
*
|
|
178
|
+
* @param tData The `TData` to insert into.
|
|
179
|
+
* @param tNode `TNode` associated with the styling element.
|
|
180
|
+
* @param tStylingKey See `TStylingKey`.
|
|
181
|
+
* @param index location of where `tStyleValue` should be stored (and linked into list.)
|
|
182
|
+
* @param isHostBinding `true` if the insertion is for a `hostBinding`. (insertion is in front of
|
|
183
|
+
* template.)
|
|
184
|
+
* @param isClassBinding True if the associated `tStylingKey` as a `class` styling.
|
|
185
|
+
* `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
|
|
186
|
+
*/
|
|
187
|
+
export function insertTStylingBinding(tData, tNode, tStylingKeyWithStatic, index, isHostBinding, isClassBinding) {
|
|
188
|
+
ngDevMode && assertFirstUpdatePass(getTView());
|
|
189
|
+
var tBindings = isClassBinding ? tNode.classBindings : tNode.styleBindings;
|
|
190
|
+
var tmplHead = getTStylingRangePrev(tBindings);
|
|
191
|
+
var tmplTail = getTStylingRangeNext(tBindings);
|
|
192
|
+
tData[index] = tStylingKeyWithStatic;
|
|
193
|
+
var isKeyDuplicateOfStatic = false;
|
|
194
|
+
var tStylingKey;
|
|
195
|
+
if (Array.isArray(tStylingKeyWithStatic)) {
|
|
196
|
+
// We are case when the `TStylingKey` contains static fields as well.
|
|
197
|
+
var staticKeyValueArray = tStylingKeyWithStatic;
|
|
198
|
+
tStylingKey = staticKeyValueArray[1]; // unwrap.
|
|
199
|
+
// We need to check if our key is present in the static so that we can mark it as duplicate.
|
|
200
|
+
if (tStylingKey === null ||
|
|
201
|
+
keyValueArrayIndexOf(staticKeyValueArray, tStylingKey) > 0) {
|
|
202
|
+
// tStylingKey is present in the statics, need to mark it as duplicate.
|
|
203
|
+
isKeyDuplicateOfStatic = true;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
else {
|
|
207
|
+
tStylingKey = tStylingKeyWithStatic;
|
|
208
|
+
}
|
|
209
|
+
if (isHostBinding) {
|
|
210
|
+
// We are inserting host bindings
|
|
211
|
+
// If we don't have template bindings then `tail` is 0.
|
|
212
|
+
var hasTemplateBindings = tmplTail !== 0;
|
|
213
|
+
// This is important to know because that means that the `head` can't point to the first
|
|
214
|
+
// template bindings (there are none.) Instead the head points to the tail of the template.
|
|
215
|
+
if (hasTemplateBindings) {
|
|
216
|
+
// template head's "prev" will point to last host binding or to 0 if no host bindings yet
|
|
217
|
+
var previousNode = getTStylingRangePrev(tData[tmplHead + 1]);
|
|
218
|
+
tData[index + 1] = toTStylingRange(previousNode, tmplHead);
|
|
219
|
+
// if a host binding has already been registered, we need to update the next of that host
|
|
220
|
+
// binding to point to this one
|
|
221
|
+
if (previousNode !== 0) {
|
|
222
|
+
// We need to update the template-tail value to point to us.
|
|
223
|
+
tData[previousNode + 1] =
|
|
224
|
+
setTStylingRangeNext(tData[previousNode + 1], index);
|
|
225
|
+
}
|
|
226
|
+
// The "previous" of the template binding head should point to this host binding
|
|
227
|
+
tData[tmplHead + 1] = setTStylingRangePrev(tData[tmplHead + 1], index);
|
|
228
|
+
}
|
|
229
|
+
else {
|
|
230
|
+
tData[index + 1] = toTStylingRange(tmplHead, 0);
|
|
231
|
+
// if a host binding has already been registered, we need to update the next of that host
|
|
232
|
+
// binding to point to this one
|
|
233
|
+
if (tmplHead !== 0) {
|
|
234
|
+
// We need to update the template-tail value to point to us.
|
|
235
|
+
tData[tmplHead + 1] = setTStylingRangeNext(tData[tmplHead + 1], index);
|
|
236
|
+
}
|
|
237
|
+
// if we don't have template, the head points to template-tail, and needs to be advanced.
|
|
238
|
+
tmplHead = index;
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
else {
|
|
242
|
+
// We are inserting in template section.
|
|
243
|
+
// We need to set this binding's "previous" to the current template tail
|
|
244
|
+
tData[index + 1] = toTStylingRange(tmplTail, 0);
|
|
245
|
+
ngDevMode && assertEqual(tmplHead !== 0 && tmplTail === 0, false, 'Adding template bindings after hostBindings is not allowed.');
|
|
246
|
+
if (tmplHead === 0) {
|
|
247
|
+
tmplHead = index;
|
|
248
|
+
}
|
|
249
|
+
else {
|
|
250
|
+
// We need to update the previous value "next" to point to this binding
|
|
251
|
+
tData[tmplTail + 1] = setTStylingRangeNext(tData[tmplTail + 1], index);
|
|
252
|
+
}
|
|
253
|
+
tmplTail = index;
|
|
254
|
+
}
|
|
255
|
+
// Now we need to update / compute the duplicates.
|
|
256
|
+
// Starting with our location search towards head (least priority)
|
|
257
|
+
if (isKeyDuplicateOfStatic) {
|
|
258
|
+
tData[index + 1] = setTStylingRangePrevDuplicate(tData[index + 1]);
|
|
259
|
+
}
|
|
260
|
+
markDuplicates(tData, tStylingKey, index, true, isClassBinding);
|
|
261
|
+
markDuplicates(tData, tStylingKey, index, false, isClassBinding);
|
|
262
|
+
markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding);
|
|
263
|
+
tBindings = toTStylingRange(tmplHead, tmplTail);
|
|
264
|
+
if (isClassBinding) {
|
|
265
|
+
tNode.classBindings = tBindings;
|
|
266
|
+
}
|
|
267
|
+
else {
|
|
268
|
+
tNode.styleBindings = tBindings;
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
/**
|
|
272
|
+
* Look into the residual styling to see if the current `tStylingKey` is duplicate of residual.
|
|
273
|
+
*
|
|
274
|
+
* @param tNode `TNode` where the residual is stored.
|
|
275
|
+
* @param tStylingKey `TStylingKey` to store.
|
|
276
|
+
* @param tData `TData` associated with the current `LView`.
|
|
277
|
+
* @param index location of where `tStyleValue` should be stored (and linked into list.)
|
|
278
|
+
* @param isClassBinding True if the associated `tStylingKey` as a `class` styling.
|
|
279
|
+
* `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
|
|
280
|
+
*/
|
|
281
|
+
function markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding) {
|
|
282
|
+
var residual = isClassBinding ? tNode.residualClasses : tNode.residualStyles;
|
|
283
|
+
if (residual != null /* or undefined */ && typeof tStylingKey == 'string' &&
|
|
284
|
+
keyValueArrayIndexOf(residual, tStylingKey) >= 0) {
|
|
285
|
+
// We have duplicate in the residual so mark ourselves as duplicate.
|
|
286
|
+
tData[index + 1] = setTStylingRangeNextDuplicate(tData[index + 1]);
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
/**
|
|
290
|
+
* Marks `TStyleValue`s as duplicates if another style binding in the list has the same
|
|
291
|
+
* `TStyleValue`.
|
|
292
|
+
*
|
|
293
|
+
* NOTE: this function is intended to be called twice once with `isPrevDir` set to `true` and once
|
|
294
|
+
* with it set to `false` to search both the previous as well as next items in the list.
|
|
295
|
+
*
|
|
296
|
+
* No duplicate case
|
|
297
|
+
* ```
|
|
298
|
+
* [style.color]
|
|
299
|
+
* [style.width.px] <<- index
|
|
300
|
+
* [style.height.px]
|
|
301
|
+
* ```
|
|
302
|
+
*
|
|
303
|
+
* In the above case adding `[style.width.px]` to the existing `[style.color]` produces no
|
|
304
|
+
* duplicates because `width` is not found in any other part of the linked list.
|
|
305
|
+
*
|
|
306
|
+
* Duplicate case
|
|
307
|
+
* ```
|
|
308
|
+
* [style.color]
|
|
309
|
+
* [style.width.em]
|
|
310
|
+
* [style.width.px] <<- index
|
|
311
|
+
* ```
|
|
312
|
+
* In the above case adding `[style.width.px]` will produce a duplicate with `[style.width.em]`
|
|
313
|
+
* because `width` is found in the chain.
|
|
314
|
+
*
|
|
315
|
+
* Map case 1
|
|
316
|
+
* ```
|
|
317
|
+
* [style.width.px]
|
|
318
|
+
* [style.color]
|
|
319
|
+
* [style] <<- index
|
|
320
|
+
* ```
|
|
321
|
+
* In the above case adding `[style]` will produce a duplicate with any other bindings because
|
|
322
|
+
* `[style]` is a Map and as such is fully dynamic and could produce `color` or `width`.
|
|
323
|
+
*
|
|
324
|
+
* Map case 2
|
|
325
|
+
* ```
|
|
326
|
+
* [style]
|
|
327
|
+
* [style.width.px]
|
|
328
|
+
* [style.color] <<- index
|
|
329
|
+
* ```
|
|
330
|
+
* In the above case adding `[style.color]` will produce a duplicate because there is already a
|
|
331
|
+
* `[style]` binding which is a Map and as such is fully dynamic and could produce `color` or
|
|
332
|
+
* `width`.
|
|
333
|
+
*
|
|
334
|
+
* NOTE: Once `[style]` (Map) is added into the system all things are mapped as duplicates.
|
|
335
|
+
* NOTE: We use `style` as example, but same logic is applied to `class`es as well.
|
|
336
|
+
*
|
|
337
|
+
* @param tData `TData` where the linked list is stored.
|
|
338
|
+
* @param tStylingKey `TStylingKeyPrimitive` which contains the value to compare to other keys in
|
|
339
|
+
* the linked list.
|
|
340
|
+
* @param index Starting location in the linked list to search from
|
|
341
|
+
* @param isPrevDir Direction.
|
|
342
|
+
* - `true` for previous (lower priority);
|
|
343
|
+
* - `false` for next (higher priority).
|
|
344
|
+
*/
|
|
345
|
+
function markDuplicates(tData, tStylingKey, index, isPrevDir, isClassBinding) {
|
|
346
|
+
var tStylingAtIndex = tData[index + 1];
|
|
347
|
+
var isMap = tStylingKey === null;
|
|
348
|
+
var cursor = isPrevDir ? getTStylingRangePrev(tStylingAtIndex) : getTStylingRangeNext(tStylingAtIndex);
|
|
349
|
+
var foundDuplicate = false;
|
|
350
|
+
// We keep iterating as long as we have a cursor
|
|
351
|
+
// AND either:
|
|
352
|
+
// - we found what we are looking for, OR
|
|
353
|
+
// - we are a map in which case we have to continue searching even after we find what we were
|
|
354
|
+
// looking for since we are a wild card and everything needs to be flipped to duplicate.
|
|
355
|
+
while (cursor !== 0 && (foundDuplicate === false || isMap)) {
|
|
356
|
+
ngDevMode && assertDataInRange(tData, cursor);
|
|
357
|
+
var tStylingValueAtCursor = tData[cursor];
|
|
358
|
+
var tStyleRangeAtCursor = tData[cursor + 1];
|
|
359
|
+
if (isStylingMatch(tStylingValueAtCursor, tStylingKey)) {
|
|
360
|
+
foundDuplicate = true;
|
|
361
|
+
tData[cursor + 1] = isPrevDir ? setTStylingRangeNextDuplicate(tStyleRangeAtCursor) :
|
|
362
|
+
setTStylingRangePrevDuplicate(tStyleRangeAtCursor);
|
|
363
|
+
}
|
|
364
|
+
cursor = isPrevDir ? getTStylingRangePrev(tStyleRangeAtCursor) :
|
|
365
|
+
getTStylingRangeNext(tStyleRangeAtCursor);
|
|
366
|
+
}
|
|
367
|
+
if (foundDuplicate) {
|
|
368
|
+
// if we found a duplicate, than mark ourselves.
|
|
369
|
+
tData[index + 1] = isPrevDir ? setTStylingRangePrevDuplicate(tStylingAtIndex) :
|
|
370
|
+
setTStylingRangeNextDuplicate(tStylingAtIndex);
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
/**
|
|
374
|
+
* Determines if two `TStylingKey`s are a match.
|
|
375
|
+
*
|
|
376
|
+
* When computing weather a binding contains a duplicate, we need to compare if the instruction
|
|
377
|
+
* `TStylingKey` has a match.
|
|
378
|
+
*
|
|
379
|
+
* Here are examples of `TStylingKey`s which match given `tStylingKeyCursor` is:
|
|
380
|
+
* - `color`
|
|
381
|
+
* - `color` // Match another color
|
|
382
|
+
* - `null` // That means that `tStylingKey` is a `classMap`/`styleMap` instruction
|
|
383
|
+
* - `['', 'color', 'other', true]` // wrapped `color` so match
|
|
384
|
+
* - `['', null, 'other', true]` // wrapped `null` so match
|
|
385
|
+
* - `['', 'width', 'color', 'value']` // wrapped static value contains a match on `'color'`
|
|
386
|
+
* - `null` // `tStylingKeyCursor` always match as it is `classMap`/`styleMap` instruction
|
|
387
|
+
*
|
|
388
|
+
* @param tStylingKeyCursor
|
|
389
|
+
* @param tStylingKey
|
|
390
|
+
*/
|
|
391
|
+
function isStylingMatch(tStylingKeyCursor, tStylingKey) {
|
|
392
|
+
ngDevMode &&
|
|
393
|
+
assertNotEqual(Array.isArray(tStylingKey), true, 'Expected that \'tStylingKey\' has been unwrapped');
|
|
394
|
+
if (tStylingKeyCursor === null || // If the cursor is `null` it means that we have map at that
|
|
395
|
+
// location so we must assume that we have a match.
|
|
396
|
+
tStylingKey == null || // If `tStylingKey` is `null` then it is a map therefor assume that it
|
|
397
|
+
// contains a match.
|
|
398
|
+
(Array.isArray(tStylingKeyCursor) ? tStylingKeyCursor[1] : tStylingKeyCursor) ===
|
|
399
|
+
tStylingKey // If the keys match explicitly than we are a match.
|
|
400
|
+
) {
|
|
401
|
+
return true;
|
|
402
|
+
}
|
|
403
|
+
else if (Array.isArray(tStylingKeyCursor) && typeof tStylingKey === 'string') {
|
|
404
|
+
// if we did not find a match, but `tStylingKeyCursor` is `KeyValueArray` that means cursor has
|
|
405
|
+
// statics and we need to check those as well.
|
|
406
|
+
return keyValueArrayIndexOf(tStylingKeyCursor, tStylingKey) >=
|
|
407
|
+
0; // see if we are matching the key
|
|
408
|
+
}
|
|
409
|
+
return false;
|
|
410
|
+
}
|
|
411
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"style_binding_list.js","sourceRoot":"","sources":["../../../../../../../../../../../packages/core/src/render3/styling/style_binding_list.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n* @license\n* Copyright Google Inc. 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 {KeyValueArray, keyValueArrayIndexOf} from '../../util/array_utils';\nimport {assertDataInRange, assertEqual, assertNotEqual} from '../../util/assert';\nimport {assertFirstUpdatePass} from '../assert';\nimport {TNode} from '../interfaces/node';\nimport {TStylingKey, TStylingKeyPrimitive, TStylingRange, getTStylingRangeNext, getTStylingRangePrev, setTStylingRangeNext, setTStylingRangeNextDuplicate, setTStylingRangePrev, setTStylingRangePrevDuplicate, toTStylingRange} from '../interfaces/styling';\nimport {TData} from '../interfaces/view';\nimport {getTView} from '../state';\n\n\n/**\n * NOTE: The word `styling` is used interchangeably as style or class styling.\n *\n * This file contains code to link styling instructions together so that they can be replayed in\n * priority order. The file exists because Ivy styling instruction execution order does not match\n * that of the priority order. The purpose of this code is to create a linked list so that the\n * instructions can be traversed in priority order when computing the styles.\n *\n * Assume we are dealing with the following code:\n * ```\n * @Component({\n *   template: `\n *     <my-cmp [style]=\" {color: '#001'} \"\n *             [style.color]=\" #002 \"\n *             dir-style-color-1\n *             dir-style-color-2> `\n * })\n * class ExampleComponent {\n *   static ngComp = ... {\n *     ...\n *     // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`\n *     ɵɵstyleMap({color: '#001'});\n *     ɵɵstyleProp('color', '#002');\n *     ...\n *   }\n * }\n *\n * @Directive({\n *   selector: `[dir-style-color-1]',\n * })\n * class Style1Directive {\n *   @HostBinding('style') style = {color: '#005'};\n *   @HostBinding('style.color') color = '#006';\n *\n *   static ngDir = ... {\n *     ...\n *     // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`\n *     ɵɵstyleMap({color: '#005'});\n *     ɵɵstyleProp('color', '#006');\n *     ...\n *   }\n * }\n *\n * @Directive({\n *   selector: `[dir-style-color-2]',\n * })\n * class Style2Directive {\n *   @HostBinding('style') style = {color: '#007'};\n *   @HostBinding('style.color') color = '#008';\n *\n *   static ngDir = ... {\n *     ...\n *     // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`\n *     ɵɵstyleMap({color: '#007'});\n *     ɵɵstyleProp('color', '#008');\n *     ...\n *   }\n * }\n *\n * @Directive({\n *   selector: `my-cmp',\n * })\n * class MyComponent {\n *   @HostBinding('style') style = {color: '#003'};\n *   @HostBinding('style.color') color = '#004';\n *\n *   static ngComp = ... {\n *     ...\n *     // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`\n *     ɵɵstyleMap({color: '#003'});\n *     ɵɵstyleProp('color', '#004');\n *     ...\n *   }\n * }\n * ```\n *\n * The Order of instruction execution is:\n *\n * NOTE: the comment binding location is for illustrative purposes only.\n *\n * ```\n * // Template: (ExampleComponent)\n *     ɵɵstyleMap({color: '#001'});   // Binding index: 10\n *     ɵɵstyleProp('color', '#002');  // Binding index: 12\n * // MyComponent\n *     ɵɵstyleMap({color: '#003'});   // Binding index: 20\n *     ɵɵstyleProp('color', '#004');  // Binding index: 22\n * // Style1Directive\n *     ɵɵstyleMap({color: '#005'});   // Binding index: 24\n *     ɵɵstyleProp('color', '#006');  // Binding index: 26\n * // Style2Directive\n *     ɵɵstyleMap({color: '#007'});   // Binding index: 28\n *     ɵɵstyleProp('color', '#008');  // Binding index: 30\n * ```\n *\n * The correct priority order of concatenation is:\n *\n * ```\n * // MyComponent\n *     ɵɵstyleMap({color: '#003'});   // Binding index: 20\n *     ɵɵstyleProp('color', '#004');  // Binding index: 22\n * // Style1Directive\n *     ɵɵstyleMap({color: '#005'});   // Binding index: 24\n *     ɵɵstyleProp('color', '#006');  // Binding index: 26\n * // Style2Directive\n *     ɵɵstyleMap({color: '#007'});   // Binding index: 28\n *     ɵɵstyleProp('color', '#008');  // Binding index: 30\n * // Template: (ExampleComponent)\n *     ɵɵstyleMap({color: '#001'});   // Binding index: 10\n *     ɵɵstyleProp('color', '#002');  // Binding index: 12\n * ```\n *\n * What color should be rendered?\n *\n * Once the items are correctly sorted in the list, the answer is simply the last item in the\n * concatenation list which is `#002`.\n *\n * To do so we keep a linked list of all of the bindings which pertain to this element.\n * Notice that the bindings are inserted in the order of execution, but the `TView.data` allows\n * us to traverse them in the order of priority.\n *\n * |Idx|`TView.data`|`LView`          | Notes\n * |---|------------|-----------------|--------------\n * |...|            |                 |\n * |10 |`null`      |`{color: '#001'}`| `ɵɵstyleMap('color', {color: '#001'})`\n * |11 |`30 | 12`   | ...             |\n * |12 |`color`     |`'#002'`         | `ɵɵstyleProp('color', '#002')`\n * |13 |`10 | 0`    | ...             |\n * |...|            |                 |\n * |20 |`null`      |`{color: '#003'}`| `ɵɵstyleMap('color', {color: '#003'})`\n * |21 |`0 | 22`    | ...             |\n * |22 |`color`     |`'#004'`         | `ɵɵstyleProp('color', '#004')`\n * |23 |`20 | 24`   | ...             |\n * |24 |`null`      |`{color: '#005'}`| `ɵɵstyleMap('color', {color: '#005'})`\n * |25 |`22 | 26`   | ...             |\n * |26 |`color`     |`'#006'`         | `ɵɵstyleProp('color', '#006')`\n * |27 |`24 | 28`   | ...             |\n * |28 |`null`      |`{color: '#007'}`| `ɵɵstyleMap('color', {color: '#007'})`\n * |29 |`26 | 30`   | ...             |\n * |30 |`color`     |`'#008'`         | `ɵɵstyleProp('color', '#008')`\n * |31 |`28 | 10`   | ...             |\n *\n * The above data structure allows us to re-concatenate the styling no matter which data binding\n * changes.\n *\n * NOTE: in addition to keeping track of next/previous index the `TView.data` also stores prev/next\n * duplicate bit. The duplicate bit if true says there either is a binding with the same name or\n * there is a map (which may contain the name). This information is useful in knowing if other\n * styles with higher priority need to be searched for overwrites.\n *\n * NOTE: See `should support example in 'tnode_linked_list.ts' documentation` in\n * `tnode_linked_list_spec.ts` for working example.\n */\nlet __unused_const_as_closure_does_not_like_standalone_comment_blocks__: undefined;\n\n/**\n * Insert new `tStyleValue` at `TData` and link existing style bindings such that we maintain linked\n * list of styles and compute the duplicate flag.\n *\n * Note: this function is executed during `firstUpdatePass` only to populate the `TView.data`.\n *\n * The function works by keeping track of `tStylingRange` which contains two pointers pointing to\n * the head/tail of the template portion of the styles.\n *  - if `isHost === false` (we are template) then insertion is at tail of `TStylingRange`\n *  - if `isHost === true` (we are host binding) then insertion is at head of `TStylingRange`\n *\n * @param tData The `TData` to insert into.\n * @param tNode `TNode` associated with the styling element.\n * @param tStylingKey See `TStylingKey`.\n * @param index location of where `tStyleValue` should be stored (and linked into list.)\n * @param isHostBinding `true` if the insertion is for a `hostBinding`. (insertion is in front of\n *               template.)\n * @param isClassBinding True if the associated `tStylingKey` as a `class` styling.\n *                       `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)\n */\nexport function insertTStylingBinding(\n    tData: TData, tNode: TNode, tStylingKeyWithStatic: TStylingKey, index: number,\n    isHostBinding: boolean, isClassBinding: boolean): void {\n  ngDevMode && assertFirstUpdatePass(getTView());\n  let tBindings = isClassBinding ? tNode.classBindings : tNode.styleBindings;\n  let tmplHead = getTStylingRangePrev(tBindings);\n  let tmplTail = getTStylingRangeNext(tBindings);\n\n  tData[index] = tStylingKeyWithStatic;\n  let isKeyDuplicateOfStatic = false;\n  let tStylingKey: TStylingKeyPrimitive;\n  if (Array.isArray(tStylingKeyWithStatic)) {\n    // We are case when the `TStylingKey` contains static fields as well.\n    const staticKeyValueArray = tStylingKeyWithStatic as KeyValueArray<any>;\n    tStylingKey = staticKeyValueArray[1];  // unwrap.\n    // We need to check if our key is present in the static so that we can mark it as duplicate.\n    if (tStylingKey === null ||\n        keyValueArrayIndexOf(staticKeyValueArray, tStylingKey as string) > 0) {\n      // tStylingKey is present in the statics, need to mark it as duplicate.\n      isKeyDuplicateOfStatic = true;\n    }\n  } else {\n    tStylingKey = tStylingKeyWithStatic;\n  }\n  if (isHostBinding) {\n    // We are inserting host bindings\n\n    // If we don't have template bindings then `tail` is 0.\n    const hasTemplateBindings = tmplTail !== 0;\n    // This is important to know because that means that the `head` can't point to the first\n    // template bindings (there are none.) Instead the head points to the tail of the template.\n    if (hasTemplateBindings) {\n      // template head's \"prev\" will point to last host binding or to 0 if no host bindings yet\n      const previousNode = getTStylingRangePrev(tData[tmplHead + 1] as TStylingRange);\n      tData[index + 1] = toTStylingRange(previousNode, tmplHead);\n      // if a host binding has already been registered, we need to update the next of that host\n      // binding to point to this one\n      if (previousNode !== 0) {\n        // We need to update the template-tail value to point to us.\n        tData[previousNode + 1] =\n            setTStylingRangeNext(tData[previousNode + 1] as TStylingRange, index);\n      }\n      // The \"previous\" of the template binding head should point to this host binding\n      tData[tmplHead + 1] = setTStylingRangePrev(tData[tmplHead + 1] as TStylingRange, index);\n    } else {\n      tData[index + 1] = toTStylingRange(tmplHead, 0);\n      // if a host binding has already been registered, we need to update the next of that host\n      // binding to point to this one\n      if (tmplHead !== 0) {\n        // We need to update the template-tail value to point to us.\n        tData[tmplHead + 1] = setTStylingRangeNext(tData[tmplHead + 1] as TStylingRange, index);\n      }\n      // if we don't have template, the head points to template-tail, and needs to be advanced.\n      tmplHead = index;\n    }\n  } else {\n    // We are inserting in template section.\n    // We need to set this binding's \"previous\" to the current template tail\n    tData[index + 1] = toTStylingRange(tmplTail, 0);\n    ngDevMode && assertEqual(\n                     tmplHead !== 0 && tmplTail === 0, false,\n                     'Adding template bindings after hostBindings is not allowed.');\n    if (tmplHead === 0) {\n      tmplHead = index;\n    } else {\n      // We need to update the previous value \"next\" to point to this binding\n      tData[tmplTail + 1] = setTStylingRangeNext(tData[tmplTail + 1] as TStylingRange, index);\n    }\n    tmplTail = index;\n  }\n\n  // Now we need to update / compute the duplicates.\n  // Starting with our location search towards head (least priority)\n  if (isKeyDuplicateOfStatic) {\n    tData[index + 1] = setTStylingRangePrevDuplicate(tData[index + 1] as TStylingRange);\n  }\n  markDuplicates(tData, tStylingKey, index, true, isClassBinding);\n  markDuplicates(tData, tStylingKey, index, false, isClassBinding);\n  markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding);\n\n  tBindings = toTStylingRange(tmplHead, tmplTail);\n  if (isClassBinding) {\n    tNode.classBindings = tBindings;\n  } else {\n    tNode.styleBindings = tBindings;\n  }\n}\n\n/**\n * Look into the residual styling to see if the current `tStylingKey` is duplicate of residual.\n *\n * @param tNode `TNode` where the residual is stored.\n * @param tStylingKey `TStylingKey` to store.\n * @param tData `TData` associated with the current `LView`.\n * @param index location of where `tStyleValue` should be stored (and linked into list.)\n * @param isClassBinding True if the associated `tStylingKey` as a `class` styling.\n *                       `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)\n */\nfunction markDuplicateOfResidualStyling(\n    tNode: TNode, tStylingKey: TStylingKey, tData: TData, index: number, isClassBinding: boolean) {\n  const residual = isClassBinding ? tNode.residualClasses : tNode.residualStyles;\n  if (residual != null /* or undefined */ && typeof tStylingKey == 'string' &&\n      keyValueArrayIndexOf(residual, tStylingKey) >= 0) {\n    // We have duplicate in the residual so mark ourselves as duplicate.\n    tData[index + 1] = setTStylingRangeNextDuplicate(tData[index + 1] as TStylingRange);\n  }\n}\n\n\n/**\n * Marks `TStyleValue`s as duplicates if another style binding in the list has the same\n * `TStyleValue`.\n *\n * NOTE: this function is intended to be called twice once with `isPrevDir` set to `true` and once\n * with it set to `false` to search both the previous as well as next items in the list.\n *\n * No duplicate case\n * ```\n *   [style.color]\n *   [style.width.px] <<- index\n *   [style.height.px]\n * ```\n *\n * In the above case adding `[style.width.px]` to the existing `[style.color]` produces no\n * duplicates because `width` is not found in any other part of the linked list.\n *\n * Duplicate case\n * ```\n *   [style.color]\n *   [style.width.em]\n *   [style.width.px] <<- index\n * ```\n * In the above case adding `[style.width.px]` will produce a duplicate with `[style.width.em]`\n * because `width` is found in the chain.\n *\n * Map case 1\n * ```\n *   [style.width.px]\n *   [style.color]\n *   [style]  <<- index\n * ```\n * In the above case adding `[style]` will produce a duplicate with any other bindings because\n * `[style]` is a Map and as such is fully dynamic and could produce `color` or `width`.\n *\n * Map case 2\n * ```\n *   [style]\n *   [style.width.px]\n *   [style.color]  <<- index\n * ```\n * In the above case adding `[style.color]` will produce a duplicate because there is already a\n * `[style]` binding which is a Map and as such is fully dynamic and could produce `color` or\n * `width`.\n *\n * NOTE: Once `[style]` (Map) is added into the system all things are mapped as duplicates.\n * NOTE: We use `style` as example, but same logic is applied to `class`es as well.\n *\n * @param tData `TData` where the linked list is stored.\n * @param tStylingKey `TStylingKeyPrimitive` which contains the value to compare to other keys in\n *        the linked list.\n * @param index Starting location in the linked list to search from\n * @param isPrevDir Direction.\n *        - `true` for previous (lower priority);\n *        - `false` for next (higher priority).\n */\nfunction markDuplicates(\n    tData: TData, tStylingKey: TStylingKeyPrimitive, index: number, isPrevDir: boolean,\n    isClassBinding: boolean) {\n  const tStylingAtIndex = tData[index + 1] as TStylingRange;\n  const isMap = tStylingKey === null;\n  let cursor =\n      isPrevDir ? getTStylingRangePrev(tStylingAtIndex) : getTStylingRangeNext(tStylingAtIndex);\n  let foundDuplicate = false;\n  // We keep iterating as long as we have a cursor\n  // AND either:\n  // - we found what we are looking for, OR\n  // - we are a map in which case we have to continue searching even after we find what we were\n  //   looking for since we are a wild card and everything needs to be flipped to duplicate.\n  while (cursor !== 0 && (foundDuplicate === false || isMap)) {\n    ngDevMode && assertDataInRange(tData, cursor);\n    const tStylingValueAtCursor = tData[cursor] as TStylingKey;\n    const tStyleRangeAtCursor = tData[cursor + 1] as TStylingRange;\n    if (isStylingMatch(tStylingValueAtCursor, tStylingKey)) {\n      foundDuplicate = true;\n      tData[cursor + 1] = isPrevDir ? setTStylingRangeNextDuplicate(tStyleRangeAtCursor) :\n                                      setTStylingRangePrevDuplicate(tStyleRangeAtCursor);\n    }\n    cursor = isPrevDir ? getTStylingRangePrev(tStyleRangeAtCursor) :\n                         getTStylingRangeNext(tStyleRangeAtCursor);\n  }\n  if (foundDuplicate) {\n    // if we found a duplicate, than mark ourselves.\n    tData[index + 1] = isPrevDir ? setTStylingRangePrevDuplicate(tStylingAtIndex) :\n                                   setTStylingRangeNextDuplicate(tStylingAtIndex);\n  }\n}\n\n/**\n * Determines if two `TStylingKey`s are a match.\n *\n * When computing weather a binding contains a duplicate, we need to compare if the instruction\n * `TStylingKey` has a match.\n *\n * Here are examples of `TStylingKey`s which match given `tStylingKeyCursor` is:\n * - `color`\n *    - `color`    // Match another color\n *    - `null`     // That means that `tStylingKey` is a `classMap`/`styleMap` instruction\n *    - `['', 'color', 'other', true]` // wrapped `color` so match\n *    - `['', null, 'other', true]`       // wrapped `null` so match\n *    - `['', 'width', 'color', 'value']` // wrapped static value contains a match on `'color'`\n * - `null`       // `tStylingKeyCursor` always match as it is `classMap`/`styleMap` instruction\n *\n * @param tStylingKeyCursor\n * @param tStylingKey\n */\nfunction isStylingMatch(tStylingKeyCursor: TStylingKey, tStylingKey: TStylingKeyPrimitive) {\n  ngDevMode &&\n      assertNotEqual(\n          Array.isArray(tStylingKey), true, 'Expected that \\'tStylingKey\\' has been unwrapped');\n  if (tStylingKeyCursor === null ||  // If the cursor is `null` it means that we have map at that\n                                     // location so we must assume that we have a match.\n      tStylingKey == null ||  // If `tStylingKey` is `null` then it is a map therefor assume that it\n                              // contains a match.\n      (Array.isArray(tStylingKeyCursor) ? tStylingKeyCursor[1] : tStylingKeyCursor) ===\n          tStylingKey  // If the keys match explicitly than we are a match.\n      ) {\n    return true;\n  } else if (Array.isArray(tStylingKeyCursor) && typeof tStylingKey === 'string') {\n    // if we did not find a match, but `tStylingKeyCursor` is `KeyValueArray` that means cursor has\n    // statics and we need to check those as well.\n    return keyValueArrayIndexOf(tStylingKeyCursor, tStylingKey) >=\n        0;  // see if we are matching the key\n  }\n  return false;\n}\n"]}
|