@angular/compiler 19.0.0-next.1 → 19.0.0-next.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (186) hide show
  1. package/LICENSE +21 -0
  2. package/fesm2022/compiler.mjs +5465 -4675
  3. package/fesm2022/compiler.mjs.map +1 -1
  4. package/index.d.ts +115 -15
  5. package/package.json +2 -4
  6. package/esm2022/compiler.mjs +0 -13
  7. package/esm2022/index.mjs +0 -13
  8. package/esm2022/public_api.mjs +0 -15
  9. package/esm2022/src/assertions.mjs +0 -31
  10. package/esm2022/src/chars.mjs +0 -90
  11. package/esm2022/src/compiler.mjs +0 -88
  12. package/esm2022/src/compiler_facade_interface.mjs +0 -29
  13. package/esm2022/src/config.mjs +0 -20
  14. package/esm2022/src/constant_pool.mjs +0 -285
  15. package/esm2022/src/core.mjs +0 -96
  16. package/esm2022/src/expression_parser/ast.mjs +0 -755
  17. package/esm2022/src/expression_parser/lexer.mjs +0 -421
  18. package/esm2022/src/expression_parser/parser.mjs +0 -1222
  19. package/esm2022/src/i18n/digest.mjs +0 -322
  20. package/esm2022/src/i18n/extractor_merger.mjs +0 -409
  21. package/esm2022/src/i18n/i18n_ast.mjs +0 -212
  22. package/esm2022/src/i18n/i18n_html_parser.mjs +0 -54
  23. package/esm2022/src/i18n/i18n_parser.mjs +0 -247
  24. package/esm2022/src/i18n/index.mjs +0 -16
  25. package/esm2022/src/i18n/message_bundle.mjs +0 -93
  26. package/esm2022/src/i18n/parse_util.mjs +0 -17
  27. package/esm2022/src/i18n/serializers/placeholder.mjs +0 -139
  28. package/esm2022/src/i18n/serializers/serializer.mjs +0 -77
  29. package/esm2022/src/i18n/serializers/xliff.mjs +0 -300
  30. package/esm2022/src/i18n/serializers/xliff2.mjs +0 -349
  31. package/esm2022/src/i18n/serializers/xmb.mjs +0 -188
  32. package/esm2022/src/i18n/serializers/xml_helper.mjs +0 -94
  33. package/esm2022/src/i18n/serializers/xtb.mjs +0 -186
  34. package/esm2022/src/i18n/translation_bundle.mjs +0 -164
  35. package/esm2022/src/injectable_compiler_2.mjs +0 -138
  36. package/esm2022/src/jit_compiler_facade.mjs +0 -681
  37. package/esm2022/src/ml_parser/ast.mjs +0 -167
  38. package/esm2022/src/ml_parser/defaults.mjs +0 -24
  39. package/esm2022/src/ml_parser/entities.mjs +0 -2144
  40. package/esm2022/src/ml_parser/html_parser.mjs +0 -18
  41. package/esm2022/src/ml_parser/html_tags.mjs +0 -156
  42. package/esm2022/src/ml_parser/html_whitespaces.mjs +0 -107
  43. package/esm2022/src/ml_parser/icu_ast_expander.mjs +0 -125
  44. package/esm2022/src/ml_parser/lexer.mjs +0 -1210
  45. package/esm2022/src/ml_parser/parser.mjs +0 -539
  46. package/esm2022/src/ml_parser/tags.mjs +0 -47
  47. package/esm2022/src/ml_parser/tokens.mjs +0 -9
  48. package/esm2022/src/ml_parser/xml_parser.mjs +0 -19
  49. package/esm2022/src/ml_parser/xml_tags.mjs +0 -32
  50. package/esm2022/src/output/abstract_emitter.mjs +0 -524
  51. package/esm2022/src/output/abstract_js_emitter.mjs +0 -130
  52. package/esm2022/src/output/map_util.mjs +0 -19
  53. package/esm2022/src/output/output_ast.mjs +0 -1301
  54. package/esm2022/src/output/output_jit.mjs +0 -138
  55. package/esm2022/src/output/output_jit_trusted_types.mjs +0 -100
  56. package/esm2022/src/output/source_map.mjs +0 -150
  57. package/esm2022/src/parse_util.mjs +0 -198
  58. package/esm2022/src/render3/partial/api.mjs +0 -9
  59. package/esm2022/src/render3/partial/class_metadata.mjs +0 -52
  60. package/esm2022/src/render3/partial/component.mjs +0 -197
  61. package/esm2022/src/render3/partial/directive.mjs +0 -264
  62. package/esm2022/src/render3/partial/factory.mjs +0 -35
  63. package/esm2022/src/render3/partial/injectable.mjs +0 -67
  64. package/esm2022/src/render3/partial/injector.mjs +0 -41
  65. package/esm2022/src/render3/partial/ng_module.mjs +0 -63
  66. package/esm2022/src/render3/partial/pipe.mjs +0 -50
  67. package/esm2022/src/render3/partial/util.mjs +0 -79
  68. package/esm2022/src/render3/r3_ast.mjs +0 -493
  69. package/esm2022/src/render3/r3_class_debug_info_compiler.mjs +0 -35
  70. package/esm2022/src/render3/r3_class_metadata_compiler.mjs +0 -92
  71. package/esm2022/src/render3/r3_control_flow.mjs +0 -421
  72. package/esm2022/src/render3/r3_deferred_blocks.mjs +0 -170
  73. package/esm2022/src/render3/r3_deferred_triggers.mjs +0 -316
  74. package/esm2022/src/render3/r3_factory.mjs +0 -200
  75. package/esm2022/src/render3/r3_identifiers.mjs +0 -480
  76. package/esm2022/src/render3/r3_injector_compiler.mjs +0 -28
  77. package/esm2022/src/render3/r3_jit.mjs +0 -29
  78. package/esm2022/src/render3/r3_module_compiler.mjs +0 -211
  79. package/esm2022/src/render3/r3_pipe_compiler.mjs +0 -35
  80. package/esm2022/src/render3/r3_template_transform.mjs +0 -560
  81. package/esm2022/src/render3/util.mjs +0 -90
  82. package/esm2022/src/render3/view/api.mjs +0 -14
  83. package/esm2022/src/render3/view/compiler.mjs +0 -546
  84. package/esm2022/src/render3/view/i18n/get_msg_utils.mjs +0 -113
  85. package/esm2022/src/render3/view/i18n/icu_serializer.mjs +0 -43
  86. package/esm2022/src/render3/view/i18n/localize_utils.mjs +0 -118
  87. package/esm2022/src/render3/view/i18n/meta.mjs +0 -259
  88. package/esm2022/src/render3/view/i18n/util.mjs +0 -70
  89. package/esm2022/src/render3/view/query_generation.mjs +0 -182
  90. package/esm2022/src/render3/view/t2_api.mjs +0 -9
  91. package/esm2022/src/render3/view/t2_binder.mjs +0 -823
  92. package/esm2022/src/render3/view/template.mjs +0 -113
  93. package/esm2022/src/render3/view/util.mjs +0 -192
  94. package/esm2022/src/resource_loader.mjs +0 -17
  95. package/esm2022/src/schema/dom_element_schema_registry.mjs +0 -471
  96. package/esm2022/src/schema/dom_security_schema.mjs +0 -96
  97. package/esm2022/src/schema/element_schema_registry.mjs +0 -10
  98. package/esm2022/src/schema/trusted_types_sinks.mjs +0 -41
  99. package/esm2022/src/selector.mjs +0 -400
  100. package/esm2022/src/shadow_css.mjs +0 -1053
  101. package/esm2022/src/style_url_resolver.mjs +0 -17
  102. package/esm2022/src/template/pipeline/ir/index.mjs +0 -18
  103. package/esm2022/src/template/pipeline/ir/src/enums.mjs +0 -503
  104. package/esm2022/src/template/pipeline/ir/src/expression.mjs +0 -997
  105. package/esm2022/src/template/pipeline/ir/src/handle.mjs +0 -13
  106. package/esm2022/src/template/pipeline/ir/src/operations.mjs +0 -261
  107. package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +0 -405
  108. package/esm2022/src/template/pipeline/ir/src/ops/host.mjs +0 -25
  109. package/esm2022/src/template/pipeline/ir/src/ops/shared.mjs +0 -43
  110. package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +0 -277
  111. package/esm2022/src/template/pipeline/ir/src/traits.mjs +0 -64
  112. package/esm2022/src/template/pipeline/ir/src/variable.mjs +0 -13
  113. package/esm2022/src/template/pipeline/src/compilation.mjs +0 -194
  114. package/esm2022/src/template/pipeline/src/conversion.mjs +0 -64
  115. package/esm2022/src/template/pipeline/src/emit.mjs +0 -238
  116. package/esm2022/src/template/pipeline/src/ingest.mjs +0 -1058
  117. package/esm2022/src/template/pipeline/src/instruction.mjs +0 -564
  118. package/esm2022/src/template/pipeline/src/phases/any_cast.mjs +0 -32
  119. package/esm2022/src/template/pipeline/src/phases/apply_i18n_expressions.mjs +0 -63
  120. package/esm2022/src/template/pipeline/src/phases/assign_i18n_slot_dependencies.mjs +0 -63
  121. package/esm2022/src/template/pipeline/src/phases/attribute_extraction.mjs +0 -138
  122. package/esm2022/src/template/pipeline/src/phases/binding_specialization.mjs +0 -75
  123. package/esm2022/src/template/pipeline/src/phases/chaining.mjs +0 -112
  124. package/esm2022/src/template/pipeline/src/phases/collapse_singleton_interpolations.mjs +0 -32
  125. package/esm2022/src/template/pipeline/src/phases/conditionals.mjs +0 -58
  126. package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +0 -214
  127. package/esm2022/src/template/pipeline/src/phases/convert_i18n_bindings.mjs +0 -52
  128. package/esm2022/src/template/pipeline/src/phases/create_i18n_contexts.mjs +0 -105
  129. package/esm2022/src/template/pipeline/src/phases/deduplicate_text_bindings.mjs +0 -40
  130. package/esm2022/src/template/pipeline/src/phases/defer_configs.mjs +0 -29
  131. package/esm2022/src/template/pipeline/src/phases/defer_resolve_targets.mjs +0 -113
  132. package/esm2022/src/template/pipeline/src/phases/empty_elements.mjs +0 -47
  133. package/esm2022/src/template/pipeline/src/phases/expand_safe_reads.mjs +0 -192
  134. package/esm2022/src/template/pipeline/src/phases/extract_i18n_messages.mjs +0 -218
  135. package/esm2022/src/template/pipeline/src/phases/generate_advance.mjs +0 -66
  136. package/esm2022/src/template/pipeline/src/phases/generate_local_let_references.mjs +0 -29
  137. package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +0 -46
  138. package/esm2022/src/template/pipeline/src/phases/generate_variables.mjs +0 -170
  139. package/esm2022/src/template/pipeline/src/phases/has_const_expression_collection.mjs +0 -28
  140. package/esm2022/src/template/pipeline/src/phases/host_style_property_parsing.mjs +0 -81
  141. package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +0 -309
  142. package/esm2022/src/template/pipeline/src/phases/i18n_text_extraction.mjs +0 -97
  143. package/esm2022/src/template/pipeline/src/phases/local_refs.mjs +0 -43
  144. package/esm2022/src/template/pipeline/src/phases/namespace.mjs +0 -26
  145. package/esm2022/src/template/pipeline/src/phases/naming.mjs +0 -179
  146. package/esm2022/src/template/pipeline/src/phases/next_context_merging.mjs +0 -75
  147. package/esm2022/src/template/pipeline/src/phases/ng_container.mjs +0 -29
  148. package/esm2022/src/template/pipeline/src/phases/nonbindable.mjs +0 -47
  149. package/esm2022/src/template/pipeline/src/phases/nullish_coalescing.mjs +0 -33
  150. package/esm2022/src/template/pipeline/src/phases/ordering.mjs +0 -140
  151. package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +0 -137
  152. package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +0 -42
  153. package/esm2022/src/template/pipeline/src/phases/pipe_creation.mjs +0 -74
  154. package/esm2022/src/template/pipeline/src/phases/pipe_variadic.mjs +0 -30
  155. package/esm2022/src/template/pipeline/src/phases/propagate_i18n_blocks.mjs +0 -81
  156. package/esm2022/src/template/pipeline/src/phases/pure_function_extraction.mjs +0 -55
  157. package/esm2022/src/template/pipeline/src/phases/pure_literal_structures.mjs +0 -58
  158. package/esm2022/src/template/pipeline/src/phases/reify.mjs +0 -433
  159. package/esm2022/src/template/pipeline/src/phases/remove_empty_bindings.mjs +0 -31
  160. package/esm2022/src/template/pipeline/src/phases/remove_i18n_contexts.mjs +0 -27
  161. package/esm2022/src/template/pipeline/src/phases/remove_illegal_let_references.mjs +0 -36
  162. package/esm2022/src/template/pipeline/src/phases/remove_unused_i18n_attrs.mjs +0 -33
  163. package/esm2022/src/template/pipeline/src/phases/resolve_contexts.mjs +0 -60
  164. package/esm2022/src/template/pipeline/src/phases/resolve_defer_deps_fns.mjs +0 -31
  165. package/esm2022/src/template/pipeline/src/phases/resolve_dollar_event.mjs +0 -36
  166. package/esm2022/src/template/pipeline/src/phases/resolve_i18n_element_placeholders.mjs +0 -266
  167. package/esm2022/src/template/pipeline/src/phases/resolve_i18n_expression_placeholders.mjs +0 -70
  168. package/esm2022/src/template/pipeline/src/phases/resolve_names.mjs +0 -124
  169. package/esm2022/src/template/pipeline/src/phases/resolve_sanitizers.mjs +0 -123
  170. package/esm2022/src/template/pipeline/src/phases/save_restore_view.mjs +0 -64
  171. package/esm2022/src/template/pipeline/src/phases/slot_allocation.mjs +0 -62
  172. package/esm2022/src/template/pipeline/src/phases/store_let_optimization.mjs +0 -35
  173. package/esm2022/src/template/pipeline/src/phases/style_binding_specialization.mjs +0 -42
  174. package/esm2022/src/template/pipeline/src/phases/temporary_variables.mjs +0 -87
  175. package/esm2022/src/template/pipeline/src/phases/track_fn_generation.mjs +0 -52
  176. package/esm2022/src/template/pipeline/src/phases/track_fn_optimization.mjs +0 -88
  177. package/esm2022/src/template/pipeline/src/phases/track_variables.mjs +0 -36
  178. package/esm2022/src/template/pipeline/src/phases/transform_two_way_binding_set.mjs +0 -43
  179. package/esm2022/src/template/pipeline/src/phases/var_counting.mjs +0 -155
  180. package/esm2022/src/template/pipeline/src/phases/variable_optimization.mjs +0 -423
  181. package/esm2022/src/template/pipeline/src/phases/wrap_icus.mjs +0 -41
  182. package/esm2022/src/template/pipeline/src/util/elements.mjs +0 -29
  183. package/esm2022/src/template_parser/binding_parser.mjs +0 -465
  184. package/esm2022/src/template_parser/template_preparser.mjs +0 -84
  185. package/esm2022/src/util.mjs +0 -124
  186. package/esm2022/src/version.mjs +0 -15
