@angular/compiler 17.0.0-rc.1 → 17.0.0-rc.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
- package/esm2022/src/render3/partial/directive.mjs +1 -1
- package/esm2022/src/render3/partial/factory.mjs +1 -1
- package/esm2022/src/render3/partial/injectable.mjs +1 -1
- package/esm2022/src/render3/partial/injector.mjs +1 -1
- package/esm2022/src/render3/partial/ng_module.mjs +1 -1
- package/esm2022/src/render3/partial/pipe.mjs +1 -1
- package/esm2022/src/render3/r3_ast.mjs +55 -61
- package/esm2022/src/render3/r3_control_flow.mjs +17 -13
- package/esm2022/src/render3/r3_deferred_blocks.mjs +7 -8
- package/esm2022/src/render3/r3_deferred_triggers.mjs +46 -27
- package/esm2022/src/render3/view/t2_api.mjs +1 -1
- package/esm2022/src/render3/view/t2_binder.mjs +22 -10
- package/esm2022/src/render3/view/template.mjs +54 -5
- package/esm2022/src/template/pipeline/ir/index.mjs +2 -1
- package/esm2022/src/template/pipeline/ir/src/enums.mjs +75 -20
- package/esm2022/src/template/pipeline/ir/src/expression.mjs +74 -71
- package/esm2022/src/template/pipeline/ir/src/handle.mjs +13 -0
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +47 -47
- package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +10 -9
- package/esm2022/src/template/pipeline/ir/src/traits.mjs +1 -32
- package/esm2022/src/template/pipeline/src/conversion.mjs +2 -2
- package/esm2022/src/template/pipeline/src/emit.mjs +117 -107
- package/esm2022/src/template/pipeline/src/ingest.mjs +155 -49
- package/esm2022/src/template/pipeline/src/instruction.mjs +36 -12
- package/esm2022/src/template/pipeline/src/phases/any_cast.mjs +4 -3
- package/esm2022/src/template/pipeline/src/phases/apply_i18n_expressions.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/assign_i18n_slot_dependencies.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/attribute_extraction.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/binding_specialization.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/chaining.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/collapse_singleton_interpolations.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/conditionals.mjs +6 -6
- package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/defer_configs.mjs +30 -0
- package/esm2022/src/template/pipeline/src/phases/defer_resolve_targets.mjs +93 -0
- package/esm2022/src/template/pipeline/src/phases/empty_elements.mjs +4 -4
- package/esm2022/src/template/pipeline/src/phases/expand_safe_reads.mjs +7 -4
- package/esm2022/src/template/pipeline/src/phases/format_i18n_params.mjs +114 -0
- package/esm2022/src/template/pipeline/src/phases/generate_advance.mjs +4 -4
- package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/generate_variables.mjs +4 -3
- package/esm2022/src/template/pipeline/src/phases/has_const_expression_collection.mjs +28 -0
- package/esm2022/src/template/pipeline/src/phases/host_style_property_parsing.mjs +8 -2
- package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +129 -7
- package/esm2022/src/template/pipeline/src/phases/i18n_message_extraction.mjs +4 -105
- package/esm2022/src/template/pipeline/src/phases/i18n_text_extraction.mjs +9 -6
- package/esm2022/src/template/pipeline/src/phases/icu_extraction.mjs +8 -5
- package/esm2022/src/template/pipeline/src/phases/local_refs.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/namespace.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/naming.mjs +11 -13
- package/esm2022/src/template/pipeline/src/phases/next_context_merging.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/ng_container.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/nonbindable.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/nullish_coalescing.mjs +8 -2
- package/esm2022/src/template/pipeline/src/phases/ordering.mjs +7 -2
- package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/pipe_creation.mjs +16 -11
- package/esm2022/src/template/pipeline/src/phases/pipe_variadic.mjs +7 -3
- package/esm2022/src/template/pipeline/src/phases/propagate_i18n_blocks.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/propagate_i18n_placeholders.mjs +61 -0
- package/esm2022/src/template/pipeline/src/phases/pure_function_extraction.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/pure_literal_structures.mjs +4 -4
- package/esm2022/src/template/pipeline/src/phases/reify.mjs +49 -29
- package/esm2022/src/template/pipeline/src/phases/remove_empty_bindings.mjs +5 -2
- package/esm2022/src/template/pipeline/src/phases/repeater_derived_vars.mjs +7 -2
- package/esm2022/src/template/pipeline/src/phases/resolve_contexts.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/resolve_dollar_event.mjs +5 -5
- package/esm2022/src/template/pipeline/src/phases/resolve_i18n_element_placeholders.mjs +109 -0
- package/esm2022/src/template/pipeline/src/phases/resolve_i18n_expression_placeholders.mjs +58 -0
- package/esm2022/src/template/pipeline/src/phases/resolve_names.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/resolve_sanitizers.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/save_restore_view.mjs +14 -9
- package/esm2022/src/template/pipeline/src/phases/slot_allocation.mjs +4 -29
- package/esm2022/src/template/pipeline/src/phases/style_binding_specialization.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/temporary_variables.mjs +3 -3
- package/esm2022/src/template/pipeline/src/phases/track_fn_generation.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/track_fn_optimization.mjs +8 -2
- package/esm2022/src/template/pipeline/src/phases/track_variables.mjs +7 -2
- package/esm2022/src/template/pipeline/src/phases/var_counting.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/variable_optimization.mjs +5 -5
- package/esm2022/src/template/pipeline/src/phases/wrap_icus.mjs +2 -2
- package/esm2022/src/version.mjs +1 -1
- package/fesm2022/compiler.mjs +1467 -1017
- package/fesm2022/compiler.mjs.map +1 -1
- package/index.d.ts +41 -58
- package/package.json +3 -3
- package/esm2022/src/template/pipeline/src/phases/has_const_trait_collection.mjs +0 -34
- package/esm2022/src/template/pipeline/src/phases/resolve_i18n_placeholders.mjs +0 -288
|
@@ -10,7 +10,7 @@ import { Lexer, TokenType } from '../expression_parser/lexer';
|
|
|
10
10
|
import { ParseError, ParseSourceSpan } from '../parse_util';
|
|
11
11
|
import * as t from './r3_ast';
|
|
12
12
|
/** Pattern for a timing value in a trigger. */
|
|
13
|
-
const TIME_PATTERN = /^\d
|
|
13
|
+
const TIME_PATTERN = /^\d+\.?\d*(ms|s)?$/;
|
|
14
14
|
/** Pattern for a separator between keywords in a trigger expression. */
|
|
15
15
|
const SEPARATOR_PATTERN = /^\s$/;
|
|
16
16
|
/** Pairs of characters that form syntax that is comma-delimited. */
|
|
@@ -32,6 +32,8 @@ var OnTriggerType;
|
|
|
32
32
|
/** Parses a `when` deferred trigger. */
|
|
33
33
|
export function parseWhenTrigger({ expression, sourceSpan }, bindingParser, triggers, errors) {
|
|
34
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);
|
|
35
37
|
// This is here just to be safe, we shouldn't enter this function
|
|
36
38
|
// in the first place if a block doesn't have the "when" keyword.
|
|
37
39
|
if (whenIndex === -1) {
|
|
@@ -40,12 +42,14 @@ export function parseWhenTrigger({ expression, sourceSpan }, bindingParser, trig
|
|
|
40
42
|
else {
|
|
41
43
|
const start = getTriggerParametersStart(expression, whenIndex + 1);
|
|
42
44
|
const parsed = bindingParser.parseBinding(expression.slice(start), false, sourceSpan, sourceSpan.start.offset + start);
|
|
43
|
-
trackTrigger('when', triggers, errors, new t.BoundDeferredTrigger(parsed, sourceSpan));
|
|
45
|
+
trackTrigger('when', triggers, errors, new t.BoundDeferredTrigger(parsed, sourceSpan, prefetchSpan, whenSourceSpan));
|
|
44
46
|
}
|
|
45
47
|
}
|
|
46
48
|
/** Parses an `on` trigger */
|
|
47
49
|
export function parseOnTrigger({ expression, sourceSpan }, triggers, errors, placeholder) {
|
|
48
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);
|
|
49
53
|
// This is here just to be safe, we shouldn't enter this function
|
|
50
54
|
// in the first place if a block doesn't have the "on" keyword.
|
|
51
55
|
if (onIndex === -1) {
|
|
@@ -53,18 +57,26 @@ export function parseOnTrigger({ expression, sourceSpan }, triggers, errors, pla
|
|
|
53
57
|
}
|
|
54
58
|
else {
|
|
55
59
|
const start = getTriggerParametersStart(expression, onIndex + 1);
|
|
56
|
-
const parser = new OnTriggerParser(expression, start, sourceSpan, triggers, errors, placeholder);
|
|
60
|
+
const parser = new OnTriggerParser(expression, start, sourceSpan, triggers, errors, placeholder, prefetchSpan, onSourceSpan);
|
|
57
61
|
parser.parse();
|
|
58
62
|
}
|
|
59
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
|
+
}
|
|
60
70
|
class OnTriggerParser {
|
|
61
|
-
constructor(expression, start, span, triggers, errors, placeholder) {
|
|
71
|
+
constructor(expression, start, span, triggers, errors, placeholder, prefetchSpan, onSourceSpan) {
|
|
62
72
|
this.expression = expression;
|
|
63
73
|
this.start = start;
|
|
64
74
|
this.span = span;
|
|
65
75
|
this.triggers = triggers;
|
|
66
76
|
this.errors = errors;
|
|
67
77
|
this.placeholder = placeholder;
|
|
78
|
+
this.prefetchSpan = prefetchSpan;
|
|
79
|
+
this.onSourceSpan = onSourceSpan;
|
|
68
80
|
this.index = 0;
|
|
69
81
|
this.tokens = new Lexer().tokenize(expression.slice(start));
|
|
70
82
|
}
|
|
@@ -110,28 +122,35 @@ class OnTriggerParser {
|
|
|
110
122
|
return this.tokens[Math.min(this.index, this.tokens.length - 1)];
|
|
111
123
|
}
|
|
112
124
|
consumeTrigger(identifier, parameters) {
|
|
113
|
-
const
|
|
114
|
-
const
|
|
115
|
-
const
|
|
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);
|
|
116
135
|
try {
|
|
117
136
|
switch (identifier.toString()) {
|
|
118
137
|
case OnTriggerType.IDLE:
|
|
119
|
-
this.trackTrigger('idle', createIdleTrigger(parameters, sourceSpan));
|
|
138
|
+
this.trackTrigger('idle', createIdleTrigger(parameters, nameSpan, sourceSpan, prefetchSourceSpan, onSourceSpan));
|
|
120
139
|
break;
|
|
121
140
|
case OnTriggerType.TIMER:
|
|
122
|
-
this.trackTrigger('timer', createTimerTrigger(parameters, sourceSpan));
|
|
141
|
+
this.trackTrigger('timer', createTimerTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));
|
|
123
142
|
break;
|
|
124
143
|
case OnTriggerType.INTERACTION:
|
|
125
|
-
this.trackTrigger('interaction', createInteractionTrigger(parameters, sourceSpan, this.placeholder));
|
|
144
|
+
this.trackTrigger('interaction', createInteractionTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
|
|
126
145
|
break;
|
|
127
146
|
case OnTriggerType.IMMEDIATE:
|
|
128
|
-
this.trackTrigger('immediate', createImmediateTrigger(parameters, sourceSpan));
|
|
147
|
+
this.trackTrigger('immediate', createImmediateTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));
|
|
129
148
|
break;
|
|
130
149
|
case OnTriggerType.HOVER:
|
|
131
|
-
this.trackTrigger('hover', createHoverTrigger(parameters, sourceSpan, this.placeholder));
|
|
150
|
+
this.trackTrigger('hover', createHoverTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
|
|
132
151
|
break;
|
|
133
152
|
case OnTriggerType.VIEWPORT:
|
|
134
|
-
this.trackTrigger('viewport', createViewportTrigger(parameters, sourceSpan, this.placeholder));
|
|
153
|
+
this.trackTrigger('viewport', createViewportTrigger(parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan, this.placeholder));
|
|
135
154
|
break;
|
|
136
155
|
default:
|
|
137
156
|
throw new Error(`Unrecognized trigger type "${identifier}"`);
|
|
@@ -221,13 +240,13 @@ function trackTrigger(name, allTriggers, errors, trigger) {
|
|
|
221
240
|
allTriggers[name] = trigger;
|
|
222
241
|
}
|
|
223
242
|
}
|
|
224
|
-
function createIdleTrigger(parameters, sourceSpan) {
|
|
243
|
+
function createIdleTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
|
|
225
244
|
if (parameters.length > 0) {
|
|
226
245
|
throw new Error(`"${OnTriggerType.IDLE}" trigger cannot have parameters`);
|
|
227
246
|
}
|
|
228
|
-
return new t.IdleDeferredTrigger(sourceSpan);
|
|
247
|
+
return new t.IdleDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
229
248
|
}
|
|
230
|
-
function createTimerTrigger(parameters, sourceSpan) {
|
|
249
|
+
function createTimerTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
|
|
231
250
|
if (parameters.length !== 1) {
|
|
232
251
|
throw new Error(`"${OnTriggerType.TIMER}" trigger must have exactly one parameter`);
|
|
233
252
|
}
|
|
@@ -235,25 +254,25 @@ function createTimerTrigger(parameters, sourceSpan) {
|
|
|
235
254
|
if (delay === null) {
|
|
236
255
|
throw new Error(`Could not parse time value of trigger "${OnTriggerType.TIMER}"`);
|
|
237
256
|
}
|
|
238
|
-
return new t.TimerDeferredTrigger(delay, sourceSpan);
|
|
257
|
+
return new t.TimerDeferredTrigger(delay, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
239
258
|
}
|
|
240
|
-
function createImmediateTrigger(parameters, sourceSpan) {
|
|
259
|
+
function createImmediateTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan) {
|
|
241
260
|
if (parameters.length > 0) {
|
|
242
261
|
throw new Error(`"${OnTriggerType.IMMEDIATE}" trigger cannot have parameters`);
|
|
243
262
|
}
|
|
244
|
-
return new t.ImmediateDeferredTrigger(sourceSpan);
|
|
263
|
+
return new t.ImmediateDeferredTrigger(nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
245
264
|
}
|
|
246
|
-
function createHoverTrigger(parameters, sourceSpan, placeholder) {
|
|
265
|
+
function createHoverTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
|
|
247
266
|
validateReferenceBasedTrigger(OnTriggerType.HOVER, parameters, placeholder);
|
|
248
|
-
return new t.HoverDeferredTrigger(parameters[0] ?? null, sourceSpan);
|
|
267
|
+
return new t.HoverDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
249
268
|
}
|
|
250
|
-
function createInteractionTrigger(parameters, sourceSpan, placeholder) {
|
|
269
|
+
function createInteractionTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
|
|
251
270
|
validateReferenceBasedTrigger(OnTriggerType.INTERACTION, parameters, placeholder);
|
|
252
|
-
return new t.InteractionDeferredTrigger(parameters[0] ?? null, sourceSpan);
|
|
271
|
+
return new t.InteractionDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
253
272
|
}
|
|
254
|
-
function createViewportTrigger(parameters, sourceSpan, placeholder) {
|
|
273
|
+
function createViewportTrigger(parameters, nameSpan, sourceSpan, prefetchSpan, onSourceSpan, placeholder) {
|
|
255
274
|
validateReferenceBasedTrigger(OnTriggerType.VIEWPORT, parameters, placeholder);
|
|
256
|
-
return new t.ViewportDeferredTrigger(parameters[0] ?? null, sourceSpan);
|
|
275
|
+
return new t.ViewportDeferredTrigger(parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);
|
|
257
276
|
}
|
|
258
277
|
function validateReferenceBasedTrigger(type, parameters, placeholder) {
|
|
259
278
|
if (parameters.length > 1) {
|
|
@@ -292,6 +311,6 @@ export function parseDeferredTime(value) {
|
|
|
292
311
|
return null;
|
|
293
312
|
}
|
|
294
313
|
const [time, units] = match;
|
|
295
|
-
return
|
|
314
|
+
return parseFloat(time) * (units === 's' ? 1000 : 1);
|
|
296
315
|
}
|
|
297
|
-
//# 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+(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, bindingParser: BindingParser,\n    triggers: t.DeferredBlockTriggers, errors: ParseError[]): void {\n  const whenIndex = expression.indexOf('when');\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), false, sourceSpan, sourceSpan.start.offset + start);\n    trackTrigger('when', triggers, errors, new t.BoundDeferredTrigger(parsed, sourceSpan));\n  }\n}\n\n/** Parses an `on` trigger */\nexport function parseOnTrigger(\n    {expression, sourceSpan}: html.BlockParameter, triggers: t.DeferredBlockTriggers,\n    errors: ParseError[], placeholder: t.DeferredBlockPlaceholder|null): void {\n  const onIndex = expression.indexOf('on');\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 =\n        new OnTriggerParser(expression, start, sourceSpan, triggers, errors, placeholder);\n    parser.parse();\n  }\n}\n\n\nclass OnTriggerParser {\n  private index = 0;\n  private tokens: Token[];\n\n  constructor(\n      private expression: string, private start: number, private span: ParseSourceSpan,\n      private triggers: t.DeferredBlockTriggers, private errors: ParseError[],\n      private placeholder: t.DeferredBlockPlaceholder|null) {\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 startSpan = this.span.start.moveBy(this.start + identifier.index - this.tokens[0].index);\n    const endSpan = startSpan.moveBy(this.token().end - identifier.index);\n    const sourceSpan = new ParseSourceSpan(startSpan, endSpan);\n\n    try {\n      switch (identifier.toString()) {\n        case OnTriggerType.IDLE:\n          this.trackTrigger('idle', createIdleTrigger(parameters, sourceSpan));\n          break;\n\n        case OnTriggerType.TIMER:\n          this.trackTrigger('timer', createTimerTrigger(parameters, sourceSpan));\n          break;\n\n        case OnTriggerType.INTERACTION:\n          this.trackTrigger(\n              'interaction', createInteractionTrigger(parameters, sourceSpan, this.placeholder));\n          break;\n\n        case OnTriggerType.IMMEDIATE:\n          this.trackTrigger('immediate', createImmediateTrigger(parameters, sourceSpan));\n          break;\n\n        case OnTriggerType.HOVER:\n          this.trackTrigger('hover', createHoverTrigger(parameters, sourceSpan, this.placeholder));\n          break;\n\n        case OnTriggerType.VIEWPORT:\n          this.trackTrigger(\n              'viewport', createViewportTrigger(parameters, sourceSpan, this.placeholder));\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 (commaDelimStack.length > 0 &&\n          token.isCharacter(commaDelimStack[commaDelimStack.length - 1])) {\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 (this.index < this.tokens.length - 1 &&\n        !this.tokens[this.index + 1].isCharacter(chars.$COMMA)) {\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, allTriggers: t.DeferredBlockTriggers, errors: ParseError[],\n    trigger: t.DeferredTrigger) {\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[], sourceSpan: ParseSourceSpan): t.IdleDeferredTrigger {\n  if (parameters.length > 0) {\n    throw new Error(`\"${OnTriggerType.IDLE}\" trigger cannot have parameters`);\n  }\n\n  return new t.IdleDeferredTrigger(sourceSpan);\n}\n\nfunction createTimerTrigger(parameters: string[], sourceSpan: ParseSourceSpan) {\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, sourceSpan);\n}\n\nfunction createImmediateTrigger(\n    parameters: string[], sourceSpan: ParseSourceSpan): t.ImmediateDeferredTrigger {\n  if (parameters.length > 0) {\n    throw new Error(`\"${OnTriggerType.IMMEDIATE}\" trigger cannot have parameters`);\n  }\n\n  return new t.ImmediateDeferredTrigger(sourceSpan);\n}\n\nfunction createHoverTrigger(\n    parameters: string[], sourceSpan: ParseSourceSpan,\n    placeholder: t.DeferredBlockPlaceholder|null): t.HoverDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.HOVER, parameters, placeholder);\n  return new t.HoverDeferredTrigger(parameters[0] ?? null, sourceSpan);\n}\n\nfunction createInteractionTrigger(\n    parameters: string[], sourceSpan: ParseSourceSpan,\n    placeholder: t.DeferredBlockPlaceholder|null): t.InteractionDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.INTERACTION, parameters, placeholder);\n  return new t.InteractionDeferredTrigger(parameters[0] ?? null, sourceSpan);\n}\n\nfunction createViewportTrigger(\n    parameters: string[], sourceSpan: ParseSourceSpan,\n    placeholder: t.DeferredBlockPlaceholder|null): t.ViewportDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.VIEWPORT, parameters, placeholder);\n  return new t.ViewportDeferredTrigger(parameters[0] ?? null, sourceSpan);\n}\n\nfunction validateReferenceBasedTrigger(\n    type: OnTriggerType, parameters: string[], placeholder: t.DeferredBlockPlaceholder|null) {\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    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/** 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 parseInt(time) * (units === 's' ? 1000 : 1);\n}\n"]}
|
|
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, bindingParser: BindingParser,\n    triggers: t.DeferredBlockTriggers, errors: ParseError[]): void {\n  const whenIndex = expression.indexOf('when');\n  const whenSourceSpan = new ParseSourceSpan(\n      sourceSpan.start.moveBy(whenIndex), sourceSpan.start.moveBy(whenIndex + 'when'.length));\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), false, sourceSpan, sourceSpan.start.offset + start);\n    trackTrigger(\n        'when', triggers, errors,\n        new t.BoundDeferredTrigger(parsed, sourceSpan, prefetchSpan, whenSourceSpan));\n  }\n}\n\n/** Parses an `on` trigger */\nexport function parseOnTrigger(\n    {expression, sourceSpan}: html.BlockParameter, triggers: t.DeferredBlockTriggers,\n    errors: ParseError[], placeholder: t.DeferredBlockPlaceholder|null): void {\n  const onIndex = expression.indexOf('on');\n  const onSourceSpan = new ParseSourceSpan(\n      sourceSpan.start.moveBy(onIndex), sourceSpan.start.moveBy(onIndex + 'on'.length));\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, start, sourceSpan, triggers, errors, placeholder, prefetchSpan, onSourceSpan);\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\n\nclass OnTriggerParser {\n  private index = 0;\n  private tokens: Token[];\n\n  constructor(\n      private expression: string, private start: number, private span: ParseSourceSpan,\n      private triggers: t.DeferredBlockTriggers, private errors: ParseError[],\n      private placeholder: t.DeferredBlockPlaceholder|null,\n      private prefetchSpan: ParseSourceSpan|null, private onSourceSpan: ParseSourceSpan) {\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 =\n        this.span.start.moveBy(this.start + identifier.index - this.tokens[0].index);\n    const nameSpan = new ParseSourceSpan(\n        triggerNameStartSpan, triggerNameStartSpan.moveBy(identifier.strValue.length));\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 =\n        new ParseSourceSpan(isFirstTrigger ? this.span.start : triggerNameStartSpan, endSpan);\n\n    try {\n      switch (identifier.toString()) {\n        case OnTriggerType.IDLE:\n          this.trackTrigger(\n              'idle',\n              createIdleTrigger(\n                  parameters, nameSpan, sourceSpan, prefetchSourceSpan, onSourceSpan));\n          break;\n\n        case OnTriggerType.TIMER:\n          this.trackTrigger(\n              'timer',\n              createTimerTrigger(\n                  parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));\n          break;\n\n        case OnTriggerType.INTERACTION:\n          this.trackTrigger(\n              'interaction',\n              createInteractionTrigger(\n                  parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan,\n                  this.placeholder));\n          break;\n\n        case OnTriggerType.IMMEDIATE:\n          this.trackTrigger(\n              'immediate',\n              createImmediateTrigger(\n                  parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan));\n          break;\n\n        case OnTriggerType.HOVER:\n          this.trackTrigger(\n              'hover',\n              createHoverTrigger(\n                  parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan,\n                  this.placeholder));\n          break;\n\n        case OnTriggerType.VIEWPORT:\n          this.trackTrigger(\n              'viewport',\n              createViewportTrigger(\n                  parameters, nameSpan, sourceSpan, this.prefetchSpan, this.onSourceSpan,\n                  this.placeholder));\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 (commaDelimStack.length > 0 &&\n          token.isCharacter(commaDelimStack[commaDelimStack.length - 1])) {\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 (this.index < this.tokens.length - 1 &&\n        !this.tokens[this.index + 1].isCharacter(chars.$COMMA)) {\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, allTriggers: t.DeferredBlockTriggers, errors: ParseError[],\n    trigger: t.DeferredTrigger) {\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[], nameSpan: ParseSourceSpan, sourceSpan: ParseSourceSpan,\n    prefetchSpan: ParseSourceSpan|null, onSourceSpan: ParseSourceSpan|null,\n    placeholder: t.DeferredBlockPlaceholder|null): t.HoverDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.HOVER, parameters, placeholder);\n  return new t.HoverDeferredTrigger(\n      parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction createInteractionTrigger(\n    parameters: string[], nameSpan: ParseSourceSpan, sourceSpan: ParseSourceSpan,\n    prefetchSpan: ParseSourceSpan|null, onSourceSpan: ParseSourceSpan|null,\n    placeholder: t.DeferredBlockPlaceholder|null): t.InteractionDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.INTERACTION, parameters, placeholder);\n  return new t.InteractionDeferredTrigger(\n      parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction createViewportTrigger(\n    parameters: string[], nameSpan: ParseSourceSpan, sourceSpan: ParseSourceSpan,\n    prefetchSpan: ParseSourceSpan|null, onSourceSpan: ParseSourceSpan|null,\n    placeholder: t.DeferredBlockPlaceholder|null): t.ViewportDeferredTrigger {\n  validateReferenceBasedTrigger(OnTriggerType.VIEWPORT, parameters, placeholder);\n  return new t.ViewportDeferredTrigger(\n      parameters[0] ?? null, nameSpan, sourceSpan, prefetchSpan, onSourceSpan);\n}\n\nfunction validateReferenceBasedTrigger(\n    type: OnTriggerType, parameters: string[], placeholder: t.DeferredBlockPlaceholder|null) {\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    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/** 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"]}
|
|
@@ -6,4 +6,4 @@
|
|
|
6
6
|
* found in the LICENSE file at https://angular.io/license
|
|
7
7
|
*/
|
|
8
8
|
export {};
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"t2_api.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler/src/render3/view/t2_api.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG","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 {AST} from '../../expression_parser/ast';\nimport {BoundAttribute, BoundEvent, DeferredBlock, DeferredBlockError, DeferredBlockLoading, DeferredBlockPlaceholder, DeferredTrigger, Element, ForLoopBlock, ForLoopBlockEmpty, IfBlockBranch, Node, Reference, SwitchBlockCase, Template, TextAttribute, Variable} from '../r3_ast';\n\n/** Node that has a `Scope` associated with it. */\nexport type ScopedNode = Template|SwitchBlockCase|IfBlockBranch|ForLoopBlock|ForLoopBlockEmpty|\n    DeferredBlock|DeferredBlockError|DeferredBlockLoading|DeferredBlockPlaceholder;\n\n/** Possible values that a reference can be resolved to. */\nexport type ReferenceTarget<DirectiveT> = {\n  directive: DirectiveT,\n  node: Element|Template\n}|Element|Template;\n\n/*\n * t2 is the replacement for the `TemplateDefinitionBuilder`. It handles the operations of\n * analyzing Angular templates, extracting semantic info, and ultimately producing a template\n * definition function which renders the template using Ivy instructions.\n *\n * t2 data is also utilized by the template type-checking facilities to understand a template enough\n * to generate type-checking code for it.\n */\n\n/**\n * A logical target for analysis, which could contain a template or other types of bindings.\n */\nexport interface Target {\n  template?: Node[];\n}\n\n/**\n * A data structure which can indicate whether a given property name is present or not.\n *\n * This is used to represent the set of inputs or outputs present on a directive, and allows the\n * binder to query for the presence of a mapping for property names.\n */\nexport interface InputOutputPropertySet {\n  hasBindingPropertyName(propertyName: string): boolean;\n}\n\n/**\n * A data structure which captures the animation trigger names that are statically resolvable\n * and whether some names could not be statically evaluated.\n */\nexport interface AnimationTriggerNames {\n  includesDynamicAnimations: boolean;\n  staticTriggerNames: string[];\n}\n\n/**\n * Metadata regarding a directive that's needed to match it against template elements. This is\n * provided by a consumer of the t2 APIs.\n */\nexport interface DirectiveMeta {\n  /**\n   * Name of the directive class (used for debugging).\n   */\n  name: string;\n\n  /** The selector for the directive or `null` if there isn't one. */\n  selector: string|null;\n\n  /**\n   * Whether the directive is a component.\n   */\n  isComponent: boolean;\n\n  /**\n   * Set of inputs which this directive claims.\n   *\n   * Goes from property names to field names.\n   */\n  inputs: InputOutputPropertySet;\n\n  /**\n   * Set of outputs which this directive claims.\n   *\n   * Goes from property names to field names.\n   */\n  outputs: InputOutputPropertySet;\n\n  /**\n   * Name under which the directive is exported, if any (exportAs in Angular).\n   *\n   * Null otherwise\n   */\n  exportAs: string[]|null;\n\n  isStructural: boolean;\n\n  /**\n   * The name of animations that the user defines in the component.\n   * Only includes the animation names.\n   */\n  animationTriggerNames: AnimationTriggerNames|null;\n}\n\n/**\n * Interface to the binding API, which processes a template and returns an object similar to the\n * `ts.TypeChecker`.\n *\n * The returned `BoundTarget` has an API for extracting information about the processed target.\n */\nexport interface TargetBinder<D extends DirectiveMeta> {\n  bind(target: Target): BoundTarget<D>;\n}\n\n/**\n * Result of performing the binding operation against a `Target`.\n *\n * The original `Target` is accessible, as well as a suite of methods for extracting binding\n * information regarding the `Target`.\n *\n * @param DirectiveT directive metadata type\n */\nexport interface BoundTarget<DirectiveT extends DirectiveMeta> {\n  /**\n   * Get the original `Target` that was bound.\n   */\n  readonly target: Target;\n\n  /**\n   * For a given template node (either an `Element` or a `Template`), get the set of directives\n   * which matched the node, if any.\n   */\n  getDirectivesOfNode(node: Element|Template): DirectiveT[]|null;\n\n  /**\n   * For a given `Reference`, get the reference's target - either an `Element`, a `Template`, or\n   * a directive on a particular node.\n   */\n  getReferenceTarget(ref: Reference): ReferenceTarget<DirectiveT>|null;\n\n  /**\n   * For a given binding, get the entity to which the binding is being made.\n   *\n   * This will either be a directive or the node itself.\n   */\n  getConsumerOfBinding(binding: BoundAttribute|BoundEvent|TextAttribute): DirectiveT|Element\n      |Template|null;\n\n  /**\n   * If the given `AST` expression refers to a `Reference` or `Variable` within the `Target`, then\n   * return that.\n   *\n   * Otherwise, returns `null`.\n   *\n   * This is only defined for `AST` expressions that read or write to a property of an\n   * `ImplicitReceiver`.\n   */\n  getExpressionTarget(expr: AST): Reference|Variable|null;\n\n  /**\n   * Given a particular `Reference` or `Variable`, get the `ScopedNode` which created it.\n   *\n   * All `Variable`s are defined on node, so this will always return a value for a `Variable`\n   * from the `Target`. Returns `null` otherwise.\n   */\n  getDefinitionNodeOfSymbol(symbol: Reference|Variable): ScopedNode|null;\n\n  /**\n   * Get the nesting level of a particular `ScopedNode`.\n   *\n   * This starts at 1 for top-level nodes within the `Target` and increases for nodes\n   * nested at deeper levels.\n   */\n  getNestingLevel(node: ScopedNode): number;\n\n  /**\n   * Get all `Reference`s and `Variables` visible within the given `ScopedNode` (or at the top\n   * level, if `null` is passed).\n   */\n  getEntitiesInScope(node: ScopedNode|null): ReadonlySet<Reference|Variable>
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"t2_api.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler/src/render3/view/t2_api.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG","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 {AST} from '../../expression_parser/ast';\nimport {BoundAttribute, BoundEvent, DeferredBlock, DeferredBlockError, DeferredBlockLoading, DeferredBlockPlaceholder, DeferredTrigger, Element, ForLoopBlock, ForLoopBlockEmpty, IfBlockBranch, Node, Reference, SwitchBlockCase, Template, TextAttribute, Variable} from '../r3_ast';\n\n/** Node that has a `Scope` associated with it. */\nexport type ScopedNode = Template|SwitchBlockCase|IfBlockBranch|ForLoopBlock|ForLoopBlockEmpty|\n    DeferredBlock|DeferredBlockError|DeferredBlockLoading|DeferredBlockPlaceholder;\n\n/** Possible values that a reference can be resolved to. */\nexport type ReferenceTarget<DirectiveT> = {\n  directive: DirectiveT,\n  node: Element|Template\n}|Element|Template;\n\n/*\n * t2 is the replacement for the `TemplateDefinitionBuilder`. It handles the operations of\n * analyzing Angular templates, extracting semantic info, and ultimately producing a template\n * definition function which renders the template using Ivy instructions.\n *\n * t2 data is also utilized by the template type-checking facilities to understand a template enough\n * to generate type-checking code for it.\n */\n\n/**\n * A logical target for analysis, which could contain a template or other types of bindings.\n */\nexport interface Target {\n  template?: Node[];\n}\n\n/**\n * A data structure which can indicate whether a given property name is present or not.\n *\n * This is used to represent the set of inputs or outputs present on a directive, and allows the\n * binder to query for the presence of a mapping for property names.\n */\nexport interface InputOutputPropertySet {\n  hasBindingPropertyName(propertyName: string): boolean;\n}\n\n/**\n * A data structure which captures the animation trigger names that are statically resolvable\n * and whether some names could not be statically evaluated.\n */\nexport interface AnimationTriggerNames {\n  includesDynamicAnimations: boolean;\n  staticTriggerNames: string[];\n}\n\n/**\n * Metadata regarding a directive that's needed to match it against template elements. This is\n * provided by a consumer of the t2 APIs.\n */\nexport interface DirectiveMeta {\n  /**\n   * Name of the directive class (used for debugging).\n   */\n  name: string;\n\n  /** The selector for the directive or `null` if there isn't one. */\n  selector: string|null;\n\n  /**\n   * Whether the directive is a component.\n   */\n  isComponent: boolean;\n\n  /**\n   * Set of inputs which this directive claims.\n   *\n   * Goes from property names to field names.\n   */\n  inputs: InputOutputPropertySet;\n\n  /**\n   * Set of outputs which this directive claims.\n   *\n   * Goes from property names to field names.\n   */\n  outputs: InputOutputPropertySet;\n\n  /**\n   * Name under which the directive is exported, if any (exportAs in Angular).\n   *\n   * Null otherwise\n   */\n  exportAs: string[]|null;\n\n  isStructural: boolean;\n\n  /**\n   * The name of animations that the user defines in the component.\n   * Only includes the animation names.\n   */\n  animationTriggerNames: AnimationTriggerNames|null;\n}\n\n/**\n * Interface to the binding API, which processes a template and returns an object similar to the\n * `ts.TypeChecker`.\n *\n * The returned `BoundTarget` has an API for extracting information about the processed target.\n */\nexport interface TargetBinder<D extends DirectiveMeta> {\n  bind(target: Target): BoundTarget<D>;\n}\n\n/**\n * Result of performing the binding operation against a `Target`.\n *\n * The original `Target` is accessible, as well as a suite of methods for extracting binding\n * information regarding the `Target`.\n *\n * @param DirectiveT directive metadata type\n */\nexport interface BoundTarget<DirectiveT extends DirectiveMeta> {\n  /**\n   * Get the original `Target` that was bound.\n   */\n  readonly target: Target;\n\n  /**\n   * For a given template node (either an `Element` or a `Template`), get the set of directives\n   * which matched the node, if any.\n   */\n  getDirectivesOfNode(node: Element|Template): DirectiveT[]|null;\n\n  /**\n   * For a given `Reference`, get the reference's target - either an `Element`, a `Template`, or\n   * a directive on a particular node.\n   */\n  getReferenceTarget(ref: Reference): ReferenceTarget<DirectiveT>|null;\n\n  /**\n   * For a given binding, get the entity to which the binding is being made.\n   *\n   * This will either be a directive or the node itself.\n   */\n  getConsumerOfBinding(binding: BoundAttribute|BoundEvent|TextAttribute): DirectiveT|Element\n      |Template|null;\n\n  /**\n   * If the given `AST` expression refers to a `Reference` or `Variable` within the `Target`, then\n   * return that.\n   *\n   * Otherwise, returns `null`.\n   *\n   * This is only defined for `AST` expressions that read or write to a property of an\n   * `ImplicitReceiver`.\n   */\n  getExpressionTarget(expr: AST): Reference|Variable|null;\n\n  /**\n   * Given a particular `Reference` or `Variable`, get the `ScopedNode` which created it.\n   *\n   * All `Variable`s are defined on node, so this will always return a value for a `Variable`\n   * from the `Target`. Returns `null` otherwise.\n   */\n  getDefinitionNodeOfSymbol(symbol: Reference|Variable): ScopedNode|null;\n\n  /**\n   * Get the nesting level of a particular `ScopedNode`.\n   *\n   * This starts at 1 for top-level nodes within the `Target` and increases for nodes\n   * nested at deeper levels.\n   */\n  getNestingLevel(node: ScopedNode): number;\n\n  /**\n   * Get all `Reference`s and `Variables` visible within the given `ScopedNode` (or at the top\n   * level, if `null` is passed).\n   */\n  getEntitiesInScope(node: ScopedNode|null): ReadonlySet<Reference|Variable>;\n\n  /**\n   * Get a list of all the directives used by the target,\n   * including directives from `@defer` blocks.\n   */\n  getUsedDirectives(): DirectiveT[];\n\n  /**\n   * Get a list of eagerly used directives from the target.\n   * Note: this list *excludes* directives from `@defer` blocks.\n   */\n  getEagerlyUsedDirectives(): DirectiveT[];\n\n  /**\n   * Get a list of all the pipes used by the target,\n   * including pipes from `@defer` blocks.\n   */\n  getUsedPipes(): string[];\n\n  /**\n   * Get a list of eagerly used pipes from the target.\n   * Note: this list *excludes* pipes from `@defer` blocks.\n   */\n  getEagerlyUsedPipes(): string[];\n\n  /**\n   * Get a list of all `@defer` blocks used by the target.\n   */\n  getDeferBlocks(): DeferredBlock[];\n\n  /**\n   * Gets the element that a specific deferred block trigger is targeting.\n   * @param block Block that the trigger belongs to.\n   * @param trigger Trigger whose target is being looked up.\n   */\n  getDeferredTriggerTarget(block: DeferredBlock, trigger: DeferredTrigger): Element|null;\n}\n"]}
|