@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.
- package/LICENSE +21 -0
- package/fesm2022/compiler.mjs +5465 -4675
- package/fesm2022/compiler.mjs.map +1 -1
- package/index.d.ts +115 -15
- package/package.json +2 -4
- package/esm2022/compiler.mjs +0 -13
- package/esm2022/index.mjs +0 -13
- package/esm2022/public_api.mjs +0 -15
- package/esm2022/src/assertions.mjs +0 -31
- package/esm2022/src/chars.mjs +0 -90
- package/esm2022/src/compiler.mjs +0 -88
- package/esm2022/src/compiler_facade_interface.mjs +0 -29
- package/esm2022/src/config.mjs +0 -20
- package/esm2022/src/constant_pool.mjs +0 -285
- package/esm2022/src/core.mjs +0 -96
- package/esm2022/src/expression_parser/ast.mjs +0 -755
- package/esm2022/src/expression_parser/lexer.mjs +0 -421
- package/esm2022/src/expression_parser/parser.mjs +0 -1222
- package/esm2022/src/i18n/digest.mjs +0 -322
- package/esm2022/src/i18n/extractor_merger.mjs +0 -409
- package/esm2022/src/i18n/i18n_ast.mjs +0 -212
- package/esm2022/src/i18n/i18n_html_parser.mjs +0 -54
- package/esm2022/src/i18n/i18n_parser.mjs +0 -247
- package/esm2022/src/i18n/index.mjs +0 -16
- package/esm2022/src/i18n/message_bundle.mjs +0 -93
- package/esm2022/src/i18n/parse_util.mjs +0 -17
- package/esm2022/src/i18n/serializers/placeholder.mjs +0 -139
- package/esm2022/src/i18n/serializers/serializer.mjs +0 -77
- package/esm2022/src/i18n/serializers/xliff.mjs +0 -300
- package/esm2022/src/i18n/serializers/xliff2.mjs +0 -349
- package/esm2022/src/i18n/serializers/xmb.mjs +0 -188
- package/esm2022/src/i18n/serializers/xml_helper.mjs +0 -94
- package/esm2022/src/i18n/serializers/xtb.mjs +0 -186
- package/esm2022/src/i18n/translation_bundle.mjs +0 -164
- package/esm2022/src/injectable_compiler_2.mjs +0 -138
- package/esm2022/src/jit_compiler_facade.mjs +0 -681
- package/esm2022/src/ml_parser/ast.mjs +0 -167
- package/esm2022/src/ml_parser/defaults.mjs +0 -24
- package/esm2022/src/ml_parser/entities.mjs +0 -2144
- package/esm2022/src/ml_parser/html_parser.mjs +0 -18
- package/esm2022/src/ml_parser/html_tags.mjs +0 -156
- package/esm2022/src/ml_parser/html_whitespaces.mjs +0 -107
- package/esm2022/src/ml_parser/icu_ast_expander.mjs +0 -125
- package/esm2022/src/ml_parser/lexer.mjs +0 -1210
- package/esm2022/src/ml_parser/parser.mjs +0 -539
- package/esm2022/src/ml_parser/tags.mjs +0 -47
- package/esm2022/src/ml_parser/tokens.mjs +0 -9
- package/esm2022/src/ml_parser/xml_parser.mjs +0 -19
- package/esm2022/src/ml_parser/xml_tags.mjs +0 -32
- package/esm2022/src/output/abstract_emitter.mjs +0 -524
- package/esm2022/src/output/abstract_js_emitter.mjs +0 -130
- package/esm2022/src/output/map_util.mjs +0 -19
- package/esm2022/src/output/output_ast.mjs +0 -1301
- package/esm2022/src/output/output_jit.mjs +0 -138
- package/esm2022/src/output/output_jit_trusted_types.mjs +0 -100
- package/esm2022/src/output/source_map.mjs +0 -150
- package/esm2022/src/parse_util.mjs +0 -198
- package/esm2022/src/render3/partial/api.mjs +0 -9
- package/esm2022/src/render3/partial/class_metadata.mjs +0 -52
- package/esm2022/src/render3/partial/component.mjs +0 -197
- package/esm2022/src/render3/partial/directive.mjs +0 -264
- package/esm2022/src/render3/partial/factory.mjs +0 -35
- package/esm2022/src/render3/partial/injectable.mjs +0 -67
- package/esm2022/src/render3/partial/injector.mjs +0 -41
- package/esm2022/src/render3/partial/ng_module.mjs +0 -63
- package/esm2022/src/render3/partial/pipe.mjs +0 -50
- package/esm2022/src/render3/partial/util.mjs +0 -79
- package/esm2022/src/render3/r3_ast.mjs +0 -493
- package/esm2022/src/render3/r3_class_debug_info_compiler.mjs +0 -35
- package/esm2022/src/render3/r3_class_metadata_compiler.mjs +0 -92
- package/esm2022/src/render3/r3_control_flow.mjs +0 -421
- package/esm2022/src/render3/r3_deferred_blocks.mjs +0 -170
- package/esm2022/src/render3/r3_deferred_triggers.mjs +0 -316
- package/esm2022/src/render3/r3_factory.mjs +0 -200
- package/esm2022/src/render3/r3_identifiers.mjs +0 -480
- package/esm2022/src/render3/r3_injector_compiler.mjs +0 -28
- package/esm2022/src/render3/r3_jit.mjs +0 -29
- package/esm2022/src/render3/r3_module_compiler.mjs +0 -211
- package/esm2022/src/render3/r3_pipe_compiler.mjs +0 -35
- package/esm2022/src/render3/r3_template_transform.mjs +0 -560
- package/esm2022/src/render3/util.mjs +0 -90
- package/esm2022/src/render3/view/api.mjs +0 -14
- package/esm2022/src/render3/view/compiler.mjs +0 -546
- package/esm2022/src/render3/view/i18n/get_msg_utils.mjs +0 -113
- package/esm2022/src/render3/view/i18n/icu_serializer.mjs +0 -43
- package/esm2022/src/render3/view/i18n/localize_utils.mjs +0 -118
- package/esm2022/src/render3/view/i18n/meta.mjs +0 -259
- package/esm2022/src/render3/view/i18n/util.mjs +0 -70
- package/esm2022/src/render3/view/query_generation.mjs +0 -182
- package/esm2022/src/render3/view/t2_api.mjs +0 -9
- package/esm2022/src/render3/view/t2_binder.mjs +0 -823
- package/esm2022/src/render3/view/template.mjs +0 -113
- package/esm2022/src/render3/view/util.mjs +0 -192
- package/esm2022/src/resource_loader.mjs +0 -17
- package/esm2022/src/schema/dom_element_schema_registry.mjs +0 -471
- package/esm2022/src/schema/dom_security_schema.mjs +0 -96
- package/esm2022/src/schema/element_schema_registry.mjs +0 -10
- package/esm2022/src/schema/trusted_types_sinks.mjs +0 -41
- package/esm2022/src/selector.mjs +0 -400
- package/esm2022/src/shadow_css.mjs +0 -1053
- package/esm2022/src/style_url_resolver.mjs +0 -17
- package/esm2022/src/template/pipeline/ir/index.mjs +0 -18
- package/esm2022/src/template/pipeline/ir/src/enums.mjs +0 -503
- package/esm2022/src/template/pipeline/ir/src/expression.mjs +0 -997
- package/esm2022/src/template/pipeline/ir/src/handle.mjs +0 -13
- package/esm2022/src/template/pipeline/ir/src/operations.mjs +0 -261
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +0 -405
- package/esm2022/src/template/pipeline/ir/src/ops/host.mjs +0 -25
- package/esm2022/src/template/pipeline/ir/src/ops/shared.mjs +0 -43
- package/esm2022/src/template/pipeline/ir/src/ops/update.mjs +0 -277
- package/esm2022/src/template/pipeline/ir/src/traits.mjs +0 -64
- package/esm2022/src/template/pipeline/ir/src/variable.mjs +0 -13
- package/esm2022/src/template/pipeline/src/compilation.mjs +0 -194
- package/esm2022/src/template/pipeline/src/conversion.mjs +0 -64
- package/esm2022/src/template/pipeline/src/emit.mjs +0 -238
- package/esm2022/src/template/pipeline/src/ingest.mjs +0 -1058
- package/esm2022/src/template/pipeline/src/instruction.mjs +0 -564
- package/esm2022/src/template/pipeline/src/phases/any_cast.mjs +0 -32
- package/esm2022/src/template/pipeline/src/phases/apply_i18n_expressions.mjs +0 -63
- package/esm2022/src/template/pipeline/src/phases/assign_i18n_slot_dependencies.mjs +0 -63
- package/esm2022/src/template/pipeline/src/phases/attribute_extraction.mjs +0 -138
- package/esm2022/src/template/pipeline/src/phases/binding_specialization.mjs +0 -75
- package/esm2022/src/template/pipeline/src/phases/chaining.mjs +0 -112
- package/esm2022/src/template/pipeline/src/phases/collapse_singleton_interpolations.mjs +0 -32
- package/esm2022/src/template/pipeline/src/phases/conditionals.mjs +0 -58
- package/esm2022/src/template/pipeline/src/phases/const_collection.mjs +0 -214
- package/esm2022/src/template/pipeline/src/phases/convert_i18n_bindings.mjs +0 -52
- package/esm2022/src/template/pipeline/src/phases/create_i18n_contexts.mjs +0 -105
- package/esm2022/src/template/pipeline/src/phases/deduplicate_text_bindings.mjs +0 -40
- package/esm2022/src/template/pipeline/src/phases/defer_configs.mjs +0 -29
- package/esm2022/src/template/pipeline/src/phases/defer_resolve_targets.mjs +0 -113
- package/esm2022/src/template/pipeline/src/phases/empty_elements.mjs +0 -47
- package/esm2022/src/template/pipeline/src/phases/expand_safe_reads.mjs +0 -192
- package/esm2022/src/template/pipeline/src/phases/extract_i18n_messages.mjs +0 -218
- package/esm2022/src/template/pipeline/src/phases/generate_advance.mjs +0 -66
- package/esm2022/src/template/pipeline/src/phases/generate_local_let_references.mjs +0 -29
- package/esm2022/src/template/pipeline/src/phases/generate_projection_def.mjs +0 -46
- package/esm2022/src/template/pipeline/src/phases/generate_variables.mjs +0 -170
- package/esm2022/src/template/pipeline/src/phases/has_const_expression_collection.mjs +0 -28
- package/esm2022/src/template/pipeline/src/phases/host_style_property_parsing.mjs +0 -81
- package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +0 -309
- package/esm2022/src/template/pipeline/src/phases/i18n_text_extraction.mjs +0 -97
- package/esm2022/src/template/pipeline/src/phases/local_refs.mjs +0 -43
- package/esm2022/src/template/pipeline/src/phases/namespace.mjs +0 -26
- package/esm2022/src/template/pipeline/src/phases/naming.mjs +0 -179
- package/esm2022/src/template/pipeline/src/phases/next_context_merging.mjs +0 -75
- package/esm2022/src/template/pipeline/src/phases/ng_container.mjs +0 -29
- package/esm2022/src/template/pipeline/src/phases/nonbindable.mjs +0 -47
- package/esm2022/src/template/pipeline/src/phases/nullish_coalescing.mjs +0 -33
- package/esm2022/src/template/pipeline/src/phases/ordering.mjs +0 -140
- package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +0 -137
- package/esm2022/src/template/pipeline/src/phases/phase_remove_content_selectors.mjs +0 -42
- package/esm2022/src/template/pipeline/src/phases/pipe_creation.mjs +0 -74
- package/esm2022/src/template/pipeline/src/phases/pipe_variadic.mjs +0 -30
- package/esm2022/src/template/pipeline/src/phases/propagate_i18n_blocks.mjs +0 -81
- package/esm2022/src/template/pipeline/src/phases/pure_function_extraction.mjs +0 -55
- package/esm2022/src/template/pipeline/src/phases/pure_literal_structures.mjs +0 -58
- package/esm2022/src/template/pipeline/src/phases/reify.mjs +0 -433
- package/esm2022/src/template/pipeline/src/phases/remove_empty_bindings.mjs +0 -31
- package/esm2022/src/template/pipeline/src/phases/remove_i18n_contexts.mjs +0 -27
- package/esm2022/src/template/pipeline/src/phases/remove_illegal_let_references.mjs +0 -36
- package/esm2022/src/template/pipeline/src/phases/remove_unused_i18n_attrs.mjs +0 -33
- package/esm2022/src/template/pipeline/src/phases/resolve_contexts.mjs +0 -60
- package/esm2022/src/template/pipeline/src/phases/resolve_defer_deps_fns.mjs +0 -31
- package/esm2022/src/template/pipeline/src/phases/resolve_dollar_event.mjs +0 -36
- package/esm2022/src/template/pipeline/src/phases/resolve_i18n_element_placeholders.mjs +0 -266
- package/esm2022/src/template/pipeline/src/phases/resolve_i18n_expression_placeholders.mjs +0 -70
- package/esm2022/src/template/pipeline/src/phases/resolve_names.mjs +0 -124
- package/esm2022/src/template/pipeline/src/phases/resolve_sanitizers.mjs +0 -123
- package/esm2022/src/template/pipeline/src/phases/save_restore_view.mjs +0 -64
- package/esm2022/src/template/pipeline/src/phases/slot_allocation.mjs +0 -62
- package/esm2022/src/template/pipeline/src/phases/store_let_optimization.mjs +0 -35
- package/esm2022/src/template/pipeline/src/phases/style_binding_specialization.mjs +0 -42
- package/esm2022/src/template/pipeline/src/phases/temporary_variables.mjs +0 -87
- package/esm2022/src/template/pipeline/src/phases/track_fn_generation.mjs +0 -52
- package/esm2022/src/template/pipeline/src/phases/track_fn_optimization.mjs +0 -88
- package/esm2022/src/template/pipeline/src/phases/track_variables.mjs +0 -36
- package/esm2022/src/template/pipeline/src/phases/transform_two_way_binding_set.mjs +0 -43
- package/esm2022/src/template/pipeline/src/phases/var_counting.mjs +0 -155
- package/esm2022/src/template/pipeline/src/phases/variable_optimization.mjs +0 -423
- package/esm2022/src/template/pipeline/src/phases/wrap_icus.mjs +0 -41
- package/esm2022/src/template/pipeline/src/util/elements.mjs +0 -29
- package/esm2022/src/template_parser/binding_parser.mjs +0 -465
- package/esm2022/src/template_parser/template_preparser.mjs +0 -84
- package/esm2022/src/util.mjs +0 -124
- package/esm2022/src/version.mjs +0 -15
|
@@ -1,823 +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 { AST, ImplicitReceiver, RecursiveAstVisitor, ThisReceiver, } from '../../expression_parser/ast';
|
|
9
|
-
import { Comment, Content, DeferredBlock, DeferredBlockError, DeferredBlockLoading, DeferredBlockPlaceholder, Element, ForLoopBlock, ForLoopBlockEmpty, HoverDeferredTrigger, IfBlockBranch, InteractionDeferredTrigger, LetDeclaration, Reference, SwitchBlockCase, Template, ViewportDeferredTrigger, } from '../r3_ast';
|
|
10
|
-
import { createCssSelectorFromNode } from './util';
|
|
11
|
-
/**
|
|
12
|
-
* Processes `Target`s with a given set of directives and performs a binding operation, which
|
|
13
|
-
* returns an object similar to TypeScript's `ts.TypeChecker` that contains knowledge about the
|
|
14
|
-
* target.
|
|
15
|
-
*/
|
|
16
|
-
export class R3TargetBinder {
|
|
17
|
-
constructor(directiveMatcher) {
|
|
18
|
-
this.directiveMatcher = directiveMatcher;
|
|
19
|
-
}
|
|
20
|
-
/**
|
|
21
|
-
* Perform a binding operation on the given `Target` and return a `BoundTarget` which contains
|
|
22
|
-
* metadata about the types referenced in the template.
|
|
23
|
-
*/
|
|
24
|
-
bind(target) {
|
|
25
|
-
if (!target.template) {
|
|
26
|
-
// TODO(alxhub): handle targets which contain things like HostBindings, etc.
|
|
27
|
-
throw new Error('Binding without a template not yet supported');
|
|
28
|
-
}
|
|
29
|
-
// First, parse the template into a `Scope` structure. This operation captures the syntactic
|
|
30
|
-
// scopes in the template and makes them available for later use.
|
|
31
|
-
const scope = Scope.apply(target.template);
|
|
32
|
-
// Use the `Scope` to extract the entities present at every level of the template.
|
|
33
|
-
const scopedNodeEntities = extractScopedNodeEntities(scope);
|
|
34
|
-
// Next, perform directive matching on the template using the `DirectiveBinder`. This returns:
|
|
35
|
-
// - directives: Map of nodes (elements & ng-templates) to the directives on them.
|
|
36
|
-
// - bindings: Map of inputs, outputs, and attributes to the directive/element that claims
|
|
37
|
-
// them. TODO(alxhub): handle multiple directives claiming an input/output/etc.
|
|
38
|
-
// - references: Map of #references to their targets.
|
|
39
|
-
const { directives, eagerDirectives, bindings, references } = DirectiveBinder.apply(target.template, this.directiveMatcher);
|
|
40
|
-
// Finally, run the TemplateBinder to bind references, variables, and other entities within the
|
|
41
|
-
// template. This extracts all the metadata that doesn't depend on directive matching.
|
|
42
|
-
const { expressions, symbols, nestingLevel, usedPipes, eagerPipes, deferBlocks } = TemplateBinder.applyWithScope(target.template, scope);
|
|
43
|
-
return new R3BoundTarget(target, directives, eagerDirectives, bindings, references, expressions, symbols, nestingLevel, scopedNodeEntities, usedPipes, eagerPipes, deferBlocks);
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
/**
|
|
47
|
-
* Represents a binding scope within a template.
|
|
48
|
-
*
|
|
49
|
-
* Any variables, references, or other named entities declared within the template will
|
|
50
|
-
* be captured and available by name in `namedEntities`. Additionally, child templates will
|
|
51
|
-
* be analyzed and have their child `Scope`s available in `childScopes`.
|
|
52
|
-
*/
|
|
53
|
-
class Scope {
|
|
54
|
-
constructor(parentScope, rootNode) {
|
|
55
|
-
this.parentScope = parentScope;
|
|
56
|
-
this.rootNode = rootNode;
|
|
57
|
-
/**
|
|
58
|
-
* Named members of the `Scope`, such as `Reference`s or `Variable`s.
|
|
59
|
-
*/
|
|
60
|
-
this.namedEntities = new Map();
|
|
61
|
-
/**
|
|
62
|
-
* Set of elements that belong to this scope.
|
|
63
|
-
*/
|
|
64
|
-
this.elementsInScope = new Set();
|
|
65
|
-
/**
|
|
66
|
-
* Child `Scope`s for immediately nested `ScopedNode`s.
|
|
67
|
-
*/
|
|
68
|
-
this.childScopes = new Map();
|
|
69
|
-
this.isDeferred =
|
|
70
|
-
parentScope !== null && parentScope.isDeferred ? true : rootNode instanceof DeferredBlock;
|
|
71
|
-
}
|
|
72
|
-
static newRootScope() {
|
|
73
|
-
return new Scope(null, null);
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Process a template (either as a `Template` sub-template with variables, or a plain array of
|
|
77
|
-
* template `Node`s) and construct its `Scope`.
|
|
78
|
-
*/
|
|
79
|
-
static apply(template) {
|
|
80
|
-
const scope = Scope.newRootScope();
|
|
81
|
-
scope.ingest(template);
|
|
82
|
-
return scope;
|
|
83
|
-
}
|
|
84
|
-
/**
|
|
85
|
-
* Internal method to process the scoped node and populate the `Scope`.
|
|
86
|
-
*/
|
|
87
|
-
ingest(nodeOrNodes) {
|
|
88
|
-
if (nodeOrNodes instanceof Template) {
|
|
89
|
-
// Variables on an <ng-template> are defined in the inner scope.
|
|
90
|
-
nodeOrNodes.variables.forEach((node) => this.visitVariable(node));
|
|
91
|
-
// Process the nodes of the template.
|
|
92
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
93
|
-
}
|
|
94
|
-
else if (nodeOrNodes instanceof IfBlockBranch) {
|
|
95
|
-
if (nodeOrNodes.expressionAlias !== null) {
|
|
96
|
-
this.visitVariable(nodeOrNodes.expressionAlias);
|
|
97
|
-
}
|
|
98
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
99
|
-
}
|
|
100
|
-
else if (nodeOrNodes instanceof ForLoopBlock) {
|
|
101
|
-
this.visitVariable(nodeOrNodes.item);
|
|
102
|
-
nodeOrNodes.contextVariables.forEach((v) => this.visitVariable(v));
|
|
103
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
104
|
-
}
|
|
105
|
-
else if (nodeOrNodes instanceof SwitchBlockCase ||
|
|
106
|
-
nodeOrNodes instanceof ForLoopBlockEmpty ||
|
|
107
|
-
nodeOrNodes instanceof DeferredBlock ||
|
|
108
|
-
nodeOrNodes instanceof DeferredBlockError ||
|
|
109
|
-
nodeOrNodes instanceof DeferredBlockPlaceholder ||
|
|
110
|
-
nodeOrNodes instanceof DeferredBlockLoading ||
|
|
111
|
-
nodeOrNodes instanceof Content) {
|
|
112
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
113
|
-
}
|
|
114
|
-
else {
|
|
115
|
-
// No overarching `Template` instance, so process the nodes directly.
|
|
116
|
-
nodeOrNodes.forEach((node) => node.visit(this));
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
visitElement(element) {
|
|
120
|
-
// `Element`s in the template may have `Reference`s which are captured in the scope.
|
|
121
|
-
element.references.forEach((node) => this.visitReference(node));
|
|
122
|
-
// Recurse into the `Element`'s children.
|
|
123
|
-
element.children.forEach((node) => node.visit(this));
|
|
124
|
-
this.elementsInScope.add(element);
|
|
125
|
-
}
|
|
126
|
-
visitTemplate(template) {
|
|
127
|
-
// References on a <ng-template> are defined in the outer scope, so capture them before
|
|
128
|
-
// processing the template's child scope.
|
|
129
|
-
template.references.forEach((node) => this.visitReference(node));
|
|
130
|
-
// Next, create an inner scope and process the template within it.
|
|
131
|
-
this.ingestScopedNode(template);
|
|
132
|
-
}
|
|
133
|
-
visitVariable(variable) {
|
|
134
|
-
// Declare the variable if it's not already.
|
|
135
|
-
this.maybeDeclare(variable);
|
|
136
|
-
}
|
|
137
|
-
visitReference(reference) {
|
|
138
|
-
// Declare the variable if it's not already.
|
|
139
|
-
this.maybeDeclare(reference);
|
|
140
|
-
}
|
|
141
|
-
visitDeferredBlock(deferred) {
|
|
142
|
-
this.ingestScopedNode(deferred);
|
|
143
|
-
deferred.placeholder?.visit(this);
|
|
144
|
-
deferred.loading?.visit(this);
|
|
145
|
-
deferred.error?.visit(this);
|
|
146
|
-
}
|
|
147
|
-
visitDeferredBlockPlaceholder(block) {
|
|
148
|
-
this.ingestScopedNode(block);
|
|
149
|
-
}
|
|
150
|
-
visitDeferredBlockError(block) {
|
|
151
|
-
this.ingestScopedNode(block);
|
|
152
|
-
}
|
|
153
|
-
visitDeferredBlockLoading(block) {
|
|
154
|
-
this.ingestScopedNode(block);
|
|
155
|
-
}
|
|
156
|
-
visitSwitchBlock(block) {
|
|
157
|
-
block.cases.forEach((node) => node.visit(this));
|
|
158
|
-
}
|
|
159
|
-
visitSwitchBlockCase(block) {
|
|
160
|
-
this.ingestScopedNode(block);
|
|
161
|
-
}
|
|
162
|
-
visitForLoopBlock(block) {
|
|
163
|
-
this.ingestScopedNode(block);
|
|
164
|
-
block.empty?.visit(this);
|
|
165
|
-
}
|
|
166
|
-
visitForLoopBlockEmpty(block) {
|
|
167
|
-
this.ingestScopedNode(block);
|
|
168
|
-
}
|
|
169
|
-
visitIfBlock(block) {
|
|
170
|
-
block.branches.forEach((node) => node.visit(this));
|
|
171
|
-
}
|
|
172
|
-
visitIfBlockBranch(block) {
|
|
173
|
-
this.ingestScopedNode(block);
|
|
174
|
-
}
|
|
175
|
-
visitContent(content) {
|
|
176
|
-
this.ingestScopedNode(content);
|
|
177
|
-
}
|
|
178
|
-
visitLetDeclaration(decl) {
|
|
179
|
-
this.maybeDeclare(decl);
|
|
180
|
-
}
|
|
181
|
-
// Unused visitors.
|
|
182
|
-
visitBoundAttribute(attr) { }
|
|
183
|
-
visitBoundEvent(event) { }
|
|
184
|
-
visitBoundText(text) { }
|
|
185
|
-
visitText(text) { }
|
|
186
|
-
visitTextAttribute(attr) { }
|
|
187
|
-
visitIcu(icu) { }
|
|
188
|
-
visitDeferredTrigger(trigger) { }
|
|
189
|
-
visitUnknownBlock(block) { }
|
|
190
|
-
maybeDeclare(thing) {
|
|
191
|
-
// Declare something with a name, as long as that name isn't taken.
|
|
192
|
-
if (!this.namedEntities.has(thing.name)) {
|
|
193
|
-
this.namedEntities.set(thing.name, thing);
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
/**
|
|
197
|
-
* Look up a variable within this `Scope`.
|
|
198
|
-
*
|
|
199
|
-
* This can recurse into a parent `Scope` if it's available.
|
|
200
|
-
*/
|
|
201
|
-
lookup(name) {
|
|
202
|
-
if (this.namedEntities.has(name)) {
|
|
203
|
-
// Found in the local scope.
|
|
204
|
-
return this.namedEntities.get(name);
|
|
205
|
-
}
|
|
206
|
-
else if (this.parentScope !== null) {
|
|
207
|
-
// Not in the local scope, but there's a parent scope so check there.
|
|
208
|
-
return this.parentScope.lookup(name);
|
|
209
|
-
}
|
|
210
|
-
else {
|
|
211
|
-
// At the top level and it wasn't found.
|
|
212
|
-
return null;
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
/**
|
|
216
|
-
* Get the child scope for a `ScopedNode`.
|
|
217
|
-
*
|
|
218
|
-
* This should always be defined.
|
|
219
|
-
*/
|
|
220
|
-
getChildScope(node) {
|
|
221
|
-
const res = this.childScopes.get(node);
|
|
222
|
-
if (res === undefined) {
|
|
223
|
-
throw new Error(`Assertion error: child scope for ${node} not found`);
|
|
224
|
-
}
|
|
225
|
-
return res;
|
|
226
|
-
}
|
|
227
|
-
ingestScopedNode(node) {
|
|
228
|
-
const scope = new Scope(this, node);
|
|
229
|
-
scope.ingest(node);
|
|
230
|
-
this.childScopes.set(node, scope);
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
/**
|
|
234
|
-
* Processes a template and matches directives on nodes (elements and templates).
|
|
235
|
-
*
|
|
236
|
-
* Usually used via the static `apply()` method.
|
|
237
|
-
*/
|
|
238
|
-
class DirectiveBinder {
|
|
239
|
-
constructor(matcher, directives, eagerDirectives, bindings, references) {
|
|
240
|
-
this.matcher = matcher;
|
|
241
|
-
this.directives = directives;
|
|
242
|
-
this.eagerDirectives = eagerDirectives;
|
|
243
|
-
this.bindings = bindings;
|
|
244
|
-
this.references = references;
|
|
245
|
-
// Indicates whether we are visiting elements within a `defer` block
|
|
246
|
-
this.isInDeferBlock = false;
|
|
247
|
-
}
|
|
248
|
-
/**
|
|
249
|
-
* Process a template (list of `Node`s) and perform directive matching against each node.
|
|
250
|
-
*
|
|
251
|
-
* @param template the list of template `Node`s to match (recursively).
|
|
252
|
-
* @param selectorMatcher a `SelectorMatcher` containing the directives that are in scope for
|
|
253
|
-
* this template.
|
|
254
|
-
* @returns three maps which contain information about directives in the template: the
|
|
255
|
-
* `directives` map which lists directives matched on each node, the `bindings` map which
|
|
256
|
-
* indicates which directives claimed which bindings (inputs, outputs, etc), and the `references`
|
|
257
|
-
* map which resolves #references (`Reference`s) within the template to the named directive or
|
|
258
|
-
* template node.
|
|
259
|
-
*/
|
|
260
|
-
static apply(template, selectorMatcher) {
|
|
261
|
-
const directives = new Map();
|
|
262
|
-
const bindings = new Map();
|
|
263
|
-
const references = new Map();
|
|
264
|
-
const eagerDirectives = [];
|
|
265
|
-
const matcher = new DirectiveBinder(selectorMatcher, directives, eagerDirectives, bindings, references);
|
|
266
|
-
matcher.ingest(template);
|
|
267
|
-
return { directives, eagerDirectives, bindings, references };
|
|
268
|
-
}
|
|
269
|
-
ingest(template) {
|
|
270
|
-
template.forEach((node) => node.visit(this));
|
|
271
|
-
}
|
|
272
|
-
visitElement(element) {
|
|
273
|
-
this.visitElementOrTemplate(element);
|
|
274
|
-
}
|
|
275
|
-
visitTemplate(template) {
|
|
276
|
-
this.visitElementOrTemplate(template);
|
|
277
|
-
}
|
|
278
|
-
visitElementOrTemplate(node) {
|
|
279
|
-
// First, determine the HTML shape of the node for the purpose of directive matching.
|
|
280
|
-
// Do this by building up a `CssSelector` for the node.
|
|
281
|
-
const cssSelector = createCssSelectorFromNode(node);
|
|
282
|
-
// Next, use the `SelectorMatcher` to get the list of directives on the node.
|
|
283
|
-
const directives = [];
|
|
284
|
-
this.matcher.match(cssSelector, (_selector, results) => directives.push(...results));
|
|
285
|
-
if (directives.length > 0) {
|
|
286
|
-
this.directives.set(node, directives);
|
|
287
|
-
if (!this.isInDeferBlock) {
|
|
288
|
-
this.eagerDirectives.push(...directives);
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
// Resolve any references that are created on this node.
|
|
292
|
-
node.references.forEach((ref) => {
|
|
293
|
-
let dirTarget = null;
|
|
294
|
-
// If the reference expression is empty, then it matches the "primary" directive on the node
|
|
295
|
-
// (if there is one). Otherwise it matches the host node itself (either an element or
|
|
296
|
-
// <ng-template> node).
|
|
297
|
-
if (ref.value.trim() === '') {
|
|
298
|
-
// This could be a reference to a component if there is one.
|
|
299
|
-
dirTarget = directives.find((dir) => dir.isComponent) || null;
|
|
300
|
-
}
|
|
301
|
-
else {
|
|
302
|
-
// This should be a reference to a directive exported via exportAs.
|
|
303
|
-
dirTarget =
|
|
304
|
-
directives.find((dir) => dir.exportAs !== null && dir.exportAs.some((value) => value === ref.value)) || null;
|
|
305
|
-
// Check if a matching directive was found.
|
|
306
|
-
if (dirTarget === null) {
|
|
307
|
-
// No matching directive was found - this reference points to an unknown target. Leave it
|
|
308
|
-
// unmapped.
|
|
309
|
-
return;
|
|
310
|
-
}
|
|
311
|
-
}
|
|
312
|
-
if (dirTarget !== null) {
|
|
313
|
-
// This reference points to a directive.
|
|
314
|
-
this.references.set(ref, { directive: dirTarget, node });
|
|
315
|
-
}
|
|
316
|
-
else {
|
|
317
|
-
// This reference points to the node itself.
|
|
318
|
-
this.references.set(ref, node);
|
|
319
|
-
}
|
|
320
|
-
});
|
|
321
|
-
const setAttributeBinding = (attribute, ioType) => {
|
|
322
|
-
const dir = directives.find((dir) => dir[ioType].hasBindingPropertyName(attribute.name));
|
|
323
|
-
const binding = dir !== undefined ? dir : node;
|
|
324
|
-
this.bindings.set(attribute, binding);
|
|
325
|
-
};
|
|
326
|
-
// Node inputs (bound attributes) and text attributes can be bound to an
|
|
327
|
-
// input on a directive.
|
|
328
|
-
node.inputs.forEach((input) => setAttributeBinding(input, 'inputs'));
|
|
329
|
-
node.attributes.forEach((attr) => setAttributeBinding(attr, 'inputs'));
|
|
330
|
-
if (node instanceof Template) {
|
|
331
|
-
node.templateAttrs.forEach((attr) => setAttributeBinding(attr, 'inputs'));
|
|
332
|
-
}
|
|
333
|
-
// Node outputs (bound events) can be bound to an output on a directive.
|
|
334
|
-
node.outputs.forEach((output) => setAttributeBinding(output, 'outputs'));
|
|
335
|
-
// Recurse into the node's children.
|
|
336
|
-
node.children.forEach((child) => child.visit(this));
|
|
337
|
-
}
|
|
338
|
-
visitDeferredBlock(deferred) {
|
|
339
|
-
const wasInDeferBlock = this.isInDeferBlock;
|
|
340
|
-
this.isInDeferBlock = true;
|
|
341
|
-
deferred.children.forEach((child) => child.visit(this));
|
|
342
|
-
this.isInDeferBlock = wasInDeferBlock;
|
|
343
|
-
deferred.placeholder?.visit(this);
|
|
344
|
-
deferred.loading?.visit(this);
|
|
345
|
-
deferred.error?.visit(this);
|
|
346
|
-
}
|
|
347
|
-
visitDeferredBlockPlaceholder(block) {
|
|
348
|
-
block.children.forEach((child) => child.visit(this));
|
|
349
|
-
}
|
|
350
|
-
visitDeferredBlockError(block) {
|
|
351
|
-
block.children.forEach((child) => child.visit(this));
|
|
352
|
-
}
|
|
353
|
-
visitDeferredBlockLoading(block) {
|
|
354
|
-
block.children.forEach((child) => child.visit(this));
|
|
355
|
-
}
|
|
356
|
-
visitSwitchBlock(block) {
|
|
357
|
-
block.cases.forEach((node) => node.visit(this));
|
|
358
|
-
}
|
|
359
|
-
visitSwitchBlockCase(block) {
|
|
360
|
-
block.children.forEach((node) => node.visit(this));
|
|
361
|
-
}
|
|
362
|
-
visitForLoopBlock(block) {
|
|
363
|
-
block.item.visit(this);
|
|
364
|
-
block.contextVariables.forEach((v) => v.visit(this));
|
|
365
|
-
block.children.forEach((node) => node.visit(this));
|
|
366
|
-
block.empty?.visit(this);
|
|
367
|
-
}
|
|
368
|
-
visitForLoopBlockEmpty(block) {
|
|
369
|
-
block.children.forEach((node) => node.visit(this));
|
|
370
|
-
}
|
|
371
|
-
visitIfBlock(block) {
|
|
372
|
-
block.branches.forEach((node) => node.visit(this));
|
|
373
|
-
}
|
|
374
|
-
visitIfBlockBranch(block) {
|
|
375
|
-
block.expressionAlias?.visit(this);
|
|
376
|
-
block.children.forEach((node) => node.visit(this));
|
|
377
|
-
}
|
|
378
|
-
visitContent(content) {
|
|
379
|
-
content.children.forEach((child) => child.visit(this));
|
|
380
|
-
}
|
|
381
|
-
// Unused visitors.
|
|
382
|
-
visitVariable(variable) { }
|
|
383
|
-
visitReference(reference) { }
|
|
384
|
-
visitTextAttribute(attribute) { }
|
|
385
|
-
visitBoundAttribute(attribute) { }
|
|
386
|
-
visitBoundEvent(attribute) { }
|
|
387
|
-
visitBoundAttributeOrEvent(node) { }
|
|
388
|
-
visitText(text) { }
|
|
389
|
-
visitBoundText(text) { }
|
|
390
|
-
visitIcu(icu) { }
|
|
391
|
-
visitDeferredTrigger(trigger) { }
|
|
392
|
-
visitUnknownBlock(block) { }
|
|
393
|
-
visitLetDeclaration(decl) { }
|
|
394
|
-
}
|
|
395
|
-
/**
|
|
396
|
-
* Processes a template and extract metadata about expressions and symbols within.
|
|
397
|
-
*
|
|
398
|
-
* This is a companion to the `DirectiveBinder` that doesn't require knowledge of directives matched
|
|
399
|
-
* within the template in order to operate.
|
|
400
|
-
*
|
|
401
|
-
* Expressions are visited by the superclass `RecursiveAstVisitor`, with custom logic provided
|
|
402
|
-
* by overridden methods from that visitor.
|
|
403
|
-
*/
|
|
404
|
-
class TemplateBinder extends RecursiveAstVisitor {
|
|
405
|
-
constructor(bindings, symbols, usedPipes, eagerPipes, deferBlocks, nestingLevel, scope, rootNode, level) {
|
|
406
|
-
super();
|
|
407
|
-
this.bindings = bindings;
|
|
408
|
-
this.symbols = symbols;
|
|
409
|
-
this.usedPipes = usedPipes;
|
|
410
|
-
this.eagerPipes = eagerPipes;
|
|
411
|
-
this.deferBlocks = deferBlocks;
|
|
412
|
-
this.nestingLevel = nestingLevel;
|
|
413
|
-
this.scope = scope;
|
|
414
|
-
this.rootNode = rootNode;
|
|
415
|
-
this.level = level;
|
|
416
|
-
// Save a bit of processing time by constructing this closure in advance.
|
|
417
|
-
this.visitNode = (node) => node.visit(this);
|
|
418
|
-
}
|
|
419
|
-
// This method is defined to reconcile the type of TemplateBinder since both
|
|
420
|
-
// RecursiveAstVisitor and Visitor define the visit() method in their
|
|
421
|
-
// interfaces.
|
|
422
|
-
visit(node, context) {
|
|
423
|
-
if (node instanceof AST) {
|
|
424
|
-
node.visit(this, context);
|
|
425
|
-
}
|
|
426
|
-
else {
|
|
427
|
-
node.visit(this);
|
|
428
|
-
}
|
|
429
|
-
}
|
|
430
|
-
/**
|
|
431
|
-
* Process a template and extract metadata about expressions and symbols within.
|
|
432
|
-
*
|
|
433
|
-
* @param nodes the nodes of the template to process
|
|
434
|
-
* @param scope the `Scope` of the template being processed.
|
|
435
|
-
* @returns three maps which contain metadata about the template: `expressions` which interprets
|
|
436
|
-
* special `AST` nodes in expressions as pointing to references or variables declared within the
|
|
437
|
-
* template, `symbols` which maps those variables and references to the nested `Template` which
|
|
438
|
-
* declares them, if any, and `nestingLevel` which associates each `Template` with a integer
|
|
439
|
-
* nesting level (how many levels deep within the template structure the `Template` is), starting
|
|
440
|
-
* at 1.
|
|
441
|
-
*/
|
|
442
|
-
static applyWithScope(nodes, scope) {
|
|
443
|
-
const expressions = new Map();
|
|
444
|
-
const symbols = new Map();
|
|
445
|
-
const nestingLevel = new Map();
|
|
446
|
-
const usedPipes = new Set();
|
|
447
|
-
const eagerPipes = new Set();
|
|
448
|
-
const template = nodes instanceof Template ? nodes : null;
|
|
449
|
-
const deferBlocks = [];
|
|
450
|
-
// The top-level template has nesting level 0.
|
|
451
|
-
const binder = new TemplateBinder(expressions, symbols, usedPipes, eagerPipes, deferBlocks, nestingLevel, scope, template, 0);
|
|
452
|
-
binder.ingest(nodes);
|
|
453
|
-
return { expressions, symbols, nestingLevel, usedPipes, eagerPipes, deferBlocks };
|
|
454
|
-
}
|
|
455
|
-
ingest(nodeOrNodes) {
|
|
456
|
-
if (nodeOrNodes instanceof Template) {
|
|
457
|
-
// For <ng-template>s, process only variables and child nodes. Inputs, outputs, templateAttrs,
|
|
458
|
-
// and references were all processed in the scope of the containing template.
|
|
459
|
-
nodeOrNodes.variables.forEach(this.visitNode);
|
|
460
|
-
nodeOrNodes.children.forEach(this.visitNode);
|
|
461
|
-
// Set the nesting level.
|
|
462
|
-
this.nestingLevel.set(nodeOrNodes, this.level);
|
|
463
|
-
}
|
|
464
|
-
else if (nodeOrNodes instanceof IfBlockBranch) {
|
|
465
|
-
if (nodeOrNodes.expressionAlias !== null) {
|
|
466
|
-
this.visitNode(nodeOrNodes.expressionAlias);
|
|
467
|
-
}
|
|
468
|
-
nodeOrNodes.children.forEach(this.visitNode);
|
|
469
|
-
this.nestingLevel.set(nodeOrNodes, this.level);
|
|
470
|
-
}
|
|
471
|
-
else if (nodeOrNodes instanceof ForLoopBlock) {
|
|
472
|
-
this.visitNode(nodeOrNodes.item);
|
|
473
|
-
nodeOrNodes.contextVariables.forEach((v) => this.visitNode(v));
|
|
474
|
-
nodeOrNodes.trackBy.visit(this);
|
|
475
|
-
nodeOrNodes.children.forEach(this.visitNode);
|
|
476
|
-
this.nestingLevel.set(nodeOrNodes, this.level);
|
|
477
|
-
}
|
|
478
|
-
else if (nodeOrNodes instanceof DeferredBlock) {
|
|
479
|
-
if (this.scope.rootNode !== nodeOrNodes) {
|
|
480
|
-
throw new Error(`Assertion error: resolved incorrect scope for deferred block ${nodeOrNodes}`);
|
|
481
|
-
}
|
|
482
|
-
this.deferBlocks.push([nodeOrNodes, this.scope]);
|
|
483
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
484
|
-
this.nestingLevel.set(nodeOrNodes, this.level);
|
|
485
|
-
}
|
|
486
|
-
else if (nodeOrNodes instanceof SwitchBlockCase ||
|
|
487
|
-
nodeOrNodes instanceof ForLoopBlockEmpty ||
|
|
488
|
-
nodeOrNodes instanceof DeferredBlockError ||
|
|
489
|
-
nodeOrNodes instanceof DeferredBlockPlaceholder ||
|
|
490
|
-
nodeOrNodes instanceof DeferredBlockLoading ||
|
|
491
|
-
nodeOrNodes instanceof Content) {
|
|
492
|
-
nodeOrNodes.children.forEach((node) => node.visit(this));
|
|
493
|
-
this.nestingLevel.set(nodeOrNodes, this.level);
|
|
494
|
-
}
|
|
495
|
-
else {
|
|
496
|
-
// Visit each node from the top-level template.
|
|
497
|
-
nodeOrNodes.forEach(this.visitNode);
|
|
498
|
-
}
|
|
499
|
-
}
|
|
500
|
-
visitElement(element) {
|
|
501
|
-
// Visit the inputs, outputs, and children of the element.
|
|
502
|
-
element.inputs.forEach(this.visitNode);
|
|
503
|
-
element.outputs.forEach(this.visitNode);
|
|
504
|
-
element.children.forEach(this.visitNode);
|
|
505
|
-
element.references.forEach(this.visitNode);
|
|
506
|
-
}
|
|
507
|
-
visitTemplate(template) {
|
|
508
|
-
// First, visit inputs, outputs and template attributes of the template node.
|
|
509
|
-
template.inputs.forEach(this.visitNode);
|
|
510
|
-
template.outputs.forEach(this.visitNode);
|
|
511
|
-
template.templateAttrs.forEach(this.visitNode);
|
|
512
|
-
template.references.forEach(this.visitNode);
|
|
513
|
-
// Next, recurse into the template.
|
|
514
|
-
this.ingestScopedNode(template);
|
|
515
|
-
}
|
|
516
|
-
visitVariable(variable) {
|
|
517
|
-
// Register the `Variable` as a symbol in the current `Template`.
|
|
518
|
-
if (this.rootNode !== null) {
|
|
519
|
-
this.symbols.set(variable, this.rootNode);
|
|
520
|
-
}
|
|
521
|
-
}
|
|
522
|
-
visitReference(reference) {
|
|
523
|
-
// Register the `Reference` as a symbol in the current `Template`.
|
|
524
|
-
if (this.rootNode !== null) {
|
|
525
|
-
this.symbols.set(reference, this.rootNode);
|
|
526
|
-
}
|
|
527
|
-
}
|
|
528
|
-
// Unused template visitors
|
|
529
|
-
visitText(text) { }
|
|
530
|
-
visitTextAttribute(attribute) { }
|
|
531
|
-
visitUnknownBlock(block) { }
|
|
532
|
-
visitDeferredTrigger() { }
|
|
533
|
-
visitIcu(icu) {
|
|
534
|
-
Object.keys(icu.vars).forEach((key) => icu.vars[key].visit(this));
|
|
535
|
-
Object.keys(icu.placeholders).forEach((key) => icu.placeholders[key].visit(this));
|
|
536
|
-
}
|
|
537
|
-
// The remaining visitors are concerned with processing AST expressions within template bindings
|
|
538
|
-
visitBoundAttribute(attribute) {
|
|
539
|
-
attribute.value.visit(this);
|
|
540
|
-
}
|
|
541
|
-
visitBoundEvent(event) {
|
|
542
|
-
event.handler.visit(this);
|
|
543
|
-
}
|
|
544
|
-
visitDeferredBlock(deferred) {
|
|
545
|
-
this.ingestScopedNode(deferred);
|
|
546
|
-
deferred.triggers.when?.value.visit(this);
|
|
547
|
-
deferred.prefetchTriggers.when?.value.visit(this);
|
|
548
|
-
deferred.placeholder && this.visitNode(deferred.placeholder);
|
|
549
|
-
deferred.loading && this.visitNode(deferred.loading);
|
|
550
|
-
deferred.error && this.visitNode(deferred.error);
|
|
551
|
-
}
|
|
552
|
-
visitDeferredBlockPlaceholder(block) {
|
|
553
|
-
this.ingestScopedNode(block);
|
|
554
|
-
}
|
|
555
|
-
visitDeferredBlockError(block) {
|
|
556
|
-
this.ingestScopedNode(block);
|
|
557
|
-
}
|
|
558
|
-
visitDeferredBlockLoading(block) {
|
|
559
|
-
this.ingestScopedNode(block);
|
|
560
|
-
}
|
|
561
|
-
visitSwitchBlock(block) {
|
|
562
|
-
block.expression.visit(this);
|
|
563
|
-
block.cases.forEach(this.visitNode);
|
|
564
|
-
}
|
|
565
|
-
visitSwitchBlockCase(block) {
|
|
566
|
-
block.expression?.visit(this);
|
|
567
|
-
this.ingestScopedNode(block);
|
|
568
|
-
}
|
|
569
|
-
visitForLoopBlock(block) {
|
|
570
|
-
block.expression.visit(this);
|
|
571
|
-
this.ingestScopedNode(block);
|
|
572
|
-
block.empty?.visit(this);
|
|
573
|
-
}
|
|
574
|
-
visitForLoopBlockEmpty(block) {
|
|
575
|
-
this.ingestScopedNode(block);
|
|
576
|
-
}
|
|
577
|
-
visitIfBlock(block) {
|
|
578
|
-
block.branches.forEach((node) => node.visit(this));
|
|
579
|
-
}
|
|
580
|
-
visitIfBlockBranch(block) {
|
|
581
|
-
block.expression?.visit(this);
|
|
582
|
-
this.ingestScopedNode(block);
|
|
583
|
-
}
|
|
584
|
-
visitContent(content) {
|
|
585
|
-
this.ingestScopedNode(content);
|
|
586
|
-
}
|
|
587
|
-
visitBoundText(text) {
|
|
588
|
-
text.value.visit(this);
|
|
589
|
-
}
|
|
590
|
-
visitLetDeclaration(decl) {
|
|
591
|
-
decl.value.visit(this);
|
|
592
|
-
if (this.rootNode !== null) {
|
|
593
|
-
this.symbols.set(decl, this.rootNode);
|
|
594
|
-
}
|
|
595
|
-
}
|
|
596
|
-
visitPipe(ast, context) {
|
|
597
|
-
this.usedPipes.add(ast.name);
|
|
598
|
-
if (!this.scope.isDeferred) {
|
|
599
|
-
this.eagerPipes.add(ast.name);
|
|
600
|
-
}
|
|
601
|
-
return super.visitPipe(ast, context);
|
|
602
|
-
}
|
|
603
|
-
// These five types of AST expressions can refer to expression roots, which could be variables
|
|
604
|
-
// or references in the current scope.
|
|
605
|
-
visitPropertyRead(ast, context) {
|
|
606
|
-
this.maybeMap(ast, ast.name);
|
|
607
|
-
return super.visitPropertyRead(ast, context);
|
|
608
|
-
}
|
|
609
|
-
visitSafePropertyRead(ast, context) {
|
|
610
|
-
this.maybeMap(ast, ast.name);
|
|
611
|
-
return super.visitSafePropertyRead(ast, context);
|
|
612
|
-
}
|
|
613
|
-
visitPropertyWrite(ast, context) {
|
|
614
|
-
this.maybeMap(ast, ast.name);
|
|
615
|
-
return super.visitPropertyWrite(ast, context);
|
|
616
|
-
}
|
|
617
|
-
ingestScopedNode(node) {
|
|
618
|
-
const childScope = this.scope.getChildScope(node);
|
|
619
|
-
const binder = new TemplateBinder(this.bindings, this.symbols, this.usedPipes, this.eagerPipes, this.deferBlocks, this.nestingLevel, childScope, node, this.level + 1);
|
|
620
|
-
binder.ingest(node);
|
|
621
|
-
}
|
|
622
|
-
maybeMap(ast, name) {
|
|
623
|
-
// If the receiver of the expression isn't the `ImplicitReceiver`, this isn't the root of an
|
|
624
|
-
// `AST` expression that maps to a `Variable` or `Reference`.
|
|
625
|
-
if (!(ast.receiver instanceof ImplicitReceiver)) {
|
|
626
|
-
return;
|
|
627
|
-
}
|
|
628
|
-
// Check whether the name exists in the current scope. If so, map it. Otherwise, the name is
|
|
629
|
-
// probably a property on the top-level component context.
|
|
630
|
-
const target = this.scope.lookup(name);
|
|
631
|
-
// It's not allowed to read template entities via `this`, however it previously worked by
|
|
632
|
-
// accident (see #55115). Since `@let` declarations are new, we can fix it from the beginning,
|
|
633
|
-
// whereas pre-existing template entities will be fixed in #55115.
|
|
634
|
-
if (target instanceof LetDeclaration && ast.receiver instanceof ThisReceiver) {
|
|
635
|
-
return;
|
|
636
|
-
}
|
|
637
|
-
if (target !== null) {
|
|
638
|
-
this.bindings.set(ast, target);
|
|
639
|
-
}
|
|
640
|
-
}
|
|
641
|
-
}
|
|
642
|
-
/**
|
|
643
|
-
* Metadata container for a `Target` that allows queries for specific bits of metadata.
|
|
644
|
-
*
|
|
645
|
-
* See `BoundTarget` for documentation on the individual methods.
|
|
646
|
-
*/
|
|
647
|
-
export class R3BoundTarget {
|
|
648
|
-
constructor(target, directives, eagerDirectives, bindings, references, exprTargets, symbols, nestingLevel, scopedNodeEntities, usedPipes, eagerPipes, rawDeferred) {
|
|
649
|
-
this.target = target;
|
|
650
|
-
this.directives = directives;
|
|
651
|
-
this.eagerDirectives = eagerDirectives;
|
|
652
|
-
this.bindings = bindings;
|
|
653
|
-
this.references = references;
|
|
654
|
-
this.exprTargets = exprTargets;
|
|
655
|
-
this.symbols = symbols;
|
|
656
|
-
this.nestingLevel = nestingLevel;
|
|
657
|
-
this.scopedNodeEntities = scopedNodeEntities;
|
|
658
|
-
this.usedPipes = usedPipes;
|
|
659
|
-
this.eagerPipes = eagerPipes;
|
|
660
|
-
this.deferredBlocks = rawDeferred.map((current) => current[0]);
|
|
661
|
-
this.deferredScopes = new Map(rawDeferred);
|
|
662
|
-
}
|
|
663
|
-
getEntitiesInScope(node) {
|
|
664
|
-
return this.scopedNodeEntities.get(node) ?? new Set();
|
|
665
|
-
}
|
|
666
|
-
getDirectivesOfNode(node) {
|
|
667
|
-
return this.directives.get(node) || null;
|
|
668
|
-
}
|
|
669
|
-
getReferenceTarget(ref) {
|
|
670
|
-
return this.references.get(ref) || null;
|
|
671
|
-
}
|
|
672
|
-
getConsumerOfBinding(binding) {
|
|
673
|
-
return this.bindings.get(binding) || null;
|
|
674
|
-
}
|
|
675
|
-
getExpressionTarget(expr) {
|
|
676
|
-
return this.exprTargets.get(expr) || null;
|
|
677
|
-
}
|
|
678
|
-
getDefinitionNodeOfSymbol(symbol) {
|
|
679
|
-
return this.symbols.get(symbol) || null;
|
|
680
|
-
}
|
|
681
|
-
getNestingLevel(node) {
|
|
682
|
-
return this.nestingLevel.get(node) || 0;
|
|
683
|
-
}
|
|
684
|
-
getUsedDirectives() {
|
|
685
|
-
const set = new Set();
|
|
686
|
-
this.directives.forEach((dirs) => dirs.forEach((dir) => set.add(dir)));
|
|
687
|
-
return Array.from(set.values());
|
|
688
|
-
}
|
|
689
|
-
getEagerlyUsedDirectives() {
|
|
690
|
-
const set = new Set(this.eagerDirectives);
|
|
691
|
-
return Array.from(set.values());
|
|
692
|
-
}
|
|
693
|
-
getUsedPipes() {
|
|
694
|
-
return Array.from(this.usedPipes);
|
|
695
|
-
}
|
|
696
|
-
getEagerlyUsedPipes() {
|
|
697
|
-
return Array.from(this.eagerPipes);
|
|
698
|
-
}
|
|
699
|
-
getDeferBlocks() {
|
|
700
|
-
return this.deferredBlocks;
|
|
701
|
-
}
|
|
702
|
-
getDeferredTriggerTarget(block, trigger) {
|
|
703
|
-
// Only triggers that refer to DOM nodes can be resolved.
|
|
704
|
-
if (!(trigger instanceof InteractionDeferredTrigger) &&
|
|
705
|
-
!(trigger instanceof ViewportDeferredTrigger) &&
|
|
706
|
-
!(trigger instanceof HoverDeferredTrigger)) {
|
|
707
|
-
return null;
|
|
708
|
-
}
|
|
709
|
-
const name = trigger.reference;
|
|
710
|
-
if (name === null) {
|
|
711
|
-
let trigger = null;
|
|
712
|
-
if (block.placeholder !== null) {
|
|
713
|
-
for (const child of block.placeholder.children) {
|
|
714
|
-
// Skip over comment nodes. Currently by default the template parser doesn't capture
|
|
715
|
-
// comments, but we have a safeguard here just in case since it can be enabled.
|
|
716
|
-
if (child instanceof Comment) {
|
|
717
|
-
continue;
|
|
718
|
-
}
|
|
719
|
-
// We can only infer the trigger if there's one root element node. Any other
|
|
720
|
-
// nodes at the root make it so that we can't infer the trigger anymore.
|
|
721
|
-
if (trigger !== null) {
|
|
722
|
-
return null;
|
|
723
|
-
}
|
|
724
|
-
if (child instanceof Element) {
|
|
725
|
-
trigger = child;
|
|
726
|
-
}
|
|
727
|
-
}
|
|
728
|
-
}
|
|
729
|
-
return trigger;
|
|
730
|
-
}
|
|
731
|
-
const outsideRef = this.findEntityInScope(block, name);
|
|
732
|
-
// First try to resolve the target in the scope of the main deferred block. Note that we
|
|
733
|
-
// skip triggers defined inside the main block itself, because they might not exist yet.
|
|
734
|
-
if (outsideRef instanceof Reference && this.getDefinitionNodeOfSymbol(outsideRef) !== block) {
|
|
735
|
-
const target = this.getReferenceTarget(outsideRef);
|
|
736
|
-
if (target !== null) {
|
|
737
|
-
return this.referenceTargetToElement(target);
|
|
738
|
-
}
|
|
739
|
-
}
|
|
740
|
-
// If the trigger couldn't be found in the main block, check the
|
|
741
|
-
// placeholder block which is shown before the main block has loaded.
|
|
742
|
-
if (block.placeholder !== null) {
|
|
743
|
-
const refInPlaceholder = this.findEntityInScope(block.placeholder, name);
|
|
744
|
-
const targetInPlaceholder = refInPlaceholder instanceof Reference ? this.getReferenceTarget(refInPlaceholder) : null;
|
|
745
|
-
if (targetInPlaceholder !== null) {
|
|
746
|
-
return this.referenceTargetToElement(targetInPlaceholder);
|
|
747
|
-
}
|
|
748
|
-
}
|
|
749
|
-
return null;
|
|
750
|
-
}
|
|
751
|
-
isDeferred(element) {
|
|
752
|
-
for (const block of this.deferredBlocks) {
|
|
753
|
-
if (!this.deferredScopes.has(block)) {
|
|
754
|
-
continue;
|
|
755
|
-
}
|
|
756
|
-
const stack = [this.deferredScopes.get(block)];
|
|
757
|
-
while (stack.length > 0) {
|
|
758
|
-
const current = stack.pop();
|
|
759
|
-
if (current.elementsInScope.has(element)) {
|
|
760
|
-
return true;
|
|
761
|
-
}
|
|
762
|
-
stack.push(...current.childScopes.values());
|
|
763
|
-
}
|
|
764
|
-
}
|
|
765
|
-
return false;
|
|
766
|
-
}
|
|
767
|
-
/**
|
|
768
|
-
* Finds an entity with a specific name in a scope.
|
|
769
|
-
* @param rootNode Root node of the scope.
|
|
770
|
-
* @param name Name of the entity.
|
|
771
|
-
*/
|
|
772
|
-
findEntityInScope(rootNode, name) {
|
|
773
|
-
const entities = this.getEntitiesInScope(rootNode);
|
|
774
|
-
for (const entity of entities) {
|
|
775
|
-
if (entity.name === name) {
|
|
776
|
-
return entity;
|
|
777
|
-
}
|
|
778
|
-
}
|
|
779
|
-
return null;
|
|
780
|
-
}
|
|
781
|
-
/** Coerces a `ReferenceTarget` to an `Element`, if possible. */
|
|
782
|
-
referenceTargetToElement(target) {
|
|
783
|
-
if (target instanceof Element) {
|
|
784
|
-
return target;
|
|
785
|
-
}
|
|
786
|
-
if (target instanceof Template) {
|
|
787
|
-
return null;
|
|
788
|
-
}
|
|
789
|
-
return this.referenceTargetToElement(target.node);
|
|
790
|
-
}
|
|
791
|
-
}
|
|
792
|
-
function extractScopedNodeEntities(rootScope) {
|
|
793
|
-
const entityMap = new Map();
|
|
794
|
-
function extractScopeEntities(scope) {
|
|
795
|
-
if (entityMap.has(scope.rootNode)) {
|
|
796
|
-
return entityMap.get(scope.rootNode);
|
|
797
|
-
}
|
|
798
|
-
const currentEntities = scope.namedEntities;
|
|
799
|
-
let entities;
|
|
800
|
-
if (scope.parentScope !== null) {
|
|
801
|
-
entities = new Map([...extractScopeEntities(scope.parentScope), ...currentEntities]);
|
|
802
|
-
}
|
|
803
|
-
else {
|
|
804
|
-
entities = new Map(currentEntities);
|
|
805
|
-
}
|
|
806
|
-
entityMap.set(scope.rootNode, entities);
|
|
807
|
-
return entities;
|
|
808
|
-
}
|
|
809
|
-
const scopesToProcess = [rootScope];
|
|
810
|
-
while (scopesToProcess.length > 0) {
|
|
811
|
-
const scope = scopesToProcess.pop();
|
|
812
|
-
for (const childScope of scope.childScopes.values()) {
|
|
813
|
-
scopesToProcess.push(childScope);
|
|
814
|
-
}
|
|
815
|
-
extractScopeEntities(scope);
|
|
816
|
-
}
|
|
817
|
-
const templateEntities = new Map();
|
|
818
|
-
for (const [template, entities] of entityMap) {
|
|
819
|
-
templateEntities.set(template, new Set(entities.values()));
|
|
820
|
-
}
|
|
821
|
-
return templateEntities;
|
|
822
|
-
}
|
|
823
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"t2_binder.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler/src/render3/view/t2_binder.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 {\n  AST,\n  BindingPipe,\n  ImplicitReceiver,\n  PropertyRead,\n  PropertyWrite,\n  RecursiveAstVisitor,\n  SafePropertyRead,\n  ThisReceiver,\n} from '../../expression_parser/ast';\nimport {SelectorMatcher} from '../../selector';\nimport {\n  BoundAttribute,\n  BoundEvent,\n  BoundText,\n  Comment,\n  Content,\n  DeferredBlock,\n  DeferredBlockError,\n  DeferredBlockLoading,\n  DeferredBlockPlaceholder,\n  DeferredTrigger,\n  Element,\n  ForLoopBlock,\n  ForLoopBlockEmpty,\n  HoverDeferredTrigger,\n  Icu,\n  IfBlock,\n  IfBlockBranch,\n  InteractionDeferredTrigger,\n  LetDeclaration,\n  Node,\n  Reference,\n  SwitchBlock,\n  SwitchBlockCase,\n  Template,\n  Text,\n  TextAttribute,\n  UnknownBlock,\n  Variable,\n  ViewportDeferredTrigger,\n  Visitor,\n} from '../r3_ast';\n\nimport {\n  BoundTarget,\n  DirectiveMeta,\n  ReferenceTarget,\n  ScopedNode,\n  Target,\n  TargetBinder,\n  TemplateEntity,\n} from './t2_api';\nimport {createCssSelectorFromNode} from './util';\n\n/**\n * Processes `Target`s with a given set of directives and performs a binding operation, which\n * returns an object similar to TypeScript's `ts.TypeChecker` that contains knowledge about the\n * target.\n */\nexport class R3TargetBinder<DirectiveT extends DirectiveMeta> implements TargetBinder<DirectiveT> {\n  constructor(private directiveMatcher: SelectorMatcher<DirectiveT[]>) {}\n\n  /**\n   * Perform a binding operation on the given `Target` and return a `BoundTarget` which contains\n   * metadata about the types referenced in the template.\n   */\n  bind(target: Target): BoundTarget<DirectiveT> {\n    if (!target.template) {\n      // TODO(alxhub): handle targets which contain things like HostBindings, etc.\n      throw new Error('Binding without a template not yet supported');\n    }\n\n    // First, parse the template into a `Scope` structure. This operation captures the syntactic\n    // scopes in the template and makes them available for later use.\n    const scope = Scope.apply(target.template);\n\n    // Use the `Scope` to extract the entities present at every level of the template.\n    const scopedNodeEntities = extractScopedNodeEntities(scope);\n\n    // Next, perform directive matching on the template using the `DirectiveBinder`. This returns:\n    //   - directives: Map of nodes (elements & ng-templates) to the directives on them.\n    //   - bindings: Map of inputs, outputs, and attributes to the directive/element that claims\n    //     them. TODO(alxhub): handle multiple directives claiming an input/output/etc.\n    //   - references: Map of #references to their targets.\n    const {directives, eagerDirectives, bindings, references} = DirectiveBinder.apply(\n      target.template,\n      this.directiveMatcher,\n    );\n    // Finally, run the TemplateBinder to bind references, variables, and other entities within the\n    // template. This extracts all the metadata that doesn't depend on directive matching.\n    const {expressions, symbols, nestingLevel, usedPipes, eagerPipes, deferBlocks} =\n      TemplateBinder.applyWithScope(target.template, scope);\n    return new R3BoundTarget(\n      target,\n      directives,\n      eagerDirectives,\n      bindings,\n      references,\n      expressions,\n      symbols,\n      nestingLevel,\n      scopedNodeEntities,\n      usedPipes,\n      eagerPipes,\n      deferBlocks,\n    );\n  }\n}\n\n/**\n * Represents a binding scope within a template.\n *\n * Any variables, references, or other named entities declared within the template will\n * be captured and available by name in `namedEntities`. Additionally, child templates will\n * be analyzed and have their child `Scope`s available in `childScopes`.\n */\nclass Scope implements Visitor {\n  /**\n   * Named members of the `Scope`, such as `Reference`s or `Variable`s.\n   */\n  readonly namedEntities = new Map<string, TemplateEntity>();\n\n  /**\n   * Set of elements that belong to this scope.\n   */\n  readonly elementsInScope = new Set<Element>();\n\n  /**\n   * Child `Scope`s for immediately nested `ScopedNode`s.\n   */\n  readonly childScopes = new Map<ScopedNode, Scope>();\n\n  /** Whether this scope is deferred or if any of its ancestors are deferred. */\n  readonly isDeferred: boolean;\n\n  private constructor(\n    readonly parentScope: Scope | null,\n    readonly rootNode: ScopedNode | null,\n  ) {\n    this.isDeferred =\n      parentScope !== null && parentScope.isDeferred ? true : rootNode instanceof DeferredBlock;\n  }\n\n  static newRootScope(): Scope {\n    return new Scope(null, null);\n  }\n\n  /**\n   * Process a template (either as a `Template` sub-template with variables, or a plain array of\n   * template `Node`s) and construct its `Scope`.\n   */\n  static apply(template: Node[]): Scope {\n    const scope = Scope.newRootScope();\n    scope.ingest(template);\n    return scope;\n  }\n\n  /**\n   * Internal method to process the scoped node and populate the `Scope`.\n   */\n  private ingest(nodeOrNodes: ScopedNode | Node[]): void {\n    if (nodeOrNodes instanceof Template) {\n      // Variables on an <ng-template> are defined in the inner scope.\n      nodeOrNodes.variables.forEach((node) => this.visitVariable(node));\n\n      // Process the nodes of the template.\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n    } else if (nodeOrNodes instanceof IfBlockBranch) {\n      if (nodeOrNodes.expressionAlias !== null) {\n        this.visitVariable(nodeOrNodes.expressionAlias);\n      }\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n    } else if (nodeOrNodes instanceof ForLoopBlock) {\n      this.visitVariable(nodeOrNodes.item);\n      nodeOrNodes.contextVariables.forEach((v) => this.visitVariable(v));\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n    } else if (\n      nodeOrNodes instanceof SwitchBlockCase ||\n      nodeOrNodes instanceof ForLoopBlockEmpty ||\n      nodeOrNodes instanceof DeferredBlock ||\n      nodeOrNodes instanceof DeferredBlockError ||\n      nodeOrNodes instanceof DeferredBlockPlaceholder ||\n      nodeOrNodes instanceof DeferredBlockLoading ||\n      nodeOrNodes instanceof Content\n    ) {\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n    } else {\n      // No overarching `Template` instance, so process the nodes directly.\n      nodeOrNodes.forEach((node) => node.visit(this));\n    }\n  }\n\n  visitElement(element: Element) {\n    // `Element`s in the template may have `Reference`s which are captured in the scope.\n    element.references.forEach((node) => this.visitReference(node));\n\n    // Recurse into the `Element`'s children.\n    element.children.forEach((node) => node.visit(this));\n\n    this.elementsInScope.add(element);\n  }\n\n  visitTemplate(template: Template) {\n    // References on a <ng-template> are defined in the outer scope, so capture them before\n    // processing the template's child scope.\n    template.references.forEach((node) => this.visitReference(node));\n\n    // Next, create an inner scope and process the template within it.\n    this.ingestScopedNode(template);\n  }\n\n  visitVariable(variable: Variable) {\n    // Declare the variable if it's not already.\n    this.maybeDeclare(variable);\n  }\n\n  visitReference(reference: Reference) {\n    // Declare the variable if it's not already.\n    this.maybeDeclare(reference);\n  }\n\n  visitDeferredBlock(deferred: DeferredBlock) {\n    this.ingestScopedNode(deferred);\n    deferred.placeholder?.visit(this);\n    deferred.loading?.visit(this);\n    deferred.error?.visit(this);\n  }\n\n  visitDeferredBlockPlaceholder(block: DeferredBlockPlaceholder) {\n    this.ingestScopedNode(block);\n  }\n\n  visitDeferredBlockError(block: DeferredBlockError) {\n    this.ingestScopedNode(block);\n  }\n\n  visitDeferredBlockLoading(block: DeferredBlockLoading) {\n    this.ingestScopedNode(block);\n  }\n\n  visitSwitchBlock(block: SwitchBlock) {\n    block.cases.forEach((node) => node.visit(this));\n  }\n\n  visitSwitchBlockCase(block: SwitchBlockCase) {\n    this.ingestScopedNode(block);\n  }\n\n  visitForLoopBlock(block: ForLoopBlock) {\n    this.ingestScopedNode(block);\n    block.empty?.visit(this);\n  }\n\n  visitForLoopBlockEmpty(block: ForLoopBlockEmpty) {\n    this.ingestScopedNode(block);\n  }\n\n  visitIfBlock(block: IfBlock) {\n    block.branches.forEach((node) => node.visit(this));\n  }\n\n  visitIfBlockBranch(block: IfBlockBranch) {\n    this.ingestScopedNode(block);\n  }\n\n  visitContent(content: Content) {\n    this.ingestScopedNode(content);\n  }\n\n  visitLetDeclaration(decl: LetDeclaration) {\n    this.maybeDeclare(decl);\n  }\n\n  // Unused visitors.\n  visitBoundAttribute(attr: BoundAttribute) {}\n  visitBoundEvent(event: BoundEvent) {}\n  visitBoundText(text: BoundText) {}\n  visitText(text: Text) {}\n  visitTextAttribute(attr: TextAttribute) {}\n  visitIcu(icu: Icu) {}\n  visitDeferredTrigger(trigger: DeferredTrigger) {}\n  visitUnknownBlock(block: UnknownBlock) {}\n\n  private maybeDeclare(thing: TemplateEntity) {\n    // Declare something with a name, as long as that name isn't taken.\n    if (!this.namedEntities.has(thing.name)) {\n      this.namedEntities.set(thing.name, thing);\n    }\n  }\n\n  /**\n   * Look up a variable within this `Scope`.\n   *\n   * This can recurse into a parent `Scope` if it's available.\n   */\n  lookup(name: string): TemplateEntity | null {\n    if (this.namedEntities.has(name)) {\n      // Found in the local scope.\n      return this.namedEntities.get(name)!;\n    } else if (this.parentScope !== null) {\n      // Not in the local scope, but there's a parent scope so check there.\n      return this.parentScope.lookup(name);\n    } else {\n      // At the top level and it wasn't found.\n      return null;\n    }\n  }\n\n  /**\n   * Get the child scope for a `ScopedNode`.\n   *\n   * This should always be defined.\n   */\n  getChildScope(node: ScopedNode): Scope {\n    const res = this.childScopes.get(node);\n    if (res === undefined) {\n      throw new Error(`Assertion error: child scope for ${node} not found`);\n    }\n    return res;\n  }\n\n  private ingestScopedNode(node: ScopedNode) {\n    const scope = new Scope(this, node);\n    scope.ingest(node);\n    this.childScopes.set(node, scope);\n  }\n}\n\n/**\n * Processes a template and matches directives on nodes (elements and templates).\n *\n * Usually used via the static `apply()` method.\n */\nclass DirectiveBinder<DirectiveT extends DirectiveMeta> implements Visitor {\n  // Indicates whether we are visiting elements within a `defer` block\n  private isInDeferBlock = false;\n\n  private constructor(\n    private matcher: SelectorMatcher<DirectiveT[]>,\n    private directives: Map<Element | Template, DirectiveT[]>,\n    private eagerDirectives: DirectiveT[],\n    private bindings: Map<\n      BoundAttribute | BoundEvent | TextAttribute,\n      DirectiveT | Element | Template\n    >,\n    private references: Map<\n      Reference,\n      {directive: DirectiveT; node: Element | Template} | Element | Template\n    >,\n  ) {}\n\n  /**\n   * Process a template (list of `Node`s) and perform directive matching against each node.\n   *\n   * @param template the list of template `Node`s to match (recursively).\n   * @param selectorMatcher a `SelectorMatcher` containing the directives that are in scope for\n   * this template.\n   * @returns three maps which contain information about directives in the template: the\n   * `directives` map which lists directives matched on each node, the `bindings` map which\n   * indicates which directives claimed which bindings (inputs, outputs, etc), and the `references`\n   * map which resolves #references (`Reference`s) within the template to the named directive or\n   * template node.\n   */\n  static apply<DirectiveT extends DirectiveMeta>(\n    template: Node[],\n    selectorMatcher: SelectorMatcher<DirectiveT[]>,\n  ): {\n    directives: Map<Element | Template, DirectiveT[]>;\n    eagerDirectives: DirectiveT[];\n    bindings: Map<BoundAttribute | BoundEvent | TextAttribute, DirectiveT | Element | Template>;\n    references: Map<\n      Reference,\n      {directive: DirectiveT; node: Element | Template} | Element | Template\n    >;\n  } {\n    const directives = new Map<Element | Template, DirectiveT[]>();\n    const bindings = new Map<\n      BoundAttribute | BoundEvent | TextAttribute,\n      DirectiveT | Element | Template\n    >();\n    const references = new Map<\n      Reference,\n      {directive: DirectiveT; node: Element | Template} | Element | Template\n    >();\n    const eagerDirectives: DirectiveT[] = [];\n    const matcher = new DirectiveBinder(\n      selectorMatcher,\n      directives,\n      eagerDirectives,\n      bindings,\n      references,\n    );\n    matcher.ingest(template);\n    return {directives, eagerDirectives, bindings, references};\n  }\n\n  private ingest(template: Node[]): void {\n    template.forEach((node) => node.visit(this));\n  }\n\n  visitElement(element: Element): void {\n    this.visitElementOrTemplate(element);\n  }\n\n  visitTemplate(template: Template): void {\n    this.visitElementOrTemplate(template);\n  }\n\n  visitElementOrTemplate(node: Element | Template): void {\n    // First, determine the HTML shape of the node for the purpose of directive matching.\n    // Do this by building up a `CssSelector` for the node.\n    const cssSelector = createCssSelectorFromNode(node);\n\n    // Next, use the `SelectorMatcher` to get the list of directives on the node.\n    const directives: DirectiveT[] = [];\n    this.matcher.match(cssSelector, (_selector, results) => directives.push(...results));\n    if (directives.length > 0) {\n      this.directives.set(node, directives);\n      if (!this.isInDeferBlock) {\n        this.eagerDirectives.push(...directives);\n      }\n    }\n\n    // Resolve any references that are created on this node.\n    node.references.forEach((ref) => {\n      let dirTarget: DirectiveT | null = null;\n\n      // If the reference expression is empty, then it matches the \"primary\" directive on the node\n      // (if there is one). Otherwise it matches the host node itself (either an element or\n      // <ng-template> node).\n      if (ref.value.trim() === '') {\n        // This could be a reference to a component if there is one.\n        dirTarget = directives.find((dir) => dir.isComponent) || null;\n      } else {\n        // This should be a reference to a directive exported via exportAs.\n        dirTarget =\n          directives.find(\n            (dir) => dir.exportAs !== null && dir.exportAs.some((value) => value === ref.value),\n          ) || null;\n        // Check if a matching directive was found.\n        if (dirTarget === null) {\n          // No matching directive was found - this reference points to an unknown target. Leave it\n          // unmapped.\n          return;\n        }\n      }\n\n      if (dirTarget !== null) {\n        // This reference points to a directive.\n        this.references.set(ref, {directive: dirTarget, node});\n      } else {\n        // This reference points to the node itself.\n        this.references.set(ref, node);\n      }\n    });\n\n    // Associate attributes/bindings on the node with directives or with the node itself.\n    type BoundNode = BoundAttribute | BoundEvent | TextAttribute;\n    const setAttributeBinding = (\n      attribute: BoundNode,\n      ioType: keyof Pick<DirectiveMeta, 'inputs' | 'outputs'>,\n    ) => {\n      const dir = directives.find((dir) => dir[ioType].hasBindingPropertyName(attribute.name));\n      const binding = dir !== undefined ? dir : node;\n      this.bindings.set(attribute, binding);\n    };\n\n    // Node inputs (bound attributes) and text attributes can be bound to an\n    // input on a directive.\n    node.inputs.forEach((input) => setAttributeBinding(input, 'inputs'));\n    node.attributes.forEach((attr) => setAttributeBinding(attr, 'inputs'));\n    if (node instanceof Template) {\n      node.templateAttrs.forEach((attr) => setAttributeBinding(attr, 'inputs'));\n    }\n    // Node outputs (bound events) can be bound to an output on a directive.\n    node.outputs.forEach((output) => setAttributeBinding(output, 'outputs'));\n\n    // Recurse into the node's children.\n    node.children.forEach((child) => child.visit(this));\n  }\n\n  visitDeferredBlock(deferred: DeferredBlock): void {\n    const wasInDeferBlock = this.isInDeferBlock;\n    this.isInDeferBlock = true;\n    deferred.children.forEach((child) => child.visit(this));\n    this.isInDeferBlock = wasInDeferBlock;\n\n    deferred.placeholder?.visit(this);\n    deferred.loading?.visit(this);\n    deferred.error?.visit(this);\n  }\n\n  visitDeferredBlockPlaceholder(block: DeferredBlockPlaceholder): void {\n    block.children.forEach((child) => child.visit(this));\n  }\n\n  visitDeferredBlockError(block: DeferredBlockError): void {\n    block.children.forEach((child) => child.visit(this));\n  }\n\n  visitDeferredBlockLoading(block: DeferredBlockLoading): void {\n    block.children.forEach((child) => child.visit(this));\n  }\n\n  visitSwitchBlock(block: SwitchBlock) {\n    block.cases.forEach((node) => node.visit(this));\n  }\n\n  visitSwitchBlockCase(block: SwitchBlockCase) {\n    block.children.forEach((node) => node.visit(this));\n  }\n\n  visitForLoopBlock(block: ForLoopBlock) {\n    block.item.visit(this);\n    block.contextVariables.forEach((v) => v.visit(this));\n    block.children.forEach((node) => node.visit(this));\n    block.empty?.visit(this);\n  }\n\n  visitForLoopBlockEmpty(block: ForLoopBlockEmpty) {\n    block.children.forEach((node) => node.visit(this));\n  }\n\n  visitIfBlock(block: IfBlock) {\n    block.branches.forEach((node) => node.visit(this));\n  }\n\n  visitIfBlockBranch(block: IfBlockBranch) {\n    block.expressionAlias?.visit(this);\n    block.children.forEach((node) => node.visit(this));\n  }\n\n  visitContent(content: Content): void {\n    content.children.forEach((child) => child.visit(this));\n  }\n\n  // Unused visitors.\n  visitVariable(variable: Variable): void {}\n  visitReference(reference: Reference): void {}\n  visitTextAttribute(attribute: TextAttribute): void {}\n  visitBoundAttribute(attribute: BoundAttribute): void {}\n  visitBoundEvent(attribute: BoundEvent): void {}\n  visitBoundAttributeOrEvent(node: BoundAttribute | BoundEvent) {}\n  visitText(text: Text): void {}\n  visitBoundText(text: BoundText): void {}\n  visitIcu(icu: Icu): void {}\n  visitDeferredTrigger(trigger: DeferredTrigger): void {}\n  visitUnknownBlock(block: UnknownBlock) {}\n  visitLetDeclaration(decl: LetDeclaration) {}\n}\n\n/**\n * Processes a template and extract metadata about expressions and symbols within.\n *\n * This is a companion to the `DirectiveBinder` that doesn't require knowledge of directives matched\n * within the template in order to operate.\n *\n * Expressions are visited by the superclass `RecursiveAstVisitor`, with custom logic provided\n * by overridden methods from that visitor.\n */\nclass TemplateBinder extends RecursiveAstVisitor implements Visitor {\n  private visitNode: (node: Node) => void;\n\n  private constructor(\n    private bindings: Map<AST, TemplateEntity>,\n    private symbols: Map<TemplateEntity, ScopedNode>,\n    private usedPipes: Set<string>,\n    private eagerPipes: Set<string>,\n    private deferBlocks: [DeferredBlock, Scope][],\n    private nestingLevel: Map<ScopedNode, number>,\n    private scope: Scope,\n    private rootNode: ScopedNode | null,\n    private level: number,\n  ) {\n    super();\n\n    // Save a bit of processing time by constructing this closure in advance.\n    this.visitNode = (node: Node) => node.visit(this);\n  }\n\n  // This method is defined to reconcile the type of TemplateBinder since both\n  // RecursiveAstVisitor and Visitor define the visit() method in their\n  // interfaces.\n  override visit(node: AST | Node, context?: any) {\n    if (node instanceof AST) {\n      node.visit(this, context);\n    } else {\n      node.visit(this);\n    }\n  }\n\n  /**\n   * Process a template and extract metadata about expressions and symbols within.\n   *\n   * @param nodes the nodes of the template to process\n   * @param scope the `Scope` of the template being processed.\n   * @returns three maps which contain metadata about the template: `expressions` which interprets\n   * special `AST` nodes in expressions as pointing to references or variables declared within the\n   * template, `symbols` which maps those variables and references to the nested `Template` which\n   * declares them, if any, and `nestingLevel` which associates each `Template` with a integer\n   * nesting level (how many levels deep within the template structure the `Template` is), starting\n   * at 1.\n   */\n  static applyWithScope(\n    nodes: Node[],\n    scope: Scope,\n  ): {\n    expressions: Map<AST, TemplateEntity>;\n    symbols: Map<TemplateEntity, Template>;\n    nestingLevel: Map<ScopedNode, number>;\n    usedPipes: Set<string>;\n    eagerPipes: Set<string>;\n    deferBlocks: [DeferredBlock, Scope][];\n  } {\n    const expressions = new Map<AST, TemplateEntity>();\n    const symbols = new Map<TemplateEntity, Template>();\n    const nestingLevel = new Map<ScopedNode, number>();\n    const usedPipes = new Set<string>();\n    const eagerPipes = new Set<string>();\n    const template = nodes instanceof Template ? nodes : null;\n    const deferBlocks: [DeferredBlock, Scope][] = [];\n    // The top-level template has nesting level 0.\n    const binder = new TemplateBinder(\n      expressions,\n      symbols,\n      usedPipes,\n      eagerPipes,\n      deferBlocks,\n      nestingLevel,\n      scope,\n      template,\n      0,\n    );\n    binder.ingest(nodes);\n    return {expressions, symbols, nestingLevel, usedPipes, eagerPipes, deferBlocks};\n  }\n\n  private ingest(nodeOrNodes: ScopedNode | Node[]): void {\n    if (nodeOrNodes instanceof Template) {\n      // For <ng-template>s, process only variables and child nodes. Inputs, outputs, templateAttrs,\n      // and references were all processed in the scope of the containing template.\n      nodeOrNodes.variables.forEach(this.visitNode);\n      nodeOrNodes.children.forEach(this.visitNode);\n\n      // Set the nesting level.\n      this.nestingLevel.set(nodeOrNodes, this.level);\n    } else if (nodeOrNodes instanceof IfBlockBranch) {\n      if (nodeOrNodes.expressionAlias !== null) {\n        this.visitNode(nodeOrNodes.expressionAlias);\n      }\n      nodeOrNodes.children.forEach(this.visitNode);\n      this.nestingLevel.set(nodeOrNodes, this.level);\n    } else if (nodeOrNodes instanceof ForLoopBlock) {\n      this.visitNode(nodeOrNodes.item);\n      nodeOrNodes.contextVariables.forEach((v) => this.visitNode(v));\n      nodeOrNodes.trackBy.visit(this);\n      nodeOrNodes.children.forEach(this.visitNode);\n      this.nestingLevel.set(nodeOrNodes, this.level);\n    } else if (nodeOrNodes instanceof DeferredBlock) {\n      if (this.scope.rootNode !== nodeOrNodes) {\n        throw new Error(\n          `Assertion error: resolved incorrect scope for deferred block ${nodeOrNodes}`,\n        );\n      }\n      this.deferBlocks.push([nodeOrNodes, this.scope]);\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n      this.nestingLevel.set(nodeOrNodes, this.level);\n    } else if (\n      nodeOrNodes instanceof SwitchBlockCase ||\n      nodeOrNodes instanceof ForLoopBlockEmpty ||\n      nodeOrNodes instanceof DeferredBlockError ||\n      nodeOrNodes instanceof DeferredBlockPlaceholder ||\n      nodeOrNodes instanceof DeferredBlockLoading ||\n      nodeOrNodes instanceof Content\n    ) {\n      nodeOrNodes.children.forEach((node) => node.visit(this));\n      this.nestingLevel.set(nodeOrNodes, this.level);\n    } else {\n      // Visit each node from the top-level template.\n      nodeOrNodes.forEach(this.visitNode);\n    }\n  }\n\n  visitElement(element: Element) {\n    // Visit the inputs, outputs, and children of the element.\n    element.inputs.forEach(this.visitNode);\n    element.outputs.forEach(this.visitNode);\n    element.children.forEach(this.visitNode);\n    element.references.forEach(this.visitNode);\n  }\n\n  visitTemplate(template: Template) {\n    // First, visit inputs, outputs and template attributes of the template node.\n    template.inputs.forEach(this.visitNode);\n    template.outputs.forEach(this.visitNode);\n    template.templateAttrs.forEach(this.visitNode);\n    template.references.forEach(this.visitNode);\n\n    // Next, recurse into the template.\n    this.ingestScopedNode(template);\n  }\n\n  visitVariable(variable: Variable) {\n    // Register the `Variable` as a symbol in the current `Template`.\n    if (this.rootNode !== null) {\n      this.symbols.set(variable, this.rootNode);\n    }\n  }\n\n  visitReference(reference: Reference) {\n    // Register the `Reference` as a symbol in the current `Template`.\n    if (this.rootNode !== null) {\n      this.symbols.set(reference, this.rootNode);\n    }\n  }\n\n  // Unused template visitors\n  visitText(text: Text) {}\n  visitTextAttribute(attribute: TextAttribute) {}\n  visitUnknownBlock(block: UnknownBlock) {}\n  visitDeferredTrigger(): void {}\n  visitIcu(icu: Icu): void {\n    Object.keys(icu.vars).forEach((key) => icu.vars[key].visit(this));\n    Object.keys(icu.placeholders).forEach((key) => icu.placeholders[key].visit(this));\n  }\n\n  // The remaining visitors are concerned with processing AST expressions within template bindings\n\n  visitBoundAttribute(attribute: BoundAttribute) {\n    attribute.value.visit(this);\n  }\n\n  visitBoundEvent(event: BoundEvent) {\n    event.handler.visit(this);\n  }\n\n  visitDeferredBlock(deferred: DeferredBlock) {\n    this.ingestScopedNode(deferred);\n    deferred.triggers.when?.value.visit(this);\n    deferred.prefetchTriggers.when?.value.visit(this);\n    deferred.placeholder && this.visitNode(deferred.placeholder);\n    deferred.loading && this.visitNode(deferred.loading);\n    deferred.error && this.visitNode(deferred.error);\n  }\n\n  visitDeferredBlockPlaceholder(block: DeferredBlockPlaceholder) {\n    this.ingestScopedNode(block);\n  }\n\n  visitDeferredBlockError(block: DeferredBlockError) {\n    this.ingestScopedNode(block);\n  }\n\n  visitDeferredBlockLoading(block: DeferredBlockLoading) {\n    this.ingestScopedNode(block);\n  }\n\n  visitSwitchBlock(block: SwitchBlock) {\n    block.expression.visit(this);\n    block.cases.forEach(this.visitNode);\n  }\n\n  visitSwitchBlockCase(block: SwitchBlockCase) {\n    block.expression?.visit(this);\n    this.ingestScopedNode(block);\n  }\n\n  visitForLoopBlock(block: ForLoopBlock) {\n    block.expression.visit(this);\n    this.ingestScopedNode(block);\n    block.empty?.visit(this);\n  }\n\n  visitForLoopBlockEmpty(block: ForLoopBlockEmpty) {\n    this.ingestScopedNode(block);\n  }\n\n  visitIfBlock(block: IfBlock) {\n    block.branches.forEach((node) => node.visit(this));\n  }\n\n  visitIfBlockBranch(block: IfBlockBranch) {\n    block.expression?.visit(this);\n    this.ingestScopedNode(block);\n  }\n\n  visitContent(content: Content) {\n    this.ingestScopedNode(content);\n  }\n\n  visitBoundText(text: BoundText) {\n    text.value.visit(this);\n  }\n\n  visitLetDeclaration(decl: LetDeclaration) {\n    decl.value.visit(this);\n\n    if (this.rootNode !== null) {\n      this.symbols.set(decl, this.rootNode);\n    }\n  }\n\n  override visitPipe(ast: BindingPipe, context: any): any {\n    this.usedPipes.add(ast.name);\n    if (!this.scope.isDeferred) {\n      this.eagerPipes.add(ast.name);\n    }\n    return super.visitPipe(ast, context);\n  }\n\n  // These five types of AST expressions can refer to expression roots, which could be variables\n  // or references in the current scope.\n\n  override visitPropertyRead(ast: PropertyRead, context: any): any {\n    this.maybeMap(ast, ast.name);\n    return super.visitPropertyRead(ast, context);\n  }\n\n  override visitSafePropertyRead(ast: SafePropertyRead, context: any): any {\n    this.maybeMap(ast, ast.name);\n    return super.visitSafePropertyRead(ast, context);\n  }\n\n  override visitPropertyWrite(ast: PropertyWrite, context: any): any {\n    this.maybeMap(ast, ast.name);\n    return super.visitPropertyWrite(ast, context);\n  }\n\n  private ingestScopedNode(node: ScopedNode) {\n    const childScope = this.scope.getChildScope(node);\n    const binder = new TemplateBinder(\n      this.bindings,\n      this.symbols,\n      this.usedPipes,\n      this.eagerPipes,\n      this.deferBlocks,\n      this.nestingLevel,\n      childScope,\n      node,\n      this.level + 1,\n    );\n    binder.ingest(node);\n  }\n\n  private maybeMap(ast: PropertyRead | SafePropertyRead | PropertyWrite, name: string): void {\n    // If the receiver of the expression isn't the `ImplicitReceiver`, this isn't the root of an\n    // `AST` expression that maps to a `Variable` or `Reference`.\n    if (!(ast.receiver instanceof ImplicitReceiver)) {\n      return;\n    }\n\n    // Check whether the name exists in the current scope. If so, map it. Otherwise, the name is\n    // probably a property on the top-level component context.\n    const target = this.scope.lookup(name);\n\n    // It's not allowed to read template entities via `this`, however it previously worked by\n    // accident (see #55115). Since `@let` declarations are new, we can fix it from the beginning,\n    // whereas pre-existing template entities will be fixed in #55115.\n    if (target instanceof LetDeclaration && ast.receiver instanceof ThisReceiver) {\n      return;\n    }\n\n    if (target !== null) {\n      this.bindings.set(ast, target);\n    }\n  }\n}\n\n/**\n * Metadata container for a `Target` that allows queries for specific bits of metadata.\n *\n * See `BoundTarget` for documentation on the individual methods.\n */\nexport class R3BoundTarget<DirectiveT extends DirectiveMeta> implements BoundTarget<DirectiveT> {\n  /** Deferred blocks, ordered as they appear in the template. */\n  private deferredBlocks: DeferredBlock[];\n\n  /** Map of deferred blocks to their scope. */\n  private deferredScopes: Map<DeferredBlock, Scope>;\n\n  constructor(\n    readonly target: Target,\n    private directives: Map<Element | Template, DirectiveT[]>,\n    private eagerDirectives: DirectiveT[],\n    private bindings: Map<\n      BoundAttribute | BoundEvent | TextAttribute,\n      DirectiveT | Element | Template\n    >,\n    private references: Map<\n      BoundAttribute | BoundEvent | Reference | TextAttribute,\n      {directive: DirectiveT; node: Element | Template} | Element | Template\n    >,\n    private exprTargets: Map<AST, TemplateEntity>,\n    private symbols: Map<TemplateEntity, Template>,\n    private nestingLevel: Map<ScopedNode, number>,\n    private scopedNodeEntities: Map<ScopedNode | null, ReadonlySet<TemplateEntity>>,\n    private usedPipes: Set<string>,\n    private eagerPipes: Set<string>,\n    rawDeferred: [DeferredBlock, Scope][],\n  ) {\n    this.deferredBlocks = rawDeferred.map((current) => current[0]);\n    this.deferredScopes = new Map(rawDeferred);\n  }\n\n  getEntitiesInScope(node: ScopedNode | null): ReadonlySet<TemplateEntity> {\n    return this.scopedNodeEntities.get(node) ?? new Set();\n  }\n\n  getDirectivesOfNode(node: Element | Template): DirectiveT[] | null {\n    return this.directives.get(node) || null;\n  }\n\n  getReferenceTarget(ref: Reference): ReferenceTarget<DirectiveT> | null {\n    return this.references.get(ref) || null;\n  }\n\n  getConsumerOfBinding(\n    binding: BoundAttribute | BoundEvent | TextAttribute,\n  ): DirectiveT | Element | Template | null {\n    return this.bindings.get(binding) || null;\n  }\n\n  getExpressionTarget(expr: AST): TemplateEntity | null {\n    return this.exprTargets.get(expr) || null;\n  }\n\n  getDefinitionNodeOfSymbol(symbol: TemplateEntity): ScopedNode | null {\n    return this.symbols.get(symbol) || null;\n  }\n\n  getNestingLevel(node: ScopedNode): number {\n    return this.nestingLevel.get(node) || 0;\n  }\n\n  getUsedDirectives(): DirectiveT[] {\n    const set = new Set<DirectiveT>();\n    this.directives.forEach((dirs) => dirs.forEach((dir) => set.add(dir)));\n    return Array.from(set.values());\n  }\n\n  getEagerlyUsedDirectives(): DirectiveT[] {\n    const set = new Set<DirectiveT>(this.eagerDirectives);\n    return Array.from(set.values());\n  }\n\n  getUsedPipes(): string[] {\n    return Array.from(this.usedPipes);\n  }\n\n  getEagerlyUsedPipes(): string[] {\n    return Array.from(this.eagerPipes);\n  }\n\n  getDeferBlocks(): DeferredBlock[] {\n    return this.deferredBlocks;\n  }\n\n  getDeferredTriggerTarget(block: DeferredBlock, trigger: DeferredTrigger): Element | null {\n    // Only triggers that refer to DOM nodes can be resolved.\n    if (\n      !(trigger instanceof InteractionDeferredTrigger) &&\n      !(trigger instanceof ViewportDeferredTrigger) &&\n      !(trigger instanceof HoverDeferredTrigger)\n    ) {\n      return null;\n    }\n\n    const name = trigger.reference;\n\n    if (name === null) {\n      let trigger: Element | null = null;\n\n      if (block.placeholder !== null) {\n        for (const child of block.placeholder.children) {\n          // Skip over comment nodes. Currently by default the template parser doesn't capture\n          // comments, but we have a safeguard here just in case since it can be enabled.\n          if (child instanceof Comment) {\n            continue;\n          }\n\n          // We can only infer the trigger if there's one root element node. Any other\n          // nodes at the root make it so that we can't infer the trigger anymore.\n          if (trigger !== null) {\n            return null;\n          }\n\n          if (child instanceof Element) {\n            trigger = child;\n          }\n        }\n      }\n\n      return trigger;\n    }\n\n    const outsideRef = this.findEntityInScope(block, name);\n\n    // First try to resolve the target in the scope of the main deferred block. Note that we\n    // skip triggers defined inside the main block itself, because they might not exist yet.\n    if (outsideRef instanceof Reference && this.getDefinitionNodeOfSymbol(outsideRef) !== block) {\n      const target = this.getReferenceTarget(outsideRef);\n\n      if (target !== null) {\n        return this.referenceTargetToElement(target);\n      }\n    }\n\n    // If the trigger couldn't be found in the main block, check the\n    // placeholder block which is shown before the main block has loaded.\n    if (block.placeholder !== null) {\n      const refInPlaceholder = this.findEntityInScope(block.placeholder, name);\n      const targetInPlaceholder =\n        refInPlaceholder instanceof Reference ? this.getReferenceTarget(refInPlaceholder) : null;\n\n      if (targetInPlaceholder !== null) {\n        return this.referenceTargetToElement(targetInPlaceholder);\n      }\n    }\n\n    return null;\n  }\n\n  isDeferred(element: Element): boolean {\n    for (const block of this.deferredBlocks) {\n      if (!this.deferredScopes.has(block)) {\n        continue;\n      }\n\n      const stack: Scope[] = [this.deferredScopes.get(block)!];\n\n      while (stack.length > 0) {\n        const current = stack.pop()!;\n\n        if (current.elementsInScope.has(element)) {\n          return true;\n        }\n\n        stack.push(...current.childScopes.values());\n      }\n    }\n\n    return false;\n  }\n\n  /**\n   * Finds an entity with a specific name in a scope.\n   * @param rootNode Root node of the scope.\n   * @param name Name of the entity.\n   */\n  private findEntityInScope(rootNode: ScopedNode, name: string): TemplateEntity | null {\n    const entities = this.getEntitiesInScope(rootNode);\n\n    for (const entity of entities) {\n      if (entity.name === name) {\n        return entity;\n      }\n    }\n\n    return null;\n  }\n\n  /** Coerces a `ReferenceTarget` to an `Element`, if possible. */\n  private referenceTargetToElement(target: ReferenceTarget<DirectiveT>): Element | null {\n    if (target instanceof Element) {\n      return target;\n    }\n\n    if (target instanceof Template) {\n      return null;\n    }\n\n    return this.referenceTargetToElement(target.node);\n  }\n}\n\nfunction extractScopedNodeEntities(rootScope: Scope): Map<ScopedNode | null, Set<TemplateEntity>> {\n  const entityMap = new Map<ScopedNode | null, Map<string, TemplateEntity>>();\n\n  function extractScopeEntities(scope: Scope): Map<string, TemplateEntity> {\n    if (entityMap.has(scope.rootNode)) {\n      return entityMap.get(scope.rootNode)!;\n    }\n\n    const currentEntities = scope.namedEntities;\n\n    let entities: Map<string, TemplateEntity>;\n    if (scope.parentScope !== null) {\n      entities = new Map([...extractScopeEntities(scope.parentScope), ...currentEntities]);\n    } else {\n      entities = new Map(currentEntities);\n    }\n\n    entityMap.set(scope.rootNode, entities);\n    return entities;\n  }\n\n  const scopesToProcess: Scope[] = [rootScope];\n  while (scopesToProcess.length > 0) {\n    const scope = scopesToProcess.pop()!;\n    for (const childScope of scope.childScopes.values()) {\n      scopesToProcess.push(childScope);\n    }\n    extractScopeEntities(scope);\n  }\n\n  const templateEntities = new Map<ScopedNode | null, Set<TemplateEntity>>();\n  for (const [template, entities] of entityMap) {\n    templateEntities.set(template, new Set(entities.values()));\n  }\n  return templateEntities;\n}\n"]}
|