@@ -1,170 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright Google LLC All Rights Reserved.
4
- *
5
- * Use of this source code is governed by an MIT-style license that can be
6
- * found in the LICENSE file at https://angular.io/license
7
- */
8
- import * as html from '../ml_parser/ast';
9
- import { ParseError, ParseSourceSpan } from '../parse_util';
10
- import * as t from './r3_ast';
11
- import { getTriggerParametersStart, parseDeferredTime, parseOnTrigger, parseWhenTrigger, } from './r3_deferred_triggers';
12
- /** Pattern to identify a `prefetch when` trigger. */
13
- const PREFETCH_WHEN_PATTERN = /^prefetch\s+when\s/;
14
- /** Pattern to identify a `prefetch on` trigger. */
15
- const PREFETCH_ON_PATTERN = /^prefetch\s+on\s/;
16
- /** Pattern to identify a `minimum` parameter in a block. */
17
- const MINIMUM_PARAMETER_PATTERN = /^minimum\s/;
18
- /** Pattern to identify a `after` parameter in a block. */
19
- const AFTER_PARAMETER_PATTERN = /^after\s/;
20
- /** Pattern to identify a `when` parameter in a block. */
21
- const WHEN_PARAMETER_PATTERN = /^when\s/;
22
- /** Pattern to identify a `on` parameter in a block. */
23
- const ON_PARAMETER_PATTERN = /^on\s/;
24
- /**
25
- * Predicate function that determines if a block with
26
- * a specific name cam be connected to a `defer` block.
27
- */
28
- export function isConnectedDeferLoopBlock(name) {
29
- return name === 'placeholder' || name === 'loading' || name === 'error';
30
- }
31
- /** Creates a deferred block from an HTML AST node. */
32
- export function createDeferredBlock(ast, connectedBlocks, visitor, bindingParser) {
33
- const errors = [];
34
- const { placeholder, loading, error } = parseConnectedBlocks(connectedBlocks, errors, visitor);
35
- const { triggers, prefetchTriggers } = parsePrimaryTriggers(ast.parameters, bindingParser, errors, placeholder);
36
- // The `defer` block has a main span encompassing all of the connected branches as well.
37
- let lastEndSourceSpan = ast.endSourceSpan;
38
- let endOfLastSourceSpan = ast.sourceSpan.end;
39
- if (connectedBlocks.length > 0) {
40
- const lastConnectedBlock = connectedBlocks[connectedBlocks.length - 1];
41
- lastEndSourceSpan = lastConnectedBlock.endSourceSpan;
42
- endOfLastSourceSpan = lastConnectedBlock.sourceSpan.end;
43
- }
44
- const sourceSpanWithConnectedBlocks = new ParseSourceSpan(ast.sourceSpan.start, endOfLastSourceSpan);
45
- const node = new t.DeferredBlock(html.visitAll(visitor, ast.children, ast.children), triggers, prefetchTriggers, placeholder, loading, error, ast.nameSpan, sourceSpanWithConnectedBlocks, ast.sourceSpan, ast.startSourceSpan, lastEndSourceSpan, ast.i18n);
46
- return { node, errors };
47
- }
48
- function parseConnectedBlocks(connectedBlocks, errors, visitor) {
49
- let placeholder = null;
50
- let loading = null;
51
- let error = null;
52
- for (const block of connectedBlocks) {
53
- try {
54
- if (!isConnectedDeferLoopBlock(block.name)) {
55
- errors.push(new ParseError(block.startSourceSpan, `Unrecognized block "@${block.name}"`));
56
- break;
57
- }
58
- switch (block.name) {
59
- case 'placeholder':
60
- if (placeholder !== null) {
61
- errors.push(new ParseError(block.startSourceSpan, `@defer block can only have one @placeholder block`));
62
- }
63
- else {
64
- placeholder = parsePlaceholderBlock(block, visitor);
65
- }
66
- break;
67
- case 'loading':
68
- if (loading !== null) {
69
- errors.push(new ParseError(block.startSourceSpan, `@defer block can only have one @loading block`));
70
- }
71
- else {
72
- loading = parseLoadingBlock(block, visitor);
73
- }
74
- break;
75
- case 'error':
76
- if (error !== null) {
77
- errors.push(new ParseError(block.startSourceSpan, `@defer block can only have one @error block`));
78
- }
79
- else {
80
- error = parseErrorBlock(block, visitor);
81
- }
82
- break;
83
- }
84
- }
85
- catch (e) {
86
- errors.push(new ParseError(block.startSourceSpan, e.message));
87
- }
88
- }
89
- return { placeholder, loading, error };
90
- }
91
- function parsePlaceholderBlock(ast, visitor) {
92
- let minimumTime = null;
93
- for (const param of ast.parameters) {
94
- if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {
95
- if (minimumTime != null) {
96
- throw new Error(`@placeholder block can only have one "minimum" parameter`);
97
- }
98
- const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
99
- if (parsedTime === null) {
100
- throw new Error(`Could not parse time value of parameter "minimum"`);
101
- }
102
- minimumTime = parsedTime;
103
- }
104
- else {
105
- throw new Error(`Unrecognized parameter in @placeholder block: "${param.expression}"`);
106
- }
107
- }
108
- return new t.DeferredBlockPlaceholder(html.visitAll(visitor, ast.children, ast.children), minimumTime, ast.nameSpan, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan, ast.i18n);
109
- }
110
- function parseLoadingBlock(ast, visitor) {
111
- let afterTime = null;
112
- let minimumTime = null;
113
- for (const param of ast.parameters) {
114
- if (AFTER_PARAMETER_PATTERN.test(param.expression)) {
115
- if (afterTime != null) {
116
- throw new Error(`@loading block can only have one "after" parameter`);
117
- }
118
- const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
119
- if (parsedTime === null) {
120
- throw new Error(`Could not parse time value of parameter "after"`);
121
- }
122
- afterTime = parsedTime;
123
- }
124
- else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {
125
- if (minimumTime != null) {
126
- throw new Error(`@loading block can only have one "minimum" parameter`);
127
- }
128
- const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));
129
- if (parsedTime === null) {
130
- throw new Error(`Could not parse time value of parameter "minimum"`);
131
- }
132
- minimumTime = parsedTime;
133
- }
134
- else {
135
- throw new Error(`Unrecognized parameter in @loading block: "${param.expression}"`);
136
- }
137
- }
138
- return new t.DeferredBlockLoading(html.visitAll(visitor, ast.children, ast.children), afterTime, minimumTime, ast.nameSpan, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan, ast.i18n);
139
- }
140
- function parseErrorBlock(ast, visitor) {
141
- if (ast.parameters.length > 0) {
142
- throw new Error(`@error block cannot have parameters`);
143
- }
144
- return new t.DeferredBlockError(html.visitAll(visitor, ast.children, ast.children), ast.nameSpan, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan, ast.i18n);
145
- }
146
- function parsePrimaryTriggers(params, bindingParser, errors, placeholder) {
147
- const triggers = {};
148
- const prefetchTriggers = {};
149
- for (const param of params) {
150
- // The lexer ignores the leading spaces so we can assume
151
- // that the expression starts with a keyword.
152
- if (WHEN_PARAMETER_PATTERN.test(param.expression)) {
153
- parseWhenTrigger(param, bindingParser, triggers, errors);
154
- }
155
- else if (ON_PARAMETER_PATTERN.test(param.expression)) {
156
- parseOnTrigger(param, triggers, errors, placeholder);
157
- }
158
- else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {
159
- parseWhenTrigger(param, bindingParser, prefetchTriggers, errors);
160
- }
161
- else if (PREFETCH_ON_PATTERN.test(param.expression)) {
162
- parseOnTrigger(param, prefetchTriggers, errors, placeholder);
163
- }
164
- else {
165
- errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));
166
- }
167
- }
168
- return { triggers, prefetchTriggers };
169
- }
170
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_deferred_blocks.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_deferred_blocks.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as html from '../ml_parser/ast';\nimport {ParseError, ParseSourceSpan} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\nimport {\n  getTriggerParametersStart,\n  parseDeferredTime,\n  parseOnTrigger,\n  parseWhenTrigger,\n} from './r3_deferred_triggers';\n\n/** Pattern to identify a `prefetch when` trigger. */\nconst PREFETCH_WHEN_PATTERN = /^prefetch\\s+when\\s/;\n\n/** Pattern to identify a `prefetch on` trigger. */\nconst PREFETCH_ON_PATTERN = /^prefetch\\s+on\\s/;\n\n/** Pattern to identify a `minimum` parameter in a block. */\nconst MINIMUM_PARAMETER_PATTERN = /^minimum\\s/;\n\n/** Pattern to identify a `after` parameter in a block. */\nconst AFTER_PARAMETER_PATTERN = /^after\\s/;\n\n/** Pattern to identify a `when` parameter in a block. */\nconst WHEN_PARAMETER_PATTERN = /^when\\s/;\n\n/** Pattern to identify a `on` parameter in a block. */\nconst ON_PARAMETER_PATTERN = /^on\\s/;\n\n/**\n * Predicate function that determines if a block with\n * a specific name cam be connected to a `defer` block.\n */\nexport function isConnectedDeferLoopBlock(name: string): boolean {\n  return name === 'placeholder' || name === 'loading' || name === 'error';\n}\n\n/** Creates a deferred block from an HTML AST node. */\nexport function createDeferredBlock(\n  ast: html.Block,\n  connectedBlocks: html.Block[],\n  visitor: html.Visitor,\n  bindingParser: BindingParser,\n): {node: t.DeferredBlock; errors: ParseError[]} {\n  const errors: ParseError[] = [];\n  const {placeholder, loading, error} = parseConnectedBlocks(connectedBlocks, errors, visitor);\n  const {triggers, prefetchTriggers} = parsePrimaryTriggers(\n    ast.parameters,\n    bindingParser,\n    errors,\n    placeholder,\n  );\n\n  // The `defer` block has a main span encompassing all of the connected branches as well.\n  let lastEndSourceSpan = ast.endSourceSpan;\n  let endOfLastSourceSpan = ast.sourceSpan.end;\n  if (connectedBlocks.length > 0) {\n    const lastConnectedBlock = connectedBlocks[connectedBlocks.length - 1];\n    lastEndSourceSpan = lastConnectedBlock.endSourceSpan;\n    endOfLastSourceSpan = lastConnectedBlock.sourceSpan.end;\n  }\n\n  const sourceSpanWithConnectedBlocks = new ParseSourceSpan(\n    ast.sourceSpan.start,\n    endOfLastSourceSpan,\n  );\n\n  const node = new t.DeferredBlock(\n    html.visitAll(visitor, ast.children, ast.children),\n    triggers,\n    prefetchTriggers,\n    placeholder,\n    loading,\n    error,\n    ast.nameSpan,\n    sourceSpanWithConnectedBlocks,\n    ast.sourceSpan,\n    ast.startSourceSpan,\n    lastEndSourceSpan,\n    ast.i18n,\n  );\n\n  return {node, errors};\n}\n\nfunction parseConnectedBlocks(\n  connectedBlocks: html.Block[],\n  errors: ParseError[],\n  visitor: html.Visitor,\n) {\n  let placeholder: t.DeferredBlockPlaceholder | null = null;\n  let loading: t.DeferredBlockLoading | null = null;\n  let error: t.DeferredBlockError | null = null;\n\n  for (const block of connectedBlocks) {\n    try {\n      if (!isConnectedDeferLoopBlock(block.name)) {\n        errors.push(new ParseError(block.startSourceSpan, `Unrecognized block \"@${block.name}\"`));\n        break;\n      }\n\n      switch (block.name) {\n        case 'placeholder':\n          if (placeholder !== null) {\n            errors.push(\n              new ParseError(\n                block.startSourceSpan,\n                `@defer block can only have one @placeholder block`,\n              ),\n            );\n          } else {\n            placeholder = parsePlaceholderBlock(block, visitor);\n          }\n          break;\n\n        case 'loading':\n          if (loading !== null) {\n            errors.push(\n              new ParseError(\n                block.startSourceSpan,\n                `@defer block can only have one @loading block`,\n              ),\n            );\n          } else {\n            loading = parseLoadingBlock(block, visitor);\n          }\n          break;\n\n        case 'error':\n          if (error !== null) {\n            errors.push(\n              new ParseError(block.startSourceSpan, `@defer block can only have one @error block`),\n            );\n          } else {\n            error = parseErrorBlock(block, visitor);\n          }\n          break;\n      }\n    } catch (e) {\n      errors.push(new ParseError(block.startSourceSpan, (e as Error).message));\n    }\n  }\n\n  return {placeholder, loading, error};\n}\n\nfunction parsePlaceholderBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockPlaceholder {\n  let minimumTime: number | null = null;\n\n  for (const param of ast.parameters) {\n    if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      if (minimumTime != null) {\n        throw new Error(`@placeholder block can only have one \"minimum\" parameter`);\n      }\n\n      const parsedTime = parseDeferredTime(\n        param.expression.slice(getTriggerParametersStart(param.expression)),\n      );\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in @placeholder block: \"${param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockPlaceholder(\n    html.visitAll(visitor, ast.children, ast.children),\n    minimumTime,\n    ast.nameSpan,\n    ast.sourceSpan,\n    ast.startSourceSpan,\n    ast.endSourceSpan,\n    ast.i18n,\n  );\n}\n\nfunction parseLoadingBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockLoading {\n  let afterTime: number | null = null;\n  let minimumTime: number | null = null;\n\n  for (const param of ast.parameters) {\n    if (AFTER_PARAMETER_PATTERN.test(param.expression)) {\n      if (afterTime != null) {\n        throw new Error(`@loading block can only have one \"after\" parameter`);\n      }\n\n      const parsedTime = parseDeferredTime(\n        param.expression.slice(getTriggerParametersStart(param.expression)),\n      );\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"after\"`);\n      }\n\n      afterTime = parsedTime;\n    } else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      if (minimumTime != null) {\n        throw new Error(`@loading block can only have one \"minimum\" parameter`);\n      }\n\n      const parsedTime = parseDeferredTime(\n        param.expression.slice(getTriggerParametersStart(param.expression)),\n      );\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in @loading block: \"${param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockLoading(\n    html.visitAll(visitor, ast.children, ast.children),\n    afterTime,\n    minimumTime,\n    ast.nameSpan,\n    ast.sourceSpan,\n    ast.startSourceSpan,\n    ast.endSourceSpan,\n    ast.i18n,\n  );\n}\n\nfunction parseErrorBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockError {\n  if (ast.parameters.length > 0) {\n    throw new Error(`@error block cannot have parameters`);\n  }\n\n  return new t.DeferredBlockError(\n    html.visitAll(visitor, ast.children, ast.children),\n    ast.nameSpan,\n    ast.sourceSpan,\n    ast.startSourceSpan,\n    ast.endSourceSpan,\n    ast.i18n,\n  );\n}\n\nfunction parsePrimaryTriggers(\n  params: html.BlockParameter[],\n  bindingParser: BindingParser,\n  errors: ParseError[],\n  placeholder: t.DeferredBlockPlaceholder | null,\n) {\n  const triggers: t.DeferredBlockTriggers = {};\n  const prefetchTriggers: t.DeferredBlockTriggers = {};\n\n  for (const param of params) {\n    // The lexer ignores the leading spaces so we can assume\n    // that the expression starts with a keyword.\n    if (WHEN_PARAMETER_PATTERN.test(param.expression)) {\n      parseWhenTrigger(param, bindingParser, triggers, errors);\n    } else if (ON_PARAMETER_PATTERN.test(param.expression)) {\n      parseOnTrigger(param, triggers, errors, placeholder);\n    } else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {\n      parseWhenTrigger(param, bindingParser, prefetchTriggers, errors);\n    } else if (PREFETCH_ON_PATTERN.test(param.expression)) {\n      parseOnTrigger(param, prefetchTriggers, errors, placeholder);\n    } else {\n      errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));\n    }\n  }\n\n  return {triggers, prefetchTriggers};\n}\n"]}
@@ -1,316 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright Google LLC All Rights Reserved.
4
- *
5
- * Use of this source code is governed by an MIT-style license that can be
6
- * found in the LICENSE file at https://angular.io/license
7
- */
8
- import * as chars from '../chars';
9
- import { Lexer, TokenType } from '../expression_parser/lexer';
10
- import { ParseError, ParseSourceSpan } from '../parse_util';
11
- import * as t from './r3_ast';
12
- /** Pattern for a timing value in a trigger. */
13
- const TIME_PATTERN = /^\d+\.?\d*(ms|s)?$/;
14
- /** Pattern for a separator between keywords in a trigger expression. */
15
- const SEPARATOR_PATTERN = /^\s$/;
16
- /** Pairs of characters that form syntax that is comma-delimited. */
17
- const COMMA_DELIMITED_SYNTAX = new Map([
18
- [chars.$LBRACE, chars.$RBRACE], // Object literals
19
- [chars.$LBRACKET, chars.$RBRACKET], // Array literals
20
- [chars.$LPAREN, chars.$RPAREN], // Function calls
21
- ]);
22
- /** Possible types of `on` triggers. */
23
- var OnTriggerType;
24
- (function (OnTriggerType) {
25
- OnTriggerType["IDLE"] = "idle";
26
- OnTriggerType["TIMER"] = "timer";
27
- OnTriggerType["INTERACTION"] = "interaction";
28
- OnTriggerType["IMMEDIATE"] = "immediate";
29
- OnTriggerType["HOVER"] = "hover";
30
- OnTriggerType["VIEWPORT"] = "viewport";
31
- })(OnTriggerType || (OnTriggerType = {}));
32
- /** Parses a `when` deferred trigger. */
33
- export function parseWhenTrigger({ expression, sourceSpan }, bindingParser, triggers, errors) {
34
- const whenIndex = expression.indexOf('when');
35
- const whenSourceSpan = new ParseSourceSpan(sourceSpan.start.moveBy(whenIndex), sourceSpan.start.moveBy(whenIndex + 'when'.length));
36
- const prefetchSpan = getPrefetchSpan(expression, sourceSpan);
37
- // This is here just to be safe, we shouldn't enter this function
38
- // in the first place if a block doesn't have the "when" keyword.
39
- if (whenIndex === -1) {
40
- errors.push(new ParseError(sourceSpan, `Could not find "when" keyword in expression`));
41
- }
42
- else {
43
- const start = getTriggerParametersStart(expression, whenIndex + 1);
44
- const parsed = bindingParser.parseBinding(expression.slice(start), false, sourceSpan, sourceSpan.start.offset + start);
45
- trackTrigger('when', triggers, errors, new t.BoundDeferredTrigger(parsed, sourceSpan, prefetchSpan, whenSourceSpan));
46
- }
47
- }
48
- /** Parses an `on` trigger */
49
- export function parseOnTrigger({ expression, sourceSpan }, triggers, errors, placeholder) {
50
- const onIndex = expression.indexOf('on');
51
- const onSourceSpan = new ParseSourceSpan(sourceSpan.start.moveBy(onIndex), sourceSpan.start.moveBy(onIndex + 'on'.length));
52
- const prefetchSpan = getPrefetchSpan(expression, sourceSpan);
53
- // This is here just to be safe, we shouldn't enter this function
54
- // in the first place if a block doesn't have the "on" keyword.
55
- if (onIndex === -1) {
56
- errors.push(new ParseError(sourceSpan, `Could not find "on" keyword in expression`));
57
- }
58
- else {
59
- const start = getTriggerParametersStart(expression, onIndex + 1);
60
- const parser = new OnTriggerParser(expression, start, sourceSpan, triggers, errors, placeholder, prefetchSpan, onSourceSpan);
61
- parser.parse();
62
- }
63
- }
64
- function getPrefetchSpan(expression, sourceSpan) {
65
- if (!expression.startsWith('prefetch')) {
66
- return null;
67
- }
68
- return new ParseSourceSpan(sourceSpan.start, sourceSpan.start.moveBy('prefetch'.length));
69
- }
70
- class OnTriggerParser {
71
- constructor(expression, start, span, triggers, errors, placeholder, prefetchSpan, onSourceSpan) {
72
- this.expression = expression;
73
- this.start = start;
74
- this.span = span;
75
- this.triggers = triggers;
76
- this.errors = errors;
77
- this.placeholder = placeholder;
78
- this.prefetchSpan = prefetchSpan;
79
- this.onSourceSpan = onSourceSpan;
80
- this.index = 0;
81
- this.tokens = new Lexer().tokenize(expression.slice(start));
82
- }
83
- parse() {
84
- while (this.tokens.length > 0 && this.index < this.tokens.length) {
85
- const token = this.token();
86
- if (!token.isIdentifier()) {
87
- this.unexpectedToken(token);
88
- break;
89
- }
90
- // An identifier immediately followed by a comma or the end of
91
- // the expression cannot have parameters so we can exit early.
92
- if (this.isFollowedByOrLast(chars.$COMMA)) {
93
- this.consumeTrigger(token, []);
94
- this.advance();
95
- }
96
- else if (this.isFollowedByOrLast(chars.$LPAREN)) {
97
- this.advance(); // Advance to the opening paren.
98
- const prevErrors = this.errors.length;
99
- const parameters = this.consumeParameters();
100
- if (this.errors.length !== prevErrors) {
101
- break;
102
- }
103
- this.consumeTrigger(token, parameters);
104
- this.advance(); // Advance past the closing paren.
105
- }
106
- else if (this.index < this.tokens.length - 1) {
107
- this.unexpectedToken(this.tokens[this.index + 1]);
108
- }
109
- this.advance();
110
- }
111
- }
112
- advance() {
113
- this.index++;
114
- }
115
- isFollowedByOrLast(char) {
116
- if (this.index === this.tokens.length - 1) {
117
- return true;
118
- }
119
- return this.tokens[this.index + 1].isCharacter(char);
120
- }
121
- token() {
122
- return this.tokens[Math.min(this.index, this.tokens.length - 1)];
123
- }
124
- consumeTrigger(identifier, parameters) {
125
- const triggerNameStartSpan = this.span.start.moveBy(this.start + identifier.index - this.tokens[0].index);
126
- const nameSpan = new ParseSourceSpan(triggerNameStartSpan, triggerNameStartSpan.moveBy(identifier.strValue.length));
127
- const endSpan = triggerNameStartSpan.moveBy(this.token().end - identifier.index);
128
- // Put the prefetch and on spans with the first trigger
129
- // This should maybe be refactored to have something like an outer OnGroup AST
130
- // Since triggers can be grouped with commas "on hover(x), interaction(y)"
131
- const isFirstTrigger = identifier.index === 0;
132
- const onSourceSpan = isFirstTrigger ? this.onSourceSpan : null;
133
- const prefetchSourceSpan = isFirstTrigger ? this.prefetchSpan : null;
134
- const sourceSpan = new ParseSourceSpan(isFirstTrigger ? this.span.start : triggerNameStartSpan, endSpan);
135
- try {
136
- switch (identifier.toString()) {
137
- case OnTriggerType.IDLE:
138
- this.trackTrigger('idle', createIdleTrigger(parameters, nameSpan, sourceSpan, prefetchSourceSpan, onSourceSpan));
139
- break;
140
- case OnTriggerType.TIMER:
141
- this.trackTrigger('timer', createTimerTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));
142
- break;
143
- case OnTriggerType.INTERACTION:
144
- this.trackTrigger('interaction', createInteractionTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
145
- break;
146
- case OnTriggerType.IMMEDIATE:
147
- this.trackTrigger('immediate', createImmediateTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));
148
- break;
149
- case OnTriggerType.HOVER:
150
- this.trackTrigger('hover', createHoverTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
151
- break;
152
- case OnTriggerType.VIEWPORT:
153
- this.trackTrigger('viewport', createViewportTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
154
- break;
155
- default:
156
- throw new Error(`Unrecognized trigger type "${identifier}"`);
157
- }
158
- }
159
- catch (e) {
160
- this.error(identifier, e.message);
161
- }
162
- }
163
- consumeParameters() {
164
- const parameters = [];
165
- if (!this.token().isCharacter(chars.$LPAREN)) {
166
- this.unexpectedToken(this.token());
167
- return parameters;
168
- }
169
- this.advance();
170
- const commaDelimStack = [];
171
- let current = '';
172
- while (this.index < this.tokens.length) {
173
- const token = this.token();
174
- // Stop parsing if we've hit the end character and we're outside of a comma-delimited syntax.
175
- // Note that we don't need to account for strings here since the lexer already parsed them
176
- // into string tokens.
177
- if (token.isCharacter(chars.$RPAREN) && commaDelimStack.length === 0) {
178
- if (current.length) {
179
- parameters.push(current);
180
- }
181
- break;
182
- }
183
- // In the `on` microsyntax "top-level" commas (e.g. ones outside of an parameters) separate
184
- // the different triggers (e.g. `on idle,timer(500)`). This is problematic, because the
185
- // function-like syntax also implies that multiple parameters can be passed into the
186
- // individual trigger (e.g. `on foo(a, b)`). To avoid tripping up the parser with commas that
187
- // are part of other sorts of syntax (object literals, arrays), we treat anything inside
188
- // a comma-delimited syntax block as plain text.
189
- if (token.type === TokenType.Character && COMMA_DELIMITED_SYNTAX.has(token.numValue)) {
190
- commaDelimStack.push(COMMA_DELIMITED_SYNTAX.get(token.numValue));
191
- }
192
- if (commaDelimStack.length > 0 &&
193
- token.isCharacter(commaDelimStack[commaDelimStack.length - 1])) {
194
- commaDelimStack.pop();
195
- }
196
- // If we hit a comma outside of a comma-delimited syntax, it means
197
- // that we're at the top level and we're starting a new parameter.
198
- if (commaDelimStack.length === 0 && token.isCharacter(chars.$COMMA) && current.length > 0) {
199
- parameters.push(current);
200
- current = '';
201
- this.advance();
202
- continue;
203
- }
204
- // Otherwise treat the token as a plain text character in the current parameter.
205
- current += this.tokenText();
206
- this.advance();
207
- }
208
- if (!this.token().isCharacter(chars.$RPAREN) || commaDelimStack.length > 0) {
209
- this.error(this.token(), 'Unexpected end of expression');
210
- }
211
- if (this.index < this.tokens.length - 1 &&
212
- !this.tokens[this.index + 1].isCharacter(chars.$COMMA)) {
213
- this.unexpectedToken(this.tokens[this.index + 1]);
214
- }
215
- return parameters;
216
- }
217
- tokenText() {
218
- // Tokens have a toString already which we could use, but for string tokens it omits the quotes.
219
- // Eventually we could expose this information on the token directly.
220
- return this.expression.slice(this.start + this.token().index, this.start + this.token().end);
221
- }
222
- trackTrigger(name, trigger) {
223
- trackTrigger(name, this.triggers, this.errors, trigger);
224
- }
225
- error(token, message) {
226
- const newStart = this.span.start.moveBy(this.start + token.index);
227
- const newEnd = newStart.moveBy(token.end - token.index);
228
- this.errors.push(new ParseError(new ParseSourceSpan(newStart, newEnd), message));
229
- }
230
- unexpectedToken(token) {
231
- this.error(token, `Unexpected token "${token}"`);
232
- }
233
- }
234
- /** Adds a trigger to a map of triggers. */
235
- function trackTrigger(name, allTriggers, errors, trigger) {
236
- if (allTriggers[name]) {
237
- errors.push(new ParseError(trigger.sourceSpan, `Duplicate "${name}" trigger is not allowed`));
238
- }
239
- else {
240
- allTriggers[name] = trigger;
241
- }
242
- }
243
- function createIdleTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
244
- if (parameters.length > 0) {
245
- throw new Error(`"${OnTriggerType.IDLE}" trigger cannot have parameters`);
246
- }
247
- return new t.IdleDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
248
- }
249
- function createTimerTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
250
- if (parameters.length !== 1) {
251
- throw new Error(`"${OnTriggerType.TIMER}" trigger must have exactly one parameter`);
252
- }
253
- const delay = parseDeferredTime(parameters[0]);
254
- if (delay === null) {
255
- throw new Error(`Could not parse time value of trigger "${OnTriggerType.TIMER}"`);
256
- }
257
- return new t.TimerDeferredTrigger(delay, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
258
- }
259
- function createImmediateTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
260
- if (parameters.length > 0) {
261
- throw new Error(`"${OnTriggerType.IMMEDIATE}" trigger cannot have parameters`);
262
- }
263
- return new t.ImmediateDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
264
- }
265
- function createHoverTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
266
- validateReferenceBasedTrigger(OnTriggerType.HOVER, parameters, placeholder);
267
- return new t.HoverDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
268
- }
269
- function createInteractionTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
270
- validateReferenceBasedTrigger(OnTriggerType.INTERACTION, parameters, placeholder);
271
- return new t.InteractionDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
272
- }
273
- function createViewportTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
274
- validateReferenceBasedTrigger(OnTriggerType.VIEWPORT, parameters, placeholder);
275
- return new t.ViewportDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
276
- }
277
- function validateReferenceBasedTrigger(type, parameters, placeholder) {
278
- if (parameters.length > 1) {
279
- throw new Error(`"${type}" trigger can only have zero or one parameters`);
280
- }
281
- if (parameters.length === 0) {
282
- if (placeholder === null) {
283
- throw new Error(`"${type}" trigger with no parameters can only be placed on an @defer that has a @placeholder block`);
284
- }
285
- if (placeholder.children.length !== 1 || !(placeholder.children[0] instanceof t.Element)) {
286
- throw new Error(`"${type}" trigger with no parameters can only be placed on an @defer that has a ` +
287
- `@placeholder block with exactly one root element node`);
288
- }
289
- }
290
- }
291
- /** Gets the index within an expression at which the trigger parameters start. */
292
- export function getTriggerParametersStart(value, startPosition = 0) {
293
- let hasFoundSeparator = false;
294
- for (let i = startPosition; i < value.length; i++) {
295
- if (SEPARATOR_PATTERN.test(value[i])) {
296
- hasFoundSeparator = true;
297
- }
298
- else if (hasFoundSeparator) {
299
- return i;
300
- }
301
- }
302
- return -1;
303
- }
304
- /**
305
- * Parses a time expression from a deferred trigger to
306
- * milliseconds. Returns null if it cannot be parsed.
307
- */
308
- export function parseDeferredTime(value) {
309
- const match = value.match(TIME_PATTERN);
310
- if (!match) {
311
- return null;
312
- }
313
- const [time, units] = match;
314
- return parseFloat(time) * (units === 's' ? 1000 : 1);
315
- }
316
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_deferred_triggers.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_deferred_triggers.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as chars from '../chars';\nimport {Lexer, Token, TokenType} from '../expression_parser/lexer';\nimport * as html from '../ml_parser/ast';\nimport {ParseError, ParseSourceSpan} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\n\n/** Pattern for a timing value in a trigger. */\nconst TIME_PATTERN = /^\\d+\\.?\\d*(ms|s)?$/;\n\n/** Pattern for a separator between keywords in a trigger expression. */\nconst SEPARATOR_PATTERN = /^\\s$/;\n\n/** Pairs of characters that form syntax that is comma-delimited. */\nconst COMMA_DELIMITED_SYNTAX = new Map([\n  [chars.$LBRACE, chars.$RBRACE], // Object literals\n  [chars.$LBRACKET, chars.$RBRACKET], // Array literals\n  [chars.$LPAREN, chars.$RPAREN], // Function calls\n]);\n\n/** Possible types of `on` triggers. */\nenum OnTriggerType {\n  IDLE = 'idle',\n  TIMER = 'timer',\n  INTERACTION = 'interaction',\n  IMMEDIATE = 'immediate',\n  HOVER = 'hover',\n  VIEWPORT = 'viewport',\n}\n\n/** Parses a `when` deferred trigger. */\nexport function parseWhenTrigger(\n  {expression, sourceSpan}: html.BlockParameter,\n  bindingParser: BindingParser,\n  triggers: t.DeferredBlockTriggers,\n  errors: ParseError[],\n): void {\n  const whenIndex = expression.indexOf('when');\n  const whenSourceSpan = new ParseSourceSpan(\n    sourceSpan.start.moveBy(whenIndex),\n    sourceSpan.start.moveBy(whenIndex + 'when'.length),\n  );\n  const prefetchSpan = getPrefetchSpan(expression, sourceSpan);\n\n  // This is here just to be safe, we shouldn't enter this function\n  // in the first place if a block doesn't have the \"when\" keyword.\n  if (whenIndex === -1) {\n    errors.push(new ParseError(sourceSpan, `Could not find \"when\" keyword in expression`));\n  } else {\n    const start = getTriggerParametersStart(expression, whenIndex + 1);\n    const parsed = bindingParser.parseBinding(\n      expression.slice(start),\n      false,\n      sourceSpan,\n      sourceSpan.start.offset + start,\n    );\n    trackTrigger(\n      'when',\n      triggers,\n      errors,\n      new t.BoundDeferredTrigger(parsed, sourceSpan, prefetchSpan, whenSourceSpan),\n    );\n  }\n}\n\n/** Parses an `on` trigger */\nexport function parseOnTrigger(\n  {expression, sourceSpan}: html.BlockParameter,\n  triggers: t.DeferredBlockTriggers,\n  errors: ParseError[],\n  placeholder: t.DeferredBlockPlaceholder | null,\n): void {\n  const onIndex = expression.indexOf('on');\n  const onSourceSpan = new ParseSourceSpan(\n    sourceSpan.start.moveBy(onIndex),\n    sourceSpan.start.moveBy(onIndex + 'on'.length),\n  );\n  const prefetchSpan = getPrefetchSpan(expression, sourceSpan);\n\n  // This is here just to be safe, we shouldn't enter this function\n  // in the first place if a block doesn't have the \"on\" keyword.\n  if (onIndex === -1) {\n    errors.push(new ParseError(sourceSpan, `Could not find \"on\" keyword in expression`));\n  } else {\n    const start = getTriggerParametersStart(expression, onIndex + 1);\n    const parser = new OnTriggerParser(\n      expression,\n      start,\n      sourceSpan,\n      triggers,\n      errors,\n      placeholder,\n      prefetchSpan,\n      onSourceSpan,\n    );\n    parser.parse();\n  }\n}\n\nfunction getPrefetchSpan(expression: string, sourceSpan: ParseSourceSpan) {\n  if (!expression.startsWith('prefetch')) {\n    return null;\n  }\n  return new ParseSourceSpan(sourceSpan.start, sourceSpan.start.moveBy('prefetch'.length));\n}\n\nclass OnTriggerParser {\n  private index = 0;\n  private tokens: Token[];\n\n  constructor(\n    private expression: string,\n    private start: number,\n    private span: ParseSourceSpan,\n    private triggers: t.DeferredBlockTriggers,\n    private errors: ParseError[],\n    private placeholder: t.DeferredBlockPlaceholder | null,\n    private prefetchSpan: ParseSourceSpan | null,\n    private onSourceSpan: ParseSourceSpan,\n  ) {\n    this.tokens = new Lexer().tokenize(expression.slice(start));\n  }\n\n  parse(): void {\n    while (this.tokens.length > 0 && this.index < this.tokens.length) {\n      const token = this.token();\n\n      if (!token.isIdentifier()) {\n        this.unexpectedToken(token);\n        break;\n      }\n\n      // An identifier immediately followed by a comma or the end of\n      // the expression cannot have parameters so we can exit early.\n      if (this.isFollowedByOrLast(chars.$COMMA)) {\n        this.consumeTrigger(token, []);\n        this.advance();\n      } else if (this.isFollowedByOrLast(chars.$LPAREN)) {\n        this.advance(); // Advance to the opening paren.\n        const prevErrors = this.errors.length;\n        const parameters = this.consumeParameters();\n        if (this.errors.length !== prevErrors) {\n          break;\n        }\n        this.consumeTrigger(token, parameters);\n        this.advance(); // Advance past the closing paren.\n      } else if (this.index < this.tokens.length - 1) {\n        this.unexpectedToken(this.tokens[this.index + 1]);\n      }\n\n      this.advance();\n    }\n  }\n\n  private advance() {\n    this.index++;\n  }\n\n  private isFollowedByOrLast(char: number): boolean {\n    if (this.index === this.tokens.length - 1) {\n      return true;\n    }\n\n    return this.tokens[this.index + 1].isCharacter(char);\n  }\n\n  private token(): Token {\n    return this.tokens[Math.min(this.index, this.tokens.length - 1)];\n  }\n\n  private consumeTrigger(identifier: Token, parameters: string[]) {\n    const triggerNameStartSpan = this.span.start.moveBy(\n      this.start + identifier.index - this.tokens[0].index,\n    );\n    const nameSpan = new ParseSourceSpan(\n      triggerNameStartSpan,\n      triggerNameStartSpan.moveBy(identifier.strValue.length),\n    );\n    const endSpan = triggerNameStartSpan.moveBy(this.token().end - identifier.index);\n\n    // Put the prefetch and on spans with the first trigger\n    // This should maybe be refactored to have something like an outer OnGroup AST\n    // Since triggers can be grouped with commas \"on hover(x), interaction(y)\"\n    const isFirstTrigger = identifier.index === 0;\n    const onSourceSpan = isFirstTrigger ? this.onSourceSpan : null;\n    const prefetchSourceSpan = isFirstTrigger ? this.prefetchSpan : null;\n    const sourceSpan = new ParseSourceSpan(\n      isFirstTrigger ? this.span.start : triggerNameStartSpan,\n      endSpan,\n    );\n\n    try {\n      switch (identifier.toString()) {\n        case OnTriggerType.IDLE:\n          this.trackTrigger(\n            'idle',\n            createIdleTrigger(parameters, nameSpan, sourceSpan, prefetchSourceSpan, onSourceSpan),\n          );\n          break;\n\n        case OnTriggerType.TIMER:\n          this.trackTrigger(\n            'timer',\n            createTimerTrigger(\n              parameters,\n              nameSpan,\n              sourceSpan,\n              this.prefetchSpan,\n              this.onSourceSpan,\n            ),\n          );\n          break;\n\n        case OnTriggerType.INTERACTION:\n          this.trackTrigger(\n            'interaction',\n            createInteractionTrigger(\n              parameters,\n              nameSpan,\n              sourceSpan,\n              this.prefetchSpan,\n              this.onSourceSpan,\n              this.placeholder,\n            ),\n          );\n          break;\n\n        case OnTriggerType.IMMEDIATE:\n          this.trackTrigger(\n            'immediate',\n            createImmediateTrigger(\n              parameters,\n              nameSpan,\n              sourceSpan,\n              this.prefetchSpan,\n              this.onSourceSpan,\n            ),\n          );\n          break;\n\n        case OnTriggerType.HOVER:\n          this.trackTrigger(\n            'hover',\n            createHoverTrigger(\n              parameters,\n              nameSpan,\n              sourceSpan,\n              this.prefetchSpan,\n              this.onSourceSpan,\n              this.placeholder,\n            ),\n          );\n          break;\n\n        case OnTriggerType.VIEWPORT:\n          this.trackTrigger(\n            'viewport',\n            createViewportTrigger(\n              parameters,\n              nameSpan,\n              sourceSpan,\n              this.prefetchSpan,\n              this.onSourceSpan,\n              this.placeholder,\n            ),\n          );\n          break;\n\n        default:\n          throw new Error(`Unrecognized trigger type \"${identifier}\"`);\n      }\n    } catch (e) {\n      this.error(identifier, (e as Error).message);\n    }\n  }\n\n  private consumeParameters(): string[] {\n    const parameters: string[] = [];\n\n    if (!this.token().isCharacter(chars.$LPAREN)) {\n      this.unexpectedToken(this.token());\n      return parameters;\n    }\n\n    this.advance();\n\n    const commaDelimStack: number[] = [];\n    let current = '';\n\n    while (this.index < this.tokens.length) {\n      const token = this.token();\n\n      // Stop parsing if we've hit the end character and we're outside of a comma-delimited syntax.\n      // Note that we don't need to account for strings here since the lexer already parsed them\n      // into string tokens.\n      if (token.isCharacter(chars.$RPAREN) && commaDelimStack.length === 0) {\n        if (current.length) {\n          parameters.push(current);\n        }\n        break;\n      }\n\n      // In the `on` microsyntax \"top-level\" commas (e.g. ones outside of an parameters) separate\n      // the different triggers (e.g. `on idle,timer(500)`). This is problematic, because the\n      // function-like syntax also implies that multiple parameters can be passed into the\n      // individual trigger (e.g. `on foo(a, b)`). To avoid tripping up the parser with commas that\n      // are part of other sorts of syntax (object literals, arrays), we treat anything inside\n      // a comma-delimited syntax block as plain text.\n      if (token.type === TokenType.Character && COMMA_DELIMITED_SYNTAX.has(token.numValue)) {\n        commaDelimStack.push(COMMA_DELIMITED_SYNTAX.get(token.numValue)!);\n      }\n\n      if (\n        commaDelimStack.length > 0 &&\n        token.isCharacter(commaDelimStack[commaDelimStack.length - 1])\n      ) {\n        commaDelimStack.pop();\n      }\n\n      // If we hit a comma outside of a comma-delimited syntax, it means\n      // that we're at the top level and we're starting a new parameter.\n      if (commaDelimStack.length === 0 && token.isCharacter(chars.$COMMA) && current.length > 0) {\n        parameters.push(current);\n        current = '';\n        this.advance();\n        continue;\n      }\n\n      // Otherwise treat the token as a plain text character in the current parameter.\n      current += this.tokenText();\n      this.advance();\n    }\n\n    if (!this.token().isCharacter(chars.$RPAREN) || commaDelimStack.length > 0) {\n      this.error(this.token(), 'Unexpected end of expression');\n    }\n\n    if (\n      this.index < this.tokens.length - 1 &&\n      !this.tokens[this.index + 1].isCharacter(chars.$COMMA)\n    ) {\n      this.unexpectedToken(this.tokens[this.index + 1]);\n    }\n\n    return parameters;\n  }\n\n  private tokenText(): string {\n    // Tokens have a toString already which we could use, but for string tokens it omits the quotes.\n    // Eventually we could expose this information on the token directly.\n    return this.expression.slice(this.start + this.token().index, this.start + this.token().end);\n  }\n\n  private trackTrigger(name: keyof t.DeferredBlockTriggers, trigger: t.DeferredTrigger): void {\n    trackTrigger(name, this.triggers, this.errors, trigger);\n  }\n\n  private error(token: Token, message: string): void {\n    const newStart = this.span.start.moveBy(this.start + token.index);\n    const newEnd = newStart.moveBy(token.end - token.index);\n    this.errors.push(new ParseError(new ParseSourceSpan(newStart, newEnd), message));\n  }\n\n  private unexpectedToken(token: Token) {\n    this.error(token, `Unexpected token \"${token}\"`);\n  }\n}\n\n/** Adds a trigger to a map of triggers. */\nfunction trackTrigger(\n  name: keyof t.DeferredBlockTriggers,\n  allTriggers: t.DeferredBlockTriggers,\n  errors: ParseError[],\n  trigger: t.DeferredTrigger,\n) {\n  if (allTriggers[name]) {\n    errors.push(new ParseError(trigger.sourceSpan, `Duplicate \"${name}\" trigger is not allowed`));\n  } else {\n    allTriggers[name] = trigger as any;\n  }\n}\n\nfunction createIdleTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n): t.IdleDeferredTrigger {\n  if (parameters.length > 0) {\n    throw new Error(`\"${OnTriggerType.IDLE}\" trigger cannot have parameters`);\n  }\n\n  return new t.IdleDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction createTimerTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n) {\n  if (parameters.length !== 1) {\n    throw new Error(`\"${OnTriggerType.TIMER}\" trigger must have exactly one parameter`);\n  }\n\n  const delay = parseDeferredTime(parameters[0]);\n\n  if (delay === null) {\n    throw new Error(`Could not parse time value of trigger \"${OnTriggerType.TIMER}\"`);\n  }\n\n  return new t.TimerDeferredTrigger(delay, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction createImmediateTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n): t.ImmediateDeferredTrigger {\n  if (parameters.length > 0) {\n    throw new Error(`\"${OnTriggerType.IMMEDIATE}\" trigger cannot have parameters`);\n  }\n\n  return new t.ImmediateDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction createHoverTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n  placeholder: t.DeferredBlockPlaceholder | null,\n): t.HoverDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.HOVER, parameters, placeholder);\n  return new t.HoverDeferredTrigger(\n    parameters[0] ?? null,\n    nameSpan,\n    sourceSpan,\n    prefetchSpan,\n    onSourceSpan,\n  );\n}\n\nfunction createInteractionTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n  placeholder: t.DeferredBlockPlaceholder | null,\n): t.InteractionDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.INTERACTION, parameters, placeholder);\n  return new t.InteractionDeferredTrigger(\n    parameters[0] ?? null,\n    nameSpan,\n    sourceSpan,\n    prefetchSpan,\n    onSourceSpan,\n  );\n}\n\nfunction createViewportTrigger(\n  parameters: string[],\n  nameSpan: ParseSourceSpan,\n  sourceSpan: ParseSourceSpan,\n  prefetchSpan: ParseSourceSpan | null,\n  onSourceSpan: ParseSourceSpan | null,\n  placeholder: t.DeferredBlockPlaceholder | null,\n): t.ViewportDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.VIEWPORT, parameters, placeholder);\n  return new t.ViewportDeferredTrigger(\n    parameters[0] ?? null,\n    nameSpan,\n    sourceSpan,\n    prefetchSpan,\n    onSourceSpan,\n  );\n}\n\nfunction validateReferenceBasedTrigger(\n  type: OnTriggerType,\n  parameters: string[],\n  placeholder: t.DeferredBlockPlaceholder | null,\n) {\n  if (parameters.length > 1) {\n    throw new Error(`\"${type}\" trigger can only have zero or one parameters`);\n  }\n\n  if (parameters.length === 0) {\n    if (placeholder === null) {\n      throw new Error(\n        `\"${type}\" trigger with no parameters can only be placed on an @defer that has a @placeholder block`,\n      );\n    }\n\n    if (placeholder.children.length !== 1 || !(placeholder.children[0] instanceof t.Element)) {\n      throw new Error(\n        `\"${type}\" trigger with no parameters can only be placed on an @defer that has a ` +\n          `@placeholder block with exactly one root element node`,\n      );\n    }\n  }\n}\n\n/** Gets the index within an expression at which the trigger parameters start. */\nexport function getTriggerParametersStart(value: string, startPosition = 0): number {\n  let hasFoundSeparator = false;\n\n  for (let i = startPosition; i < value.length; i++) {\n    if (SEPARATOR_PATTERN.test(value[i])) {\n      hasFoundSeparator = true;\n    } else if (hasFoundSeparator) {\n      return i;\n    }\n  }\n\n  return -1;\n}\n\n/**\n * Parses a time expression from a deferred trigger to\n * milliseconds. Returns null if it cannot be parsed.\n */\nexport function parseDeferredTime(value: string): number | null {\n  const match = value.match(TIME_PATTERN);\n\n  if (!match) {\n    return null;\n  }\n\n  const [time, units] = match;\n  return parseFloat(time) * (units === 's' ? 1000 : 1);\n}\n"]}