@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.
Files changed (196) hide show
  1. package/bundles/core-testing.umd.js +10 -9
  2. package/bundles/core-testing.umd.js.map +1 -1
  3. package/bundles/core-testing.umd.min.js +6 -6
  4. package/bundles/core-testing.umd.min.js.map +1 -1
  5. package/bundles/core.umd.js +3619 -3914
  6. package/bundles/core.umd.js.map +1 -1
  7. package/bundles/core.umd.min.js +182 -175
  8. package/bundles/core.umd.min.js.map +1 -1
  9. package/core.d.ts +483 -651
  10. package/core.metadata.json +1 -1
  11. package/esm2015/core.js +17 -17
  12. package/esm2015/index.js +2 -2
  13. package/esm2015/public_api.js +2 -2
  14. package/esm2015/src/application_ref.js +7 -7
  15. package/esm2015/src/core.js +3 -3
  16. package/esm2015/src/core_private_export.js +7 -7
  17. package/esm2015/src/core_render3_private_export.js +2 -2
  18. package/esm2015/src/debug/debug_node.js +44 -12
  19. package/esm2015/src/i18n/locale_data_api.js +1 -2
  20. package/esm2015/src/interface/type.js +1 -1
  21. package/esm2015/src/metadata/ng_module.js +1 -1
  22. package/esm2015/src/render/api.js +4 -1
  23. package/esm2015/src/render3/assert.js +9 -1
  24. package/esm2015/src/render3/bindings.js +16 -5
  25. package/esm2015/src/render3/component.js +54 -25
  26. package/esm2015/src/render3/component_ref.js +11 -15
  27. package/esm2015/src/render3/definition.js +3 -1
  28. package/esm2015/src/render3/di.js +3 -4
  29. package/esm2015/src/render3/di_setup.js +5 -7
  30. package/esm2015/src/render3/errors.js +3 -1
  31. package/esm2015/src/render3/features/inherit_definition_feature.js +41 -13
  32. package/esm2015/src/render3/features/ng_onchanges_feature.js +2 -2
  33. package/esm2015/src/render3/global_utils_api.js +3 -3
  34. package/esm2015/src/render3/i18n.js +60 -56
  35. package/esm2015/src/render3/index.js +2 -2
  36. package/esm2015/src/render3/instructions/advance.js +10 -11
  37. package/esm2015/src/render3/instructions/all.js +4 -5
  38. package/esm2015/src/render3/instructions/attribute.js +6 -6
  39. package/esm2015/src/render3/instructions/attribute_interpolation.js +40 -21
  40. package/esm2015/src/render3/instructions/change_detection.js +8 -23
  41. package/esm2015/src/render3/instructions/class_map_interpolation.js +13 -12
  42. package/esm2015/src/render3/instructions/container.js +15 -12
  43. package/esm2015/src/render3/instructions/element.js +40 -127
  44. package/esm2015/src/render3/instructions/element_container.js +8 -10
  45. package/esm2015/src/render3/instructions/embedded_view.js +7 -7
  46. package/esm2015/src/render3/instructions/host_property.js +10 -7
  47. package/esm2015/src/render3/instructions/listener.js +18 -16
  48. package/esm2015/src/render3/instructions/lview_debug.js +160 -23
  49. package/esm2015/src/render3/instructions/projection.js +7 -5
  50. package/esm2015/src/render3/instructions/property.js +27 -6
  51. package/esm2015/src/render3/instructions/property_interpolation.js +40 -22
  52. package/esm2015/src/render3/instructions/shared.js +267 -240
  53. package/esm2015/src/render3/instructions/storage.js +6 -8
  54. package/esm2015/src/render3/instructions/style_prop_interpolation.js +12 -12
  55. package/esm2015/src/render3/instructions/styling.js +731 -475
  56. package/esm2015/src/render3/instructions/text.js +5 -5
  57. package/esm2015/src/render3/interfaces/definition.js +41 -1
  58. package/esm2015/src/render3/interfaces/node.js +160 -115
  59. package/esm2015/src/render3/interfaces/styling.js +183 -375
  60. package/esm2015/src/render3/interfaces/view.js +10 -2
  61. package/esm2015/src/render3/jit/environment.js +1 -3
  62. package/esm2015/src/render3/node_manipulation.js +177 -57
  63. package/esm2015/src/render3/node_selector_matcher.js +39 -24
  64. package/esm2015/src/render3/node_util.js +12 -7
  65. package/esm2015/src/render3/pipe.js +4 -6
  66. package/esm2015/src/render3/query.js +32 -26
  67. package/esm2015/src/render3/state.js +54 -183
  68. package/esm2015/src/render3/styling/class_differ.js +47 -0
  69. package/esm2015/src/render3/styling/static_styling.js +54 -0
  70. package/esm2015/src/render3/styling/style_binding_list.js +437 -0
  71. package/esm2015/src/render3/styling/styling_parser.js +336 -0
  72. package/esm2015/src/render3/tokens.js +2 -2
  73. package/esm2015/src/render3/util/attrs_utils.js +125 -2
  74. package/esm2015/src/render3/util/change_detection_utils.js +33 -0
  75. package/esm2015/src/render3/util/discovery_utils.js +146 -119
  76. package/esm2015/src/render3/util/global_utils.js +5 -5
  77. package/esm2015/src/render3/util/view_utils.js +6 -6
  78. package/esm2015/src/render3/view_engine_compatibility.js +16 -17
  79. package/esm2015/src/render3/view_ref.js +16 -13
  80. package/esm2015/src/sanitization/bypass.js +1 -1
  81. package/esm2015/src/sanitization/sanitization.js +20 -5
  82. package/esm2015/src/util/array_utils.js +240 -1
  83. package/esm2015/src/util/assert.js +37 -21
  84. package/esm2015/src/util/char_code.js +8 -0
  85. package/esm2015/src/util/iterable.js +4 -1
  86. package/esm2015/src/util/ng_dev_mode.js +1 -12
  87. package/esm2015/src/util/stringify.js +14 -1
  88. package/esm2015/src/version.js +1 -1
  89. package/esm2015/testing/src/r3_test_bed.js +5 -1
  90. package/esm2015/testing/src/r3_test_bed_compiler.js +2 -12
  91. package/esm2015/testing/src/styling.js +103 -0
  92. package/esm5/core.js +17 -17
  93. package/esm5/src/application_ref.js +6 -6
  94. package/esm5/src/core_private_export.js +7 -7
  95. package/esm5/src/core_render3_private_export.js +2 -2
  96. package/esm5/src/debug/debug_node.js +29 -11
  97. package/esm5/src/i18n/locale_data_api.js +1 -2
  98. package/esm5/src/interface/type.js +1 -1
  99. package/esm5/src/metadata/ng_module.js +1 -1
  100. package/esm5/src/render/api.js +4 -1
  101. package/esm5/src/render3/assert.js +4 -1
  102. package/esm5/src/render3/bindings.js +19 -2
  103. package/esm5/src/render3/component.js +47 -22
  104. package/esm5/src/render3/component_ref.js +9 -14
  105. package/esm5/src/render3/definition.js +3 -1
  106. package/esm5/src/render3/di.js +3 -4
  107. package/esm5/src/render3/di_setup.js +4 -5
  108. package/esm5/src/render3/errors.js +3 -1
  109. package/esm5/src/render3/features/inherit_definition_feature.js +36 -12
  110. package/esm5/src/render3/features/ng_onchanges_feature.js +1 -1
  111. package/esm5/src/render3/global_utils_api.js +3 -3
  112. package/esm5/src/render3/i18n.js +51 -51
  113. package/esm5/src/render3/index.js +2 -2
  114. package/esm5/src/render3/instructions/advance.js +9 -11
  115. package/esm5/src/render3/instructions/all.js +1 -2
  116. package/esm5/src/render3/instructions/attribute.js +5 -6
  117. package/esm5/src/render3/instructions/attribute_interpolation.js +31 -21
  118. package/esm5/src/render3/instructions/change_detection.js +8 -21
  119. package/esm5/src/render3/instructions/class_map_interpolation.js +13 -12
  120. package/esm5/src/render3/instructions/container.js +13 -12
  121. package/esm5/src/render3/instructions/element.js +36 -108
  122. package/esm5/src/render3/instructions/element_container.js +8 -9
  123. package/esm5/src/render3/instructions/embedded_view.js +7 -7
  124. package/esm5/src/render3/instructions/host_property.js +8 -7
  125. package/esm5/src/render3/instructions/listener.js +13 -13
  126. package/esm5/src/render3/instructions/lview_debug.js +56 -15
  127. package/esm5/src/render3/instructions/projection.js +6 -5
  128. package/esm5/src/render3/instructions/property.js +17 -6
  129. package/esm5/src/render3/instructions/property_interpolation.js +31 -23
  130. package/esm5/src/render3/instructions/shared.js +247 -207
  131. package/esm5/src/render3/instructions/storage.js +4 -6
  132. package/esm5/src/render3/instructions/style_prop_interpolation.js +12 -12
  133. package/esm5/src/render3/instructions/styling.js +685 -367
  134. package/esm5/src/render3/instructions/text.js +5 -5
  135. package/esm5/src/render3/interfaces/definition.js +1 -1
  136. package/esm5/src/render3/interfaces/node.js +49 -1
  137. package/esm5/src/render3/interfaces/styling.js +57 -1
  138. package/esm5/src/render3/interfaces/view.js +1 -1
  139. package/esm5/src/render3/jit/environment.js +1 -3
  140. package/esm5/src/render3/node_manipulation.js +167 -54
  141. package/esm5/src/render3/node_selector_matcher.js +40 -20
  142. package/esm5/src/render3/node_util.js +12 -7
  143. package/esm5/src/render3/pipe.js +4 -6
  144. package/esm5/src/render3/query.js +25 -24
  145. package/esm5/src/render3/state.js +34 -131
  146. package/esm5/src/render3/styling/class_differ.js +39 -0
  147. package/esm5/src/render3/styling/static_styling.js +42 -0
  148. package/esm5/src/render3/styling/style_binding_list.js +411 -0
  149. package/esm5/src/render3/styling/styling_parser.js +265 -0
  150. package/esm5/src/render3/tokens.js +2 -2
  151. package/esm5/src/render3/util/attrs_utils.js +117 -2
  152. package/esm5/src/render3/util/change_detection_utils.js +23 -0
  153. package/esm5/src/render3/util/discovery_utils.js +115 -99
  154. package/esm5/src/render3/util/global_utils.js +5 -5
  155. package/esm5/src/render3/util/view_utils.js +5 -5
  156. package/esm5/src/render3/view_engine_compatibility.js +37 -39
  157. package/esm5/src/render3/view_ref.js +14 -13
  158. package/esm5/src/sanitization/bypass.js +1 -1
  159. package/esm5/src/sanitization/sanitization.js +16 -5
  160. package/esm5/src/util/array_utils.js +240 -1
  161. package/esm5/src/util/assert.js +37 -21
  162. package/esm5/src/util/char_code.js +8 -0
  163. package/esm5/src/util/iterable.js +4 -1
  164. package/esm5/src/util/ng_dev_mode.js +1 -12
  165. package/esm5/src/util/stringify.js +14 -1
  166. package/esm5/src/version.js +1 -1
  167. package/esm5/testing/src/r3_test_bed.js +9 -1
  168. package/esm5/testing/src/r3_test_bed_compiler.js +2 -9
  169. package/esm5/testing/src/styling.js +82 -0
  170. package/fesm2015/core.js +5917 -6880
  171. package/fesm2015/core.js.map +1 -1
  172. package/fesm2015/testing.js +6 -12
  173. package/fesm2015/testing.js.map +1 -1
  174. package/fesm5/core.js +3588 -3884
  175. package/fesm5/core.js.map +1 -1
  176. package/fesm5/testing.js +10 -9
  177. package/fesm5/testing.js.map +1 -1
  178. package/package.json +1 -1
  179. package/src/r3_symbols.d.ts +1 -1
  180. package/testing/testing.d.ts +1 -3
  181. package/testing/testing.metadata.json +1 -1
  182. package/testing.d.ts +1 -1
  183. package/esm2015/global.js +0 -7
  184. package/esm2015/src/render3/instructions/alloc_host_vars.js +0 -80
  185. package/esm2015/src/render3/styling/bindings.js +0 -1248
  186. package/esm2015/src/render3/styling/map_based_bindings.js +0 -384
  187. package/esm2015/src/render3/styling/state.js +0 -135
  188. package/esm2015/src/render3/styling/styling_debug.js +0 -655
  189. package/esm2015/src/render3/util/styling_utils.js +0 -625
  190. package/esm5/global.js +0 -9
  191. package/esm5/src/render3/instructions/alloc_host_vars.js +0 -62
  192. package/esm5/src/render3/styling/bindings.js +0 -949
  193. package/esm5/src/render3/styling/map_based_bindings.js +0 -310
  194. package/esm5/src/render3/styling/state.js +0 -56
  195. package/esm5/src/render3/styling/styling_debug.js +0 -315
  196. package/esm5/src/render3/util/styling_utils.js +0 -378
