/** * @license * Copyright Google Inc. All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import { __extends } from "tslib"; import { AST, ImplicitReceiver, RecursiveAstVisitor } from '../../expression_parser/ast'; import { Template } from '../r3_ast'; import { createCssSelector } from './template'; import { getAttrsForDirectiveMatching } from './util'; /** * Processes `Target`s with a given set of directives and performs a binding operation, which * returns an object similar to TypeScript's `ts.TypeChecker` that contains knowledge about the * target. */ var R3TargetBinder = /** @class */ (function () { function R3TargetBinder(directiveMatcher) { this.directiveMatcher = directiveMatcher; } /** * Perform a binding operation on the given `Target` and return a `BoundTarget` which contains * metadata about the types referenced in the template. */ R3TargetBinder.prototype.bind = function (target) { if (!target.template) { // TODO(alxhub): handle targets which contain things like HostBindings, etc. throw new Error('Binding without a template not yet supported'); } // First, parse the template into a `Scope` structure. This operation captures the syntactic // scopes in the template and makes them available for later use. var scope = Scope.apply(target.template); // Next, perform directive matching on the template using the `DirectiveBinder`. This returns: // - directives: Map of nodes (elements & ng-templates) to the directives on them. // - bindings: Map of inputs, outputs, and attributes to the directive/element that claims // them. TODO(alxhub): handle multiple directives claiming an input/output/etc. // - references: Map of #references to their targets. var _a = DirectiveBinder.apply(target.template, this.directiveMatcher), directives = _a.directives, bindings = _a.bindings, references = _a.references; // Finally, run the TemplateBinder to bind references, variables, and other entities within the // template. This extracts all the metadata that doesn't depend on directive matching. var _b = TemplateBinder.apply(target.template, scope), expressions = _b.expressions, symbols = _b.symbols, nestingLevel = _b.nestingLevel, usedPipes = _b.usedPipes; return new R3BoundTarget(target, directives, bindings, references, expressions, symbols, nestingLevel, usedPipes); }; return R3TargetBinder; }()); export { R3TargetBinder }; /** * Represents a binding scope within a template. * * Any variables, references, or other named entities declared within the template will * be captured and available by name in `namedEntities`. Additionally, child templates will * be analyzed and have their child `Scope`s available in `childScopes`. */ var Scope = /** @class */ (function () { function Scope(parentScope) { this.parentScope = parentScope; /** * Named members of the `Scope`, such as `Reference`s or `Variable`s. */ this.namedEntities = new Map(); /** * Child `Scope`s for immediately nested `Template`s. */ this.childScopes = new Map(); } /** * Process a template (either as a `Template` sub-template with variables, or a plain array of * template `Node`s) and construct its `Scope`. */ Scope.apply = function (template) { var scope = new Scope(); scope.ingest(template); return scope; }; /** * Internal method to process the template and populate the `Scope`. */ Scope.prototype.ingest = function (template) { var _this = this; if (template instanceof Template) { // Variables on an <ng-template> are defined in the inner scope. template.variables.forEach(function (node) { return _this.visitVariable(node); }); // Process the nodes of the template. template.children.forEach(function (node) { return node.visit(_this); }); } else { // No overarching `Template` instance, so process the nodes directly. template.forEach(function (node) { return node.visit(_this); }); } }; Scope.prototype.visitElement = function (element) { var _this = this; // `Element`s in the template may have `Reference`s which are captured in the scope. element.references.forEach(function (node) { return _this.visitReference(node); }); // Recurse into the `Element`'s children. element.children.forEach(function (node) { return node.visit(_this); }); }; Scope.prototype.visitTemplate = function (template) { var _this = this; // References on a <ng-template> are defined in the outer scope, so capture them before // processing the template's child scope. template.references.forEach(function (node) { return _this.visitReference(node); }); // Next, create an inner scope and process the template within it. var scope = new Scope(this); scope.ingest(template); this.childScopes.set(template, scope); }; Scope.prototype.visitVariable = function (variable) { // Declare the variable if it's not already. this.maybeDeclare(variable); }; Scope.prototype.visitReference = function (reference) { // Declare the variable if it's not already. this.maybeDeclare(reference); }; // Unused visitors. Scope.prototype.visitContent = function (content) { }; Scope.prototype.visitBoundAttribute = function (attr) { }; Scope.prototype.visitBoundEvent = function (event) { }; Scope.prototype.visitBoundText = function (text) { }; Scope.prototype.visitText = function (text) { }; Scope.prototype.visitTextAttribute = function (attr) { }; Scope.prototype.visitIcu = function (icu) { }; Scope.prototype.maybeDeclare = function (thing) { // Declare something with a name, as long as that name isn't taken. if (!this.namedEntities.has(thing.name)) { this.namedEntities.set(thing.name, thing); } }; /** * Look up a variable within this `Scope`. * * This can recurse into a parent `Scope` if it's available. */ Scope.prototype.lookup = function (name) { if (this.namedEntities.has(name)) { // Found in the local scope. return this.namedEntities.get(name); } else if (this.parentScope !== undefined) { // Not in the local scope, but there's a parent scope so check there. return this.parentScope.lookup(name); } else { // At the top level and it wasn't found. return null; } }; /** * Get the child scope for a `Template`. * * This should always be defined. */ Scope.prototype.getChildScope = function (template) { var res = this.childScopes.get(template); if (res === undefined) { throw new Error("Assertion error: child scope for " + template + " not found"); } return res; }; return Scope; }()); /** * Processes a template and matches directives on nodes (elements and templates). * * Usually used via the static `apply()` method. */ var DirectiveBinder = /** @class */ (function () { function DirectiveBinder(matcher, directives, bindings, references) { this.matcher = matcher; this.directives = directives; this.bindings = bindings; this.references = references; } /** * Process a template (list of `Node`s) and perform directive matching against each node. * * @param template the list of template `Node`s to match (recursively). * @param selectorMatcher a `SelectorMatcher` containing the directives that are in scope for * this template. * @returns three maps which contain information about directives in the template: the * `directives` map which lists directives matched on each node, the `bindings` map which * indicates which directives claimed which bindings (inputs, outputs, etc), and the `references` * map which resolves #references (`Reference`s) within the template to the named directive or * template node. */ DirectiveBinder.apply = function (template, selectorMatcher) { var directives = new Map(); var bindings = new Map(); var references = new Map(); var matcher = new DirectiveBinder(selectorMatcher, directives, bindings, references); matcher.ingest(template); return { directives: directives, bindings: bindings, references: references }; }; DirectiveBinder.prototype.ingest = function (template) { var _this = this; template.forEach(function (node) { return node.visit(_this); }); }; DirectiveBinder.prototype.visitElement = function (element) { this.visitElementOrTemplate(element.name, element); }; DirectiveBinder.prototype.visitTemplate = function (template) { this.visitElementOrTemplate('ng-template', template); }; DirectiveBinder.prototype.visitElementOrTemplate = function (elementName, node) { var _this = this; // First, determine the HTML shape of the node for the purpose of directive matching. // Do this by building up a `CssSelector` for the node. var cssSelector = createCssSelector(elementName, getAttrsForDirectiveMatching(node)); // Next, use the `SelectorMatcher` to get the list of directives on the node. var directives = []; this.matcher.match(cssSelector, function (_, directive) { return directives.push(directive); }); if (directives.length > 0) { this.directives.set(node, directives); } // Resolve any references that are created on this node. node.references.forEach(function (ref) { var dirTarget = null; // If the reference expression is empty, then it matches the "primary" directive on the node // (if there is one). Otherwise it matches the host node itself (either an element or // <ng-template> node). if (ref.value.trim() === '') { // This could be a reference to a component if there is one. dirTarget = directives.find(function (dir) { return dir.isComponent; }) || null; } else { // This should be a reference to a directive exported via exportAs. dirTarget = directives.find(function (dir) { return dir.exportAs !== null && dir.exportAs.some(function (value) { return value === ref.value; }); }) || null; // Check if a matching directive was found. if (dirTarget === null) { // No matching directive was found - this reference points to an unknown target. Leave it // unmapped. return; } } if (dirTarget !== null) { // This reference points to a directive. _this.references.set(ref, { directive: dirTarget, node: node }); } else { // This reference points to the node itself. _this.references.set(ref, node); } }); var setAttributeBinding = function (attribute, ioType) { var dir = directives.find(function (dir) { return dir[ioType].hasOwnProperty(attribute.name); }); var binding = dir !== undefined ? dir : node; _this.bindings.set(attribute, binding); }; // Node inputs (bound attributes) and text attributes can be bound to an // input on a directive. node.inputs.forEach(function (input) { return setAttributeBinding(input, 'inputs'); }); node.attributes.forEach(function (attr) { return setAttributeBinding(attr, 'inputs'); }); if (node instanceof Template) { node.templateAttrs.forEach(function (attr) { return setAttributeBinding(attr, 'inputs'); }); } // Node outputs (bound events) can be bound to an output on a directive. node.outputs.forEach(function (output) { return setAttributeBinding(output, 'outputs'); }); // Recurse into the node's children. node.children.forEach(function (child) { return child.visit(_this); }); }; // Unused visitors. DirectiveBinder.prototype.visitContent = function (content) { }; DirectiveBinder.prototype.visitVariable = function (variable) { }; DirectiveBinder.prototype.visitReference = function (reference) { }; DirectiveBinder.prototype.visitTextAttribute = function (attribute) { }; DirectiveBinder.prototype.visitBoundAttribute = function (attribute) { }; DirectiveBinder.prototype.visitBoundEvent = function (attribute) { }; DirectiveBinder.prototype.visitBoundAttributeOrEvent = function (node) { }; DirectiveBinder.prototype.visitText = function (text) { }; DirectiveBinder.prototype.visitBoundText = function (text) { }; DirectiveBinder.prototype.visitIcu = function (icu) { }; return DirectiveBinder; }()); /** * Processes a template and extract metadata about expressions and symbols within. * * This is a companion to the `DirectiveBinder` that doesn't require knowledge of directives matched * within the template in order to operate. * * Expressions are visited by the superclass `RecursiveAstVisitor`, with custom logic provided * by overridden methods from that visitor. */ var TemplateBinder = /** @class */ (function (_super) { __extends(TemplateBinder, _super); function TemplateBinder(bindings, symbols, usedPipes, nestingLevel, scope, template, level) { var _this = _super.call(this) || this; _this.bindings = bindings; _this.symbols = symbols; _this.usedPipes = usedPipes; _this.nestingLevel = nestingLevel; _this.scope = scope; _this.template = template; _this.level = level; _this.pipesUsed = []; // Save a bit of processing time by constructing this closure in advance. _this.visitNode = function (node) { return node.visit(_this); }; return _this; } // This method is defined to reconcile the type of TemplateBinder since both // RecursiveAstVisitor and Visitor define the visit() method in their // interfaces. TemplateBinder.prototype.visit = function (node, context) { if (node instanceof AST) { node.visit(this, context); } else { node.visit(this); } }; /** * Process a template and extract metadata about expressions and symbols within. * * @param template the nodes of the template to process * @param scope the `Scope` of the template being processed. * @returns three maps which contain metadata about the template: `expressions` which interprets * special `AST` nodes in expressions as pointing to references or variables declared within the * template, `symbols` which maps those variables and references to the nested `Template` which * declares them, if any, and `nestingLevel` which associates each `Template` with a integer * nesting level (how many levels deep within the template structure the `Template` is), starting * at 1. */ TemplateBinder.apply = function (template, scope) { var expressions = new Map(); var symbols = new Map(); var nestingLevel = new Map(); var usedPipes = new Set(); // The top-level template has nesting level 0. var binder = new TemplateBinder(expressions, symbols, usedPipes, nestingLevel, scope, template instanceof Template ? template : null, 0); binder.ingest(template); return { expressions: expressions, symbols: symbols, nestingLevel: nestingLevel, usedPipes: usedPipes }; }; TemplateBinder.prototype.ingest = function (template) { if (template instanceof Template) { // For <ng-template>s, process only variables and child nodes. Inputs, outputs, templateAttrs, // and references were all processed in the scope of the containing template. template.variables.forEach(this.visitNode); template.children.forEach(this.visitNode); // Set the nesting level. this.nestingLevel.set(template, this.level); } else { // Visit each node from the top-level template. template.forEach(this.visitNode); } }; TemplateBinder.prototype.visitElement = function (element) { // Visit the inputs, outputs, and children of the element. element.inputs.forEach(this.visitNode); element.outputs.forEach(this.visitNode); element.children.forEach(this.visitNode); }; TemplateBinder.prototype.visitTemplate = function (template) { // First, visit inputs, outputs and template attributes of the template node. template.inputs.forEach(this.visitNode); template.outputs.forEach(this.visitNode); template.templateAttrs.forEach(this.visitNode); // References are also evaluated in the outer context. template.references.forEach(this.visitNode); // Next, recurse into the template using its scope, and bumping the nesting level up by one. var childScope = this.scope.getChildScope(template); var binder = new TemplateBinder(this.bindings, this.symbols, this.usedPipes, this.nestingLevel, childScope, template, this.level + 1); binder.ingest(template); }; TemplateBinder.prototype.visitVariable = function (variable) { // Register the `Variable` as a symbol in the current `Template`. if (this.template !== null) { this.symbols.set(variable, this.template); } }; TemplateBinder.prototype.visitReference = function (reference) { // Register the `Reference` as a symbol in the current `Template`. if (this.template !== null) { this.symbols.set(reference, this.template); } }; // Unused template visitors TemplateBinder.prototype.visitText = function (text) { }; TemplateBinder.prototype.visitContent = function (content) { }; TemplateBinder.prototype.visitTextAttribute = function (attribute) { }; TemplateBinder.prototype.visitIcu = function (icu) { }; // The remaining visitors are concerned with processing AST expressions within template bindings TemplateBinder.prototype.visitBoundAttribute = function (attribute) { attribute.value.visit(this); }; TemplateBinder.prototype.visitBoundEvent = function (event) { event.handler.visit(this); }; TemplateBinder.prototype.visitBoundText = function (text) { text.value.visit(this); }; TemplateBinder.prototype.visitPipe = function (ast, context) { this.usedPipes.add(ast.name); return _super.prototype.visitPipe.call(this, ast, context); }; // These five types of AST expressions can refer to expression roots, which could be variables // or references in the current scope. TemplateBinder.prototype.visitPropertyRead = function (ast, context) { this.maybeMap(context, ast, ast.name); return _super.prototype.visitPropertyRead.call(this, ast, context); }; TemplateBinder.prototype.visitSafePropertyRead = function (ast, context) { this.maybeMap(context, ast, ast.name); return _super.prototype.visitSafePropertyRead.call(this, ast, context); }; TemplateBinder.prototype.visitPropertyWrite = function (ast, context) { this.maybeMap(context, ast, ast.name); return _super.prototype.visitPropertyWrite.call(this, ast, context); }; TemplateBinder.prototype.visitMethodCall = function (ast, context) { this.maybeMap(context, ast, ast.name); return _super.prototype.visitMethodCall.call(this, ast, context); }; TemplateBinder.prototype.visitSafeMethodCall = function (ast, context) { this.maybeMap(context, ast, ast.name); return _super.prototype.visitSafeMethodCall.call(this, ast, context); }; TemplateBinder.prototype.maybeMap = function (scope, ast, name) { // If the receiver of the expression isn't the `ImplicitReceiver`, this isn't the root of an // `AST` expression that maps to a `Variable` or `Reference`. if (!(ast.receiver instanceof ImplicitReceiver)) { return; } // Check whether the name exists in the current scope. If so, map it. Otherwise, the name is // probably a property on the top-level component context. var target = this.scope.lookup(name); if (target !== null) { this.bindings.set(ast, target); } }; return TemplateBinder; }(RecursiveAstVisitor)); /** * Metadata container for a `Target` that allows queries for specific bits of metadata. * * See `BoundTarget` for documentation on the individual methods. */ var R3BoundTarget = /** @class */ (function () { function R3BoundTarget(target, directives, bindings, references, exprTargets, symbols, nestingLevel, usedPipes) { this.target = target; this.directives = directives; this.bindings = bindings; this.references = references; this.exprTargets = exprTargets; this.symbols = symbols; this.nestingLevel = nestingLevel; this.usedPipes = usedPipes; } R3BoundTarget.prototype.getDirectivesOfNode = function (node) { return this.directives.get(node) || null; }; R3BoundTarget.prototype.getReferenceTarget = function (ref) { return this.references.get(ref) || null; }; R3BoundTarget.prototype.getConsumerOfBinding = function (binding) { return this.bindings.get(binding) || null; }; R3BoundTarget.prototype.getExpressionTarget = function (expr) { return this.exprTargets.get(expr) || null; }; R3BoundTarget.prototype.getTemplateOfSymbol = function (symbol) { return this.symbols.get(symbol) || null; }; R3BoundTarget.prototype.getNestingLevel = function (template) { return this.nestingLevel.get(template) || 0; }; R3BoundTarget.prototype.getUsedDirectives = function () { var set = new Set(); this.directives.forEach(function (dirs) { return dirs.forEach(function (dir) { return set.add(dir); }); }); return Array.from(set.values()); }; R3BoundTarget.prototype.getUsedPipes = function () { return Array.from(this.usedPipes); }; return R3BoundTarget; }()); export { R3BoundTarget }; //# 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 Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {AST, BindingPipe, ImplicitReceiver, MethodCall, PropertyRead, PropertyWrite, RecursiveAstVisitor, SafeMethodCall, SafePropertyRead} from '../../expression_parser/ast';\nimport {CssSelector, SelectorMatcher} from '../../selector';\nimport {BoundAttribute, BoundEvent, BoundText, Content, Element, Icu, Node, Reference, Template, Text, TextAttribute, Variable, Visitor} from '../r3_ast';\n\nimport {BoundTarget, DirectiveMeta, Target, TargetBinder} from './t2_api';\nimport {createCssSelector} from './template';\nimport {getAttrsForDirectiveMatching} from './util';\n\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    // 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, bindings, references} =\n        DirectiveBinder.apply(target.template, this.directiveMatcher);\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} =\n        TemplateBinder.apply(target.template, scope);\n    return new R3BoundTarget(\n        target, directives, bindings, references, expressions, symbols, nestingLevel, usedPipes);\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, Reference|Variable>();\n\n  /**\n   * Child `Scope`s for immediately nested `Template`s.\n   */\n  readonly childScopes = new Map<Template, Scope>();\n\n  private constructor(readonly parentScope?: Scope) {}\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: Template|Node[]): Scope {\n    const scope = new Scope();\n    scope.ingest(template);\n    return scope;\n  }\n\n  /**\n   * Internal method to process the template and populate the `Scope`.\n   */\n  private ingest(template: Template|Node[]): void {\n    if (template instanceof Template) {\n      // Variables on an <ng-template> are defined in the inner scope.\n      template.variables.forEach(node => this.visitVariable(node));\n\n      // Process the nodes of the template.\n      template.children.forEach(node => node.visit(this));\n    } else {\n      // No overarching `Template` instance, so process the nodes directly.\n      template.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\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    const scope = new Scope(this);\n    scope.ingest(template);\n    this.childScopes.set(template, scope);\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  // Unused visitors.\n  visitContent(content: Content) {}\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\n  private maybeDeclare(thing: Reference|Variable) {\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): Reference|Variable|null {\n    if (this.namedEntities.has(name)) {\n      // Found in the local scope.\n      return this.namedEntities.get(name) !;\n    } else if (this.parentScope !== undefined) {\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 `Template`.\n   *\n   * This should always be defined.\n   */\n  getChildScope(template: Template): Scope {\n    const res = this.childScopes.get(template);\n    if (res === undefined) {\n      throw new Error(`Assertion error: child scope for ${template} not found`);\n    }\n    return res;\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  constructor(\n      private matcher: SelectorMatcher<DirectiveT>,\n      private directives: Map<Element|Template, DirectiveT[]>,\n      private bindings: Map<BoundAttribute|BoundEvent|TextAttribute, DirectiveT|Element|Template>,\n      private references:\n          Map<Reference, {directive: DirectiveT, node: Element|Template}|Element|Template>) {}\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[], selectorMatcher: SelectorMatcher<DirectiveT>): {\n    directives: Map<Element|Template, DirectiveT[]>,\n    bindings: Map<BoundAttribute|BoundEvent|TextAttribute, DirectiveT|Element|Template>,\n    references: Map<Reference, {directive: DirectiveT, node: Element|Template}|Element|Template>,\n  } {\n    const directives = new Map<Element|Template, DirectiveT[]>();\n    const bindings =\n        new Map<BoundAttribute|BoundEvent|TextAttribute, DirectiveT|Element|Template>();\n    const references =\n        new Map<Reference, {directive: DirectiveT, node: Element | Template}|Element|Template>();\n    const matcher = new DirectiveBinder(selectorMatcher, directives, bindings, references);\n    matcher.ingest(template);\n    return {directives, bindings, references};\n  }\n\n  private ingest(template: Node[]): void { template.forEach(node => node.visit(this)); }\n\n  visitElement(element: Element): void { this.visitElementOrTemplate(element.name, element); }\n\n  visitTemplate(template: Template): void { this.visitElementOrTemplate('ng-template', template); }\n\n  visitElementOrTemplate(elementName: string, 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 = createCssSelector(elementName, getAttrsForDirectiveMatching(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, (_, directive) => directives.push(directive));\n    if (directives.length > 0) {\n      this.directives.set(node, directives);\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, ioType: keyof Pick<DirectiveMeta, 'inputs'|'outputs'>) => {\n          const dir = directives.find(dir => dir[ioType].hasOwnProperty(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  // Unused visitors.\n  visitContent(content: Content): void {}\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}\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 pipesUsed: string[] = [];\n\n  private constructor(\n      private bindings: Map<AST, Reference|Variable>,\n      private symbols: Map<Reference|Variable, Template>, private usedPipes: Set<string>,\n      private nestingLevel: Map<Template, number>, private scope: Scope,\n      private template: Template|null, private level: number) {\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  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 template 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 apply(template: Node[], scope: Scope): {\n    expressions: Map<AST, Reference|Variable>,\n    symbols: Map<Variable|Reference, Template>,\n    nestingLevel: Map<Template, number>,\n    usedPipes: Set<string>,\n  } {\n    const expressions = new Map<AST, Reference|Variable>();\n    const symbols = new Map<Variable|Reference, Template>();\n    const nestingLevel = new Map<Template, number>();\n    const usedPipes = new Set<string>();\n    // The top-level template has nesting level 0.\n    const binder = new TemplateBinder(\n        expressions, symbols, usedPipes, nestingLevel, scope,\n        template instanceof Template ? template : null, 0);\n    binder.ingest(template);\n    return {expressions, symbols, nestingLevel, usedPipes};\n  }\n\n  private ingest(template: Template|Node[]): void {\n    if (template 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      template.variables.forEach(this.visitNode);\n      template.children.forEach(this.visitNode);\n\n      // Set the nesting level.\n      this.nestingLevel.set(template, this.level);\n    } else {\n      // Visit each node from the top-level template.\n      template.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  }\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\n    // References are also evaluated in the outer context.\n    template.references.forEach(this.visitNode);\n\n    // Next, recurse into the template using its scope, and bumping the nesting level up by one.\n    const childScope = this.scope.getChildScope(template);\n    const binder = new TemplateBinder(\n        this.bindings, this.symbols, this.usedPipes, this.nestingLevel, childScope, template,\n        this.level + 1);\n    binder.ingest(template);\n  }\n\n  visitVariable(variable: Variable) {\n    // Register the `Variable` as a symbol in the current `Template`.\n    if (this.template !== null) {\n      this.symbols.set(variable, this.template);\n    }\n  }\n\n  visitReference(reference: Reference) {\n    // Register the `Reference` as a symbol in the current `Template`.\n    if (this.template !== null) {\n      this.symbols.set(reference, this.template);\n    }\n  }\n\n  // Unused template visitors\n\n  visitText(text: Text) {}\n  visitContent(content: Content) {}\n  visitTextAttribute(attribute: TextAttribute) {}\n  visitIcu(icu: Icu): void {}\n\n  // The remaining visitors are concerned with processing AST expressions within template bindings\n\n  visitBoundAttribute(attribute: BoundAttribute) { attribute.value.visit(this); }\n\n  visitBoundEvent(event: BoundEvent) { event.handler.visit(this); }\n\n  visitBoundText(text: BoundText) { text.value.visit(this); }\n  visitPipe(ast: BindingPipe, context: any): any {\n    this.usedPipes.add(ast.name);\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  visitPropertyRead(ast: PropertyRead, context: any): any {\n    this.maybeMap(context, ast, ast.name);\n    return super.visitPropertyRead(ast, context);\n  }\n\n  visitSafePropertyRead(ast: SafePropertyRead, context: any): any {\n    this.maybeMap(context, ast, ast.name);\n    return super.visitSafePropertyRead(ast, context);\n  }\n\n  visitPropertyWrite(ast: PropertyWrite, context: any): any {\n    this.maybeMap(context, ast, ast.name);\n    return super.visitPropertyWrite(ast, context);\n  }\n\n  visitMethodCall(ast: MethodCall, context: any): any {\n    this.maybeMap(context, ast, ast.name);\n    return super.visitMethodCall(ast, context);\n  }\n\n  visitSafeMethodCall(ast: SafeMethodCall, context: any): any {\n    this.maybeMap(context, ast, ast.name);\n    return super.visitSafeMethodCall(ast, context);\n  }\n\n  private maybeMap(\n      scope: Scope, ast: PropertyRead|SafePropertyRead|PropertyWrite|MethodCall|SafeMethodCall,\n      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    let target = this.scope.lookup(name);\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  constructor(\n      readonly target: Target, private directives: Map<Element|Template, DirectiveT[]>,\n      private bindings: Map<BoundAttribute|BoundEvent|TextAttribute, DirectiveT|Element|Template>,\n      private references:\n          Map<BoundAttribute|BoundEvent|Reference|TextAttribute,\n              {directive: DirectiveT, node: Element|Template}|Element|Template>,\n      private exprTargets: Map<AST, Reference|Variable>,\n      private symbols: Map<Reference|Variable, Template>,\n      private nestingLevel: Map<Template, number>, private usedPipes: Set<string>) {}\n\n  getDirectivesOfNode(node: Element|Template): DirectiveT[]|null {\n    return this.directives.get(node) || null;\n  }\n\n  getReferenceTarget(ref: Reference): {directive: DirectiveT, node: Element|Template}|Element\n      |Template|null {\n    return this.references.get(ref) || null;\n  }\n\n  getConsumerOfBinding(binding: BoundAttribute|BoundEvent|TextAttribute): DirectiveT|Element\n      |Template|null {\n    return this.bindings.get(binding) || null;\n  }\n\n  getExpressionTarget(expr: AST): Reference|Variable|null {\n    return this.exprTargets.get(expr) || null;\n  }\n\n  getTemplateOfSymbol(symbol: Reference|Variable): Template|null {\n    return this.symbols.get(symbol) || null;\n  }\n\n  getNestingLevel(template: Template): number { return this.nestingLevel.get(template) || 0; }\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  getUsedPipes(): string[] { return Array.from(this.usedPipes); }\n}\n"]}