@@ -0,0 +1,54 @@
1
+ /**
2
+ * @fileoverview added by tsickle
3
+ * Generated from: packages/core/src/render3/styling/static_styling.ts
4
+ * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
+ */
6
+ /**
7
+ * @license
8
+ * Copyright Google Inc. All Rights Reserved.
9
+ *
10
+ * Use of this source code is governed by an MIT-style license that can be
11
+ * found in the LICENSE file at https://angular.io/license
12
+ */
13
+ import { concatStringsWithSpace } from '../../util/stringify';
14
+ import { assertFirstCreatePass } from '../assert';
15
+ import { getTView } from '../state';
16
+ /**
17
+ * Compute the static styling (class/style) from `TAttributes`.
18
+ *
19
+ * This function should be called during `firstCreatePass` only.
20
+ *
21
+ * @param {?} tNode The `TNode` into which the styling information should be loaded.
22
+ * @param {?} attrs `TAttributes` containing the styling information.
23
+ * @return {?}
24
+ */
25
+ export function computeStaticStyling(tNode, attrs) {
26
+ ngDevMode &&
27
+ assertFirstCreatePass(getTView(), 'Expecting to be called in first template pass only');
28
+ /** @type {?} */
29
+ let styles = tNode.styles;
30
+ /** @type {?} */
31
+ let classes = tNode.classes;
32
+ /** @type {?} */
33
+ let mode = 0;
34
+ for (let i = 0; i < attrs.length; i++) {
35
+ /** @type {?} */
36
+ const value = attrs[i];
37
+ if (typeof value === 'number') {
38
+ mode = value;
39
+ }
40
+ else if (mode == 1 /* Classes */) {
41
+ classes = concatStringsWithSpace(classes, (/** @type {?} */ (value)));
42
+ }
43
+ else if (mode == 2 /* Styles */) {
44
+ /** @type {?} */
45
+ const style = (/** @type {?} */ (value));
46
+ /** @type {?} */
47
+ const styleValue = (/** @type {?} */ (attrs[++i]));
48
+ styles = concatStringsWithSpace(styles, style + ': ' + styleValue + ';');
49
+ }
50
+ }
51
+ styles !== null && (tNode.styles = styles);
52
+ classes !== null && (tNode.classes = classes);
53
+ }
54
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGljX3N0eWxpbmcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9jb3JlL3NyYy9yZW5kZXIzL3N0eWxpbmcvc3RhdGljX3N0eWxpbmcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBUUEsT0FBTyxFQUFDLHNCQUFzQixFQUFDLE1BQU0sc0JBQXNCLENBQUM7QUFDNUQsT0FBTyxFQUFDLHFCQUFxQixFQUFDLE1BQU0sV0FBVyxDQUFDO0FBRWhELE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxVQUFVLENBQUM7Ozs7Ozs7Ozs7QUFVbEMsTUFBTSxVQUFVLG9CQUFvQixDQUFDLEtBQVksRUFBRSxLQUFrQjtJQUNuRSxTQUFTO1FBQ0wscUJBQXFCLENBQUMsUUFBUSxFQUFFLEVBQUUsb0RBQW9ELENBQUMsQ0FBQzs7UUFDeEYsTUFBTSxHQUFnQixLQUFLLENBQUMsTUFBTTs7UUFDbEMsT0FBTyxHQUFnQixLQUFLLENBQUMsT0FBTzs7UUFDcEMsSUFBSSxHQUFzQixDQUFDO0lBQy9CLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxLQUFLLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFOztjQUMvQixLQUFLLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUN0QixJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsRUFBRTtZQUM3QixJQUFJLEdBQUcsS0FBSyxDQUFDO1NBQ2Q7YUFBTSxJQUFJLElBQUksbUJBQTJCLEVBQUU7WUFDMUMsT0FBTyxHQUFHLHNCQUFzQixDQUFDLE9BQU8sRUFBRSxtQkFBQSxLQUFLLEVBQVUsQ0FBQyxDQUFDO1NBQzVEO2FBQU0sSUFBSSxJQUFJLGtCQUEwQixFQUFFOztrQkFDbkMsS0FBSyxHQUFHLG1CQUFBLEtBQUssRUFBVTs7a0JBQ3ZCLFVBQVUsR0FBRyxtQkFBQSxLQUFLLENBQUMsRUFBRSxDQUFDLENBQUMsRUFBVTtZQUN2QyxNQUFNLEdBQUcsc0JBQXNCLENBQUMsTUFBTSxFQUFFLEtBQUssR0FBRyxJQUFJLEdBQUcsVUFBVSxHQUFHLEdBQUcsQ0FBQyxDQUFDO1NBQzFFO0tBQ0Y7SUFDRCxNQUFNLEtBQUssSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sR0FBRyxNQUFNLENBQUMsQ0FBQztJQUMzQyxPQUFPLEtBQUssSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUMsQ0FBQztBQUNoRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4qIEBsaWNlbnNlXG4qIENvcHlyaWdodCBHb29nbGUgSW5jLiBBbGwgUmlnaHRzIFJlc2VydmVkLlxuKlxuKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4qL1xuXG5pbXBvcnQge2NvbmNhdFN0cmluZ3NXaXRoU3BhY2V9IGZyb20gJy4uLy4uL3V0aWwvc3RyaW5naWZ5JztcbmltcG9ydCB7YXNzZXJ0Rmlyc3RDcmVhdGVQYXNzfSBmcm9tICcuLi9hc3NlcnQnO1xuaW1wb3J0IHtBdHRyaWJ1dGVNYXJrZXIsIFRBdHRyaWJ1dGVzLCBUTm9kZX0gZnJvbSAnLi4vaW50ZXJmYWNlcy9ub2RlJztcbmltcG9ydCB7Z2V0VFZpZXd9IGZyb20gJy4uL3N0YXRlJztcblxuLyoqXG4gKiBDb21wdXRlIHRoZSBzdGF0aWMgc3R5bGluZyAoY2xhc3Mvc3R5bGUpIGZyb20gYFRBdHRyaWJ1dGVzYC5cbiAqXG4gKiBUaGlzIGZ1bmN0aW9uIHNob3VsZCBiZSBjYWxsZWQgZHVyaW5nIGBmaXJzdENyZWF0ZVBhc3NgIG9ubHkuXG4gKlxuICogQHBhcmFtIHROb2RlIFRoZSBgVE5vZGVgIGludG8gd2hpY2ggdGhlIHN0eWxpbmcgaW5mb3JtYXRpb24gc2hvdWxkIGJlIGxvYWRlZC5cbiAqIEBwYXJhbSBhdHRycyBgVEF0dHJpYnV0ZXNgIGNvbnRhaW5pbmcgdGhlIHN0eWxpbmcgaW5mb3JtYXRpb24uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb21wdXRlU3RhdGljU3R5bGluZyh0Tm9kZTogVE5vZGUsIGF0dHJzOiBUQXR0cmlidXRlcyk6IHZvaWQge1xuICBuZ0Rldk1vZGUgJiZcbiAgICAgIGFzc2VydEZpcnN0Q3JlYXRlUGFzcyhnZXRUVmlldygpLCAnRXhwZWN0aW5nIHRvIGJlIGNhbGxlZCBpbiBmaXJzdCB0ZW1wbGF0ZSBwYXNzIG9ubHknKTtcbiAgbGV0IHN0eWxlczogc3RyaW5nfG51bGwgPSB0Tm9kZS5zdHlsZXM7XG4gIGxldCBjbGFzc2VzOiBzdHJpbmd8bnVsbCA9IHROb2RlLmNsYXNzZXM7XG4gIGxldCBtb2RlOiBBdHRyaWJ1dGVNYXJrZXJ8MCA9IDA7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgYXR0cnMubGVuZ3RoOyBpKyspIHtcbiAgICBjb25zdCB2YWx1ZSA9IGF0dHJzW2ldO1xuICAgIGlmICh0eXBlb2YgdmFsdWUgPT09ICdudW1iZXInKSB7XG4gICAgICBtb2RlID0gdmFsdWU7XG4gICAgfSBlbHNlIGlmIChtb2RlID09IEF0dHJpYnV0ZU1hcmtlci5DbGFzc2VzKSB7XG4gICAgICBjbGFzc2VzID0gY29uY2F0U3RyaW5nc1dpdGhTcGFjZShjbGFzc2VzLCB2YWx1ZSBhcyBzdHJpbmcpO1xuICAgIH0gZWxzZSBpZiAobW9kZSA9PSBBdHRyaWJ1dGVNYXJrZXIuU3R5bGVzKSB7XG4gICAgICBjb25zdCBzdHlsZSA9IHZhbHVlIGFzIHN0cmluZztcbiAgICAgIGNvbnN0IHN0eWxlVmFsdWUgPSBhdHRyc1srK2ldIGFzIHN0cmluZztcbiAgICAgIHN0eWxlcyA9IGNvbmNhdFN0cmluZ3NXaXRoU3BhY2Uoc3R5bGVzLCBzdHlsZSArICc6ICcgKyBzdHlsZVZhbHVlICsgJzsnKTtcbiAgICB9XG4gIH1cbiAgc3R5bGVzICE9PSBudWxsICYmICh0Tm9kZS5zdHlsZXMgPSBzdHlsZXMpO1xuICBjbGFzc2VzICE9PSBudWxsICYmICh0Tm9kZS5jbGFzc2VzID0gY2xhc3Nlcyk7XG59Il19
@@ -0,0 +1,437 @@
1
+ /**
2
+ * @fileoverview added by tsickle
3
+ * Generated from: packages/core/src/render3/styling/style_binding_list.ts
4
+ * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
+ */
6
+ /**
7
+ * @license
8
+ * Copyright Google Inc. All Rights Reserved.
9
+ *
10
+ * Use of this source code is governed by an MIT-style license that can be
11
+ * found in the LICENSE file at https://angular.io/license
12
+ */
13
+ import { keyValueArrayIndexOf } from '../../util/array_utils';
14
+ import { assertDataInRange, assertEqual, assertNotEqual } from '../../util/assert';
15
+ import { assertFirstUpdatePass } from '../assert';
16
+ import { getTStylingRangeNext, getTStylingRangePrev, setTStylingRangeNext, setTStylingRangeNextDuplicate, setTStylingRangePrev, setTStylingRangePrevDuplicate, toTStylingRange } from '../interfaces/styling';
17
+ import { getTView } from '../state';
18
+ /**
19
+ * NOTE: The word `styling` is used interchangeably as style or class styling.
20
+ *
21
+ * This file contains code to link styling instructions together so that they can be replayed in
22
+ * priority order. The file exists because Ivy styling instruction execution order does not match
23
+ * that of the priority order. The purpose of this code is to create a linked list so that the
24
+ * instructions can be traversed in priority order when computing the styles.
25
+ *
26
+ * Assume we are dealing with the following code:
27
+ * ```
28
+ * \@Component({
29
+ * template: `
30
+ * <my-cmp [style]=" {color: '#001'} "
31
+ * [style.color]=" #002 "
32
+ * dir-style-color-1
33
+ * dir-style-color-2> `
34
+ * })
35
+ * class ExampleComponent {
36
+ * static ngComp = ... {
37
+ * ...
38
+ * // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
39
+ * ɵɵstyleMap({color: '#001'});
40
+ * ɵɵstyleProp('color', '#002');
41
+ * ...
42
+ * }
43
+ * }
44
+ *
45
+ * \@Directive({
46
+ * selector: `[dir-style-color-1]',
47
+ * })
48
+ * class Style1Directive {
49
+ * \@HostBinding('style') style = {color: '#005'};
50
+ * \@HostBinding('style.color') color = '#006';
51
+ *
52
+ * static ngDir = ... {
53
+ * ...
54
+ * // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
55
+ * ɵɵstyleMap({color: '#005'});
56
+ * ɵɵstyleProp('color', '#006');
57
+ * ...
58
+ * }
59
+ * }
60
+ *
61
+ * \@Directive({
62
+ * selector: `[dir-style-color-2]',
63
+ * })
64
+ * class Style2Directive {
65
+ * \@HostBinding('style') style = {color: '#007'};
66
+ * \@HostBinding('style.color') color = '#008';
67
+ *
68
+ * static ngDir = ... {
69
+ * ...
70
+ * // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
71
+ * ɵɵstyleMap({color: '#007'});
72
+ * ɵɵstyleProp('color', '#008');
73
+ * ...
74
+ * }
75
+ * }
76
+ *
77
+ * \@Directive({
78
+ * selector: `my-cmp',
79
+ * })
80
+ * class MyComponent {
81
+ * \@HostBinding('style') style = {color: '#003'};
82
+ * \@HostBinding('style.color') color = '#004';
83
+ *
84
+ * static ngComp = ... {
85
+ * ...
86
+ * // Compiler ensures that `ɵɵstyleProp` is after `ɵɵstyleMap`
87
+ * ɵɵstyleMap({color: '#003'});
88
+ * ɵɵstyleProp('color', '#004');
89
+ * ...
90
+ * }
91
+ * }
92
+ * ```
93
+ *
94
+ * The Order of instruction execution is:
95
+ *
96
+ * NOTE: the comment binding location is for illustrative purposes only.
97
+ *
98
+ * ```
99
+ * // Template: (ExampleComponent)
100
+ * ɵɵstyleMap({color: '#001'}); // Binding index: 10
101
+ * ɵɵstyleProp('color', '#002'); // Binding index: 12
102
+ * // MyComponent
103
+ * ɵɵstyleMap({color: '#003'}); // Binding index: 20
104
+ * ɵɵstyleProp('color', '#004'); // Binding index: 22
105
+ * // Style1Directive
106
+ * ɵɵstyleMap({color: '#005'}); // Binding index: 24
107
+ * ɵɵstyleProp('color', '#006'); // Binding index: 26
108
+ * // Style2Directive
109
+ * ɵɵstyleMap({color: '#007'}); // Binding index: 28
110
+ * ɵɵstyleProp('color', '#008'); // Binding index: 30
111
+ * ```
112
+ *
113
+ * The correct priority order of concatenation is:
114
+ *
115
+ * ```
116
+ * // MyComponent
117
+ * ɵɵstyleMap({color: '#003'}); // Binding index: 20
118
+ * ɵɵstyleProp('color', '#004'); // Binding index: 22
119
+ * // Style1Directive
120
+ * ɵɵstyleMap({color: '#005'}); // Binding index: 24
121
+ * ɵɵstyleProp('color', '#006'); // Binding index: 26
122
+ * // Style2Directive
123
+ * ɵɵstyleMap({color: '#007'}); // Binding index: 28
124
+ * ɵɵstyleProp('color', '#008'); // Binding index: 30
125
+ * // Template: (ExampleComponent)
126
+ * ɵɵstyleMap({color: '#001'}); // Binding index: 10
127
+ * ɵɵstyleProp('color', '#002'); // Binding index: 12
128
+ * ```
129
+ *
130
+ * What color should be rendered?
131
+ *
132
+ * Once the items are correctly sorted in the list, the answer is simply the last item in the
133
+ * concatenation list which is `#002`.
134
+ *
135
+ * To do so we keep a linked list of all of the bindings which pertain to this element.
136
+ * Notice that the bindings are inserted in the order of execution, but the `TView.data` allows
137
+ * us to traverse them in the order of priority.
138
+ *
139
+ * |Idx|`TView.data`|`LView` | Notes
140
+ * |---|------------|-----------------|--------------
141
+ * |...| | |
142
+ * |10 |`null` |`{color: '#001'}`| `ɵɵstyleMap('color', {color: '#001'})`
143
+ * |11 |`30 | 12` | ... |
144
+ * |12 |`color` |`'#002'` | `ɵɵstyleProp('color', '#002')`
145
+ * |13 |`10 | 0` | ... |
146
+ * |...| | |
147
+ * |20 |`null` |`{color: '#003'}`| `ɵɵstyleMap('color', {color: '#003'})`
148
+ * |21 |`0 | 22` | ... |
149
+ * |22 |`color` |`'#004'` | `ɵɵstyleProp('color', '#004')`
150
+ * |23 |`20 | 24` | ... |
151
+ * |24 |`null` |`{color: '#005'}`| `ɵɵstyleMap('color', {color: '#005'})`
152
+ * |25 |`22 | 26` | ... |
153
+ * |26 |`color` |`'#006'` | `ɵɵstyleProp('color', '#006')`
154
+ * |27 |`24 | 28` | ... |
155
+ * |28 |`null` |`{color: '#007'}`| `ɵɵstyleMap('color', {color: '#007'})`
156
+ * |29 |`26 | 30` | ... |
157
+ * |30 |`color` |`'#008'` | `ɵɵstyleProp('color', '#008')`
158
+ * |31 |`28 | 10` | ... |
159
+ *
160
+ * The above data structure allows us to re-concatenate the styling no matter which data binding
161
+ * changes.
162
+ *
163
+ * NOTE: in addition to keeping track of next/previous index the `TView.data` also stores prev/next
164
+ * duplicate bit. The duplicate bit if true says there either is a binding with the same name or
165
+ * there is a map (which may contain the name). This information is useful in knowing if other
166
+ * styles with higher priority need to be searched for overwrites.
167
+ *
168
+ * NOTE: See `should support example in 'tnode_linked_list.ts' documentation` in
169
+ * `tnode_linked_list_spec.ts` for working example.
170
+ * @type {?}
171
+ */
172
+ let __unused_const_as_closure_does_not_like_standalone_comment_blocks__;
173
+ /**
174
+ * Insert new `tStyleValue` at `TData` and link existing style bindings such that we maintain linked
175
+ * list of styles and compute the duplicate flag.
176
+ *
177
+ * Note: this function is executed during `firstUpdatePass` only to populate the `TView.data`.
178
+ *
179
+ * The function works by keeping track of `tStylingRange` which contains two pointers pointing to
180
+ * the head/tail of the template portion of the styles.
181
+ * - if `isHost === false` (we are template) then insertion is at tail of `TStylingRange`
182
+ * - if `isHost === true` (we are host binding) then insertion is at head of `TStylingRange`
183
+ *
184
+ * @param {?} tData The `TData` to insert into.
185
+ * @param {?} tNode `TNode` associated with the styling element.
186
+ * @param {?} tStylingKeyWithStatic
187
+ * @param {?} index location of where `tStyleValue` should be stored (and linked into list.)
188
+ * @param {?} isHostBinding `true` if the insertion is for a `hostBinding`. (insertion is in front of
189
+ * template.)
190
+ * @param {?} isClassBinding True if the associated `tStylingKey` as a `class` styling.
191
+ * `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
192
+ * @return {?}
193
+ */
194
+ export function insertTStylingBinding(tData, tNode, tStylingKeyWithStatic, index, isHostBinding, isClassBinding) {
195
+ ngDevMode && assertFirstUpdatePass(getTView());
196
+ /** @type {?} */
197
+ let tBindings = isClassBinding ? tNode.classBindings : tNode.styleBindings;
198
+ /** @type {?} */
199
+ let tmplHead = getTStylingRangePrev(tBindings);
200
+ /** @type {?} */
201
+ let tmplTail = getTStylingRangeNext(tBindings);
202
+ tData[index] = tStylingKeyWithStatic;
203
+ /** @type {?} */
204
+ let isKeyDuplicateOfStatic = false;
205
+ /** @type {?} */
206
+ let tStylingKey;
207
+ if (Array.isArray(tStylingKeyWithStatic)) {
208
+ // We are case when the `TStylingKey` contains static fields as well.
209
+ /** @type {?} */
210
+ const staticKeyValueArray = (/** @type {?} */ (tStylingKeyWithStatic));
211
+ tStylingKey = staticKeyValueArray[1]; // unwrap.
212
+ // We need to check if our key is present in the static so that we can mark it as duplicate.
213
+ if (tStylingKey === null ||
214
+ keyValueArrayIndexOf(staticKeyValueArray, (/** @type {?} */ (tStylingKey))) > 0) {
215
+ // tStylingKey is present in the statics, need to mark it as duplicate.
216
+ isKeyDuplicateOfStatic = true;
217
+ }
218
+ }
219
+ else {
220
+ tStylingKey = tStylingKeyWithStatic;
221
+ }
222
+ if (isHostBinding) {
223
+ // We are inserting host bindings
224
+ // If we don't have template bindings then `tail` is 0.
225
+ /** @type {?} */
226
+ const hasTemplateBindings = tmplTail !== 0;
227
+ // This is important to know because that means that the `head` can't point to the first
228
+ // template bindings (there are none.) Instead the head points to the tail of the template.
229
+ if (hasTemplateBindings) {
230
+ // template head's "prev" will point to last host binding or to 0 if no host bindings yet
231
+ /** @type {?} */
232
+ const previousNode = getTStylingRangePrev((/** @type {?} */ (tData[tmplHead + 1])));
233
+ tData[index + 1] = toTStylingRange(previousNode, tmplHead);
234
+ // if a host binding has already been registered, we need to update the next of that host
235
+ // binding to point to this one
236
+ if (previousNode !== 0) {
237
+ // We need to update the template-tail value to point to us.
238
+ tData[previousNode + 1] =
239
+ setTStylingRangeNext((/** @type {?} */ (tData[previousNode + 1])), index);
240
+ }
241
+ // The "previous" of the template binding head should point to this host binding
242
+ tData[tmplHead + 1] = setTStylingRangePrev((/** @type {?} */ (tData[tmplHead + 1])), index);
243
+ }
244
+ else {
245
+ tData[index + 1] = toTStylingRange(tmplHead, 0);
246
+ // if a host binding has already been registered, we need to update the next of that host
247
+ // binding to point to this one
248
+ if (tmplHead !== 0) {
249
+ // We need to update the template-tail value to point to us.
250
+ tData[tmplHead + 1] = setTStylingRangeNext((/** @type {?} */ (tData[tmplHead + 1])), index);
251
+ }
252
+ // if we don't have template, the head points to template-tail, and needs to be advanced.
253
+ tmplHead = index;
254
+ }
255
+ }
256
+ else {
257
+ // We are inserting in template section.
258
+ // We need to set this binding's "previous" to the current template tail
259
+ tData[index + 1] = toTStylingRange(tmplTail, 0);
260
+ ngDevMode && assertEqual(tmplHead !== 0 && tmplTail === 0, false, 'Adding template bindings after hostBindings is not allowed.');
261
+ if (tmplHead === 0) {
262
+ tmplHead = index;
263
+ }
264
+ else {
265
+ // We need to update the previous value "next" to point to this binding
266
+ tData[tmplTail + 1] = setTStylingRangeNext((/** @type {?} */ (tData[tmplTail + 1])), index);
267
+ }
268
+ tmplTail = index;
269
+ }
270
+ // Now we need to update / compute the duplicates.
271
+ // Starting with our location search towards head (least priority)
272
+ if (isKeyDuplicateOfStatic) {
273
+ tData[index + 1] = setTStylingRangePrevDuplicate((/** @type {?} */ (tData[index + 1])));
274
+ }
275
+ markDuplicates(tData, tStylingKey, index, true, isClassBinding);
276
+ markDuplicates(tData, tStylingKey, index, false, isClassBinding);
277
+ markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding);
278
+ tBindings = toTStylingRange(tmplHead, tmplTail);
279
+ if (isClassBinding) {
280
+ tNode.classBindings = tBindings;
281
+ }
282
+ else {
283
+ tNode.styleBindings = tBindings;
284
+ }
285
+ }
286
+ /**
287
+ * Look into the residual styling to see if the current `tStylingKey` is duplicate of residual.
288
+ *
289
+ * @param {?} tNode `TNode` where the residual is stored.
290
+ * @param {?} tStylingKey `TStylingKey` to store.
291
+ * @param {?} tData `TData` associated with the current `LView`.
292
+ * @param {?} index location of where `tStyleValue` should be stored (and linked into list.)
293
+ * @param {?} isClassBinding True if the associated `tStylingKey` as a `class` styling.
294
+ * `tNode.classBindings` should be used (or `tNode.styleBindings` otherwise.)
295
+ * @return {?}
296
+ */
297
+ function markDuplicateOfResidualStyling(tNode, tStylingKey, tData, index, isClassBinding) {
298
+ /** @type {?} */
299
+ const residual = isClassBinding ? tNode.residualClasses : tNode.residualStyles;
300
+ if (residual != null /* or undefined */ && typeof tStylingKey == 'string' &&
301
+ keyValueArrayIndexOf(residual, tStylingKey) >= 0) {
302
+ // We have duplicate in the residual so mark ourselves as duplicate.
303
+ tData[index + 1] = setTStylingRangeNextDuplicate((/** @type {?} */ (tData[index + 1])));
304
+ }
305
+ }
306
+ /**
307
+ * Marks `TStyleValue`s as duplicates if another style binding in the list has the same
308
+ * `TStyleValue`.
309
+ *
310
+ * NOTE: this function is intended to be called twice once with `isPrevDir` set to `true` and once
311
+ * with it set to `false` to search both the previous as well as next items in the list.
312
+ *
313
+ * No duplicate case
314
+ * ```
315
+ * [style.color]
316
+ * [style.width.px] <<- index
317
+ * [style.height.px]
318
+ * ```
319
+ *
320
+ * In the above case adding `[style.width.px]` to the existing `[style.color]` produces no
321
+ * duplicates because `width` is not found in any other part of the linked list.
322
+ *
323
+ * Duplicate case
324
+ * ```
325
+ * [style.color]
326
+ * [style.width.em]
327
+ * [style.width.px] <<- index
328
+ * ```
329
+ * In the above case adding `[style.width.px]` will produce a duplicate with `[style.width.em]`
330
+ * because `width` is found in the chain.
331
+ *
332
+ * Map case 1
333
+ * ```
334
+ * [style.width.px]
335
+ * [style.color]
336
+ * [style] <<- index
337
+ * ```
338
+ * In the above case adding `[style]` will produce a duplicate with any other bindings because
339
+ * `[style]` is a Map and as such is fully dynamic and could produce `color` or `width`.
340
+ *
341
+ * Map case 2
342
+ * ```
343
+ * [style]
344
+ * [style.width.px]
345
+ * [style.color] <<- index
346
+ * ```
347
+ * In the above case adding `[style.color]` will produce a duplicate because there is already a
348
+ * `[style]` binding which is a Map and as such is fully dynamic and could produce `color` or
349
+ * `width`.
350
+ *
351
+ * NOTE: Once `[style]` (Map) is added into the system all things are mapped as duplicates.
352
+ * NOTE: We use `style` as example, but same logic is applied to `class`es as well.
353
+ *
354
+ * @param {?} tData `TData` where the linked list is stored.
355
+ * @param {?} tStylingKey `TStylingKeyPrimitive` which contains the value to compare to other keys in
356
+ * the linked list.
357
+ * @param {?} index Starting location in the linked list to search from
358
+ * @param {?} isPrevDir Direction.
359
+ * - `true` for previous (lower priority);
360
+ * - `false` for next (higher priority).
361
+ * @param {?} isClassBinding
362
+ * @return {?}
363
+ */
364
+ function markDuplicates(tData, tStylingKey, index, isPrevDir, isClassBinding) {
365
+ /** @type {?} */
366
+ const tStylingAtIndex = (/** @type {?} */ (tData[index + 1]));
367
+ /** @type {?} */
368
+ const isMap = tStylingKey === null;
369
+ /** @type {?} */
370
+ let cursor = isPrevDir ? getTStylingRangePrev(tStylingAtIndex) : getTStylingRangeNext(tStylingAtIndex);
371
+ /** @type {?} */
372
+ let foundDuplicate = false;
373
+ // We keep iterating as long as we have a cursor
374
+ // AND either:
375
+ // - we found what we are looking for, OR
376
+ // - we are a map in which case we have to continue searching even after we find what we were
377
+ // looking for since we are a wild card and everything needs to be flipped to duplicate.
378
+ while (cursor !== 0 && (foundDuplicate === false || isMap)) {
379
+ ngDevMode && assertDataInRange(tData, cursor);
380
+ /** @type {?} */
381
+ const tStylingValueAtCursor = (/** @type {?} */ (tData[cursor]));
382
+ /** @type {?} */
383
+ const tStyleRangeAtCursor = (/** @type {?} */ (tData[cursor + 1]));
384
+ if (isStylingMatch(tStylingValueAtCursor, tStylingKey)) {
385
+ foundDuplicate = true;
386
+ tData[cursor + 1] = isPrevDir ? setTStylingRangeNextDuplicate(tStyleRangeAtCursor) :
387
+ setTStylingRangePrevDuplicate(tStyleRangeAtCursor);
388
+ }
389
+ cursor = isPrevDir ? getTStylingRangePrev(tStyleRangeAtCursor) :
390
+ getTStylingRangeNext(tStyleRangeAtCursor);
391
+ }
392
+ if (foundDuplicate) {
393
+ // if we found a duplicate, than mark ourselves.
394
+ tData[index + 1] = isPrevDir ? setTStylingRangePrevDuplicate(tStylingAtIndex) :
395
+ setTStylingRangeNextDuplicate(tStylingAtIndex);
396
+ }
397
+ }
398
+ /**
399
+ * Determines if two `TStylingKey`s are a match.
400
+ *
401
+ * When computing weather a binding contains a duplicate, we need to compare if the instruction
402
+ * `TStylingKey` has a match.
403
+ *
404
+ * Here are examples of `TStylingKey`s which match given `tStylingKeyCursor` is:
405
+ * - `color`
406
+ * - `color` // Match another color
407
+ * - `null` // That means that `tStylingKey` is a `classMap`/`styleMap` instruction
408
+ * - `['', 'color', 'other', true]` // wrapped `color` so match
409
+ * - `['', null, 'other', true]` // wrapped `null` so match
410
+ * - `['', 'width', 'color', 'value']` // wrapped static value contains a match on `'color'`
411
+ * - `null` // `tStylingKeyCursor` always match as it is `classMap`/`styleMap` instruction
412
+ *
413
+ * @param {?} tStylingKeyCursor
414
+ * @param {?} tStylingKey
415
+ * @return {?}
416
+ */
417
+ function isStylingMatch(tStylingKeyCursor, tStylingKey) {
418
+ ngDevMode &&
419
+ assertNotEqual(Array.isArray(tStylingKey), true, 'Expected that \'tStylingKey\' has been unwrapped');
420
+ if (tStylingKeyCursor === null || // If the cursor is `null` it means that we have map at that
421
+ // location so we must assume that we have a match.
422
+ tStylingKey == null || // If `tStylingKey` is `null` then it is a map therefor assume that it
423
+ // contains a match.
424
+ (Array.isArray(tStylingKeyCursor) ? tStylingKeyCursor[1] : tStylingKeyCursor) ===
425
+ tStylingKey // If the keys match explicitly than we are a match.
426
+ ) {
427
+ return true;
428
+ }
429
+ else if (Array.isArray(tStylingKeyCursor) && typeof tStylingKey === 'string') {
430
+ // if we did not find a match, but `tStylingKeyCursor` is `KeyValueArray` that means cursor has
431
+ // statics and we need to check those as well.
432
+ return keyValueArrayIndexOf(tStylingKeyCursor, tStylingKey) >=
433
+ 0; // see if we are matching the key
434
+ }
435
+ return false;
436
+ }
437
+ //# 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":";;;;;;;;;;;;AAQA,OAAO,EAAgB,oBAAoB,EAAC,MAAM,wBAAwB,CAAC;AAC3E,OAAO,EAAC,iBAAiB,EAAE,WAAW,EAAE,cAAc,EAAC,MAAM,mBAAmB,CAAC;AACjF,OAAO,EAAC,qBAAqB,EAAC,MAAM,WAAW,CAAC;AAEhD,OAAO,EAAmD,oBAAoB,EAAE,oBAAoB,EAAE,oBAAoB,EAAE,6BAA6B,EAAE,oBAAoB,EAAE,6BAA6B,EAAE,eAAe,EAAC,MAAM,uBAAuB,CAAC;AAE9P,OAAO,EAAC,QAAQ,EAAC,MAAM,UAAU,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IA4J9B,mEAA8E;;;;;;;;;;;;;;;;;;;;;;AAsBlF,MAAM,UAAU,qBAAqB,CACjC,KAAY,EAAE,KAAY,EAAE,qBAAkC,EAAE,KAAa,EAC7E,aAAsB,EAAE,cAAuB;IACjD,SAAS,IAAI,qBAAqB,CAAC,QAAQ,EAAE,CAAC,CAAC;;QAC3C,SAAS,GAAG,cAAc,CAAC,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,KAAK,CAAC,aAAa;;QACtE,QAAQ,GAAG,oBAAoB,CAAC,SAAS,CAAC;;QAC1C,QAAQ,GAAG,oBAAoB,CAAC,SAAS,CAAC;IAE9C,KAAK,CAAC,KAAK,CAAC,GAAG,qBAAqB,CAAC;;QACjC,sBAAsB,GAAG,KAAK;;QAC9B,WAAiC;IACrC,IAAI,KAAK,CAAC,OAAO,CAAC,qBAAqB,CAAC,EAAE;;;cAElC,mBAAmB,GAAG,mBAAA,qBAAqB,EAAsB;QACvE,WAAW,GAAG,mBAAmB,CAAC,CAAC,CAAC,CAAC,CAAE,UAAU;QACjD,4FAA4F;QAC5F,IAAI,WAAW,KAAK,IAAI;YACpB,oBAAoB,CAAC,mBAAmB,EAAE,mBAAA,WAAW,EAAU,CAAC,GAAG,CAAC,EAAE;YACxE,uEAAuE;YACvE,sBAAsB,GAAG,IAAI,CAAC;SAC/B;KACF;SAAM;QACL,WAAW,GAAG,qBAAqB,CAAC;KACrC;IACD,IAAI,aAAa,EAAE;;;;cAIX,mBAAmB,GAAG,QAAQ,KAAK,CAAC;QAC1C,wFAAwF;QACxF,2FAA2F;QAC3F,IAAI,mBAAmB,EAAE;;;kBAEjB,YAAY,GAAG,oBAAoB,CAAC,mBAAA,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,EAAiB,CAAC;YAC/E,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,eAAe,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;YAC3D,yFAAyF;YACzF,+BAA+B;YAC/B,IAAI,YAAY,KAAK,CAAC,EAAE;gBACtB,4DAA4D;gBAC5D,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC;oBACnB,oBAAoB,CAAC,mBAAA,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,EAAiB,EAAE,KAAK,CAAC,CAAC;aAC3E;YACD,gFAAgF;YAChF,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,GAAG,oBAAoB,CAAC,mBAAA,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,EAAiB,EAAE,KAAK,CAAC,CAAC;SACzF;aAAM;YACL,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,eAAe,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;YAChD,yFAAyF;YACzF,+BAA+B;YAC/B,IAAI,QAAQ,KAAK,CAAC,EAAE;gBAClB,4DAA4D;gBAC5D,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,GAAG,oBAAoB,CAAC,mBAAA,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,EAAiB,EAAE,KAAK,CAAC,CAAC;aACzF;YACD,yFAAyF;YACzF,QAAQ,GAAG,KAAK,CAAC;SAClB;KACF;SAAM;QACL,wCAAwC;QACxC,wEAAwE;QACxE,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,eAAe,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;QAChD,SAAS,IAAI,WAAW,CACP,QAAQ,KAAK,CAAC,IAAI,QAAQ,KAAK,CAAC,EAAE,KAAK,EACvC,6DAA6D,CAAC,CAAC;QAChF,IAAI,QAAQ,KAAK,CAAC,EAAE;YAClB,QAAQ,GAAG,KAAK,CAAC;SAClB;aAAM;YACL,uEAAuE;YACvE,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,GAAG,oBAAoB,CAAC,mBAAA,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,EAAiB,EAAE,KAAK,CAAC,CAAC;SACzF;QACD,QAAQ,GAAG,KAAK,CAAC;KAClB;IAED,kDAAkD;IAClD,kEAAkE;IAClE,IAAI,sBAAsB,EAAE;QAC1B,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,6BAA6B,CAAC,mBAAA,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,EAAiB,CAAC,CAAC;KACrF;IACD,cAAc,CAAC,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC;IAChE,cAAc,CAAC,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC;IACjE,8BAA8B,CAAC,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC;IAEjF,SAAS,GAAG,eAAe,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAChD,IAAI,cAAc,EAAE;QAClB,KAAK,CAAC,aAAa,GAAG,SAAS,CAAC;KACjC;SAAM;QACL,KAAK,CAAC,aAAa,GAAG,SAAS,CAAC;KACjC;AACH,CAAC;;;;;;;;;;;;AAYD,SAAS,8BAA8B,CACnC,KAAY,EAAE,WAAwB,EAAE,KAAY,EAAE,KAAa,EAAE,cAAuB;;UACxF,QAAQ,GAAG,cAAc,CAAC,CAAC,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC,KAAK,CAAC,cAAc;IAC9E,IAAI,QAAQ,IAAI,IAAI,CAAC,kBAAkB,IAAI,OAAO,WAAW,IAAI,QAAQ;QACrE,oBAAoB,CAAC,QAAQ,EAAE,WAAW,CAAC,IAAI,CAAC,EAAE;QACpD,oEAAoE;QACpE,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,6BAA6B,CAAC,mBAAA,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,EAAiB,CAAC,CAAC;KACrF;AACH,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA2DD,SAAS,cAAc,CACnB,KAAY,EAAE,WAAiC,EAAE,KAAa,EAAE,SAAkB,EAClF,cAAuB;;UACnB,eAAe,GAAG,mBAAA,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,EAAiB;;UACnD,KAAK,GAAG,WAAW,KAAK,IAAI;;QAC9B,MAAM,GACN,SAAS,CAAC,CAAC,CAAC,oBAAoB,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,eAAe,CAAC;;QACzF,cAAc,GAAG,KAAK;IAC1B,gDAAgD;IAChD,cAAc;IACd,yCAAyC;IACzC,6FAA6F;IAC7F,0FAA0F;IAC1F,OAAO,MAAM,KAAK,CAAC,IAAI,CAAC,cAAc,KAAK,KAAK,IAAI,KAAK,CAAC,EAAE;QAC1D,SAAS,IAAI,iBAAiB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;;cACxC,qBAAqB,GAAG,mBAAA,KAAK,CAAC,MAAM,CAAC,EAAe;;cACpD,mBAAmB,GAAG,mBAAA,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,EAAiB;QAC9D,IAAI,cAAc,CAAC,qBAAqB,EAAE,WAAW,CAAC,EAAE;YACtD,cAAc,GAAG,IAAI,CAAC;YACtB,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,6BAA6B,CAAC,mBAAmB,CAAC,CAAC,CAAC;gBACpD,6BAA6B,CAAC,mBAAmB,CAAC,CAAC;SACpF;QACD,MAAM,GAAG,SAAS,CAAC,CAAC,CAAC,oBAAoB,CAAC,mBAAmB,CAAC,CAAC,CAAC;YAC3C,oBAAoB,CAAC,mBAAmB,CAAC,CAAC;KAChE;IACD,IAAI,cAAc,EAAE;QAClB,gDAAgD;QAChD,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,6BAA6B,CAAC,eAAe,CAAC,CAAC,CAAC;YAChD,6BAA6B,CAAC,eAAe,CAAC,CAAC;KAC/E;AACH,CAAC;;;;;;;;;;;;;;;;;;;;AAoBD,SAAS,cAAc,CAAC,iBAA8B,EAAE,WAAiC;IACvF,SAAS;QACL,cAAc,CACV,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,IAAI,EAAE,kDAAkD,CAAC,CAAC;IAC9F,IAAI,iBAAiB,KAAK,IAAI,IAAK,4DAA4D;QAC5D,mDAAmD;QAClF,WAAW,IAAI,IAAI,IAAK,sEAAsE;QACtE,oBAAoB;QAC5C,CAAC,KAAK,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;YACzE,WAAW,CAAE,oDAAoD;MACnE;QACJ,OAAO,IAAI,CAAC;KACb;SAAM,IAAI,KAAK,CAAC,OAAO,CAAC,iBAAiB,CAAC,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE;QAC9E,+FAA+F;QAC/F,8CAA8C;QAC9C,OAAO,oBAAoB,CAAC,iBAAiB,EAAE,WAAW,CAAC;YACvD,CAAC,CAAC,CAAE,iCAAiC;KAC1C;IACD,OAAO,KAAK,CAAC;AACf,CAAC","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"]}