@angular/core
Version:
Angular - the core framework
473 lines • 45 kB
JavaScript
/**
* @fileoverview added by tsickle
* Generated from: packages/core/src/render3/util/discovery_utils.ts
* @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* @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 { Injector } from '../../di/injector';
import { assertLView } from '../assert';
import { discoverLocalRefs, getComponentAtNodeIndex, getDirectivesAtNodeIndex, getLContext } from '../context_discovery';
import { NodeInjector } from '../di';
import { buildDebugNode } from '../instructions/lview_debug';
import { isLView } from '../interfaces/type_checks';
import { CLEANUP, CONTEXT, FLAGS, HEADER_OFFSET, HOST, T_HOST, TVIEW } from '../interfaces/view';
import { stringifyForError } from './misc_utils';
import { getLViewParent, getRootContext } from './view_traversal_utils';
import { getTNode, unwrapRNode } from './view_utils';
/**
* Retrieves the component instance associated with a given DOM element.
*
* \@usageNotes
* Given the following DOM structure:
* ```html
* <my-app>
* <div>
* <child-comp></child-comp>
* </div>
* </my-app>
* ```
* Calling `getComponent` on `<child-comp>` will return the instance of `ChildComponent`
* associated with this DOM element.
*
* Calling the function on `<my-app>` will return the `MyApp` instance.
*
*
* \@publicApi
* \@globalApi ng
* @template T
* @param {?} element DOM element from which the component should be retrieved.
* @return {?} Component instance associated with the element or `null` if there
* is no component associated with it.
*
*/
export function getComponent(element) {
assertDomElement(element);
/** @type {?} */
const context = loadLContext(element, false);
if (context === null)
return null;
if (context.component === undefined) {
context.component = getComponentAtNodeIndex(context.nodeIndex, context.lView);
}
return (/** @type {?} */ (context.component));
}
/**
* If inside an embedded view (e.g. `*ngIf` or `*ngFor`), retrieves the context of the embedded
* view that the element is part of. Otherwise retrieves the instance of the component whose view
* owns the element (in this case, the result is the same as calling `getOwningComponent`).
*
* \@publicApi
* \@globalApi ng
* @template T
* @param {?} element Element for which to get the surrounding component instance.
* @return {?} Instance of the component that is around the element or null if the element isn't
* inside any component.
*
*/
export function getContext(element) {
assertDomElement(element);
/** @type {?} */
const context = loadLContext(element, false);
return context === null ? null : (/** @type {?} */ (context.lView[CONTEXT]));
}
/**
* Retrieves the component instance whose view contains the DOM element.
*
* For example, if `<child-comp>` is used in the template of `<app-comp>`
* (i.e. a `ViewChild` of `<app-comp>`), calling `getOwningComponent` on `<child-comp>`
* would return `<app-comp>`.
*
* \@publicApi
* \@globalApi ng
* @template T
* @param {?} elementOrDir DOM element, component or directive instance
* for which to retrieve the root components.
* @return {?} Component instance whose view owns the DOM element or null if the element is not
* part of a component view.
*
*/
export function getOwningComponent(elementOrDir) {
/** @type {?} */
const context = loadLContext(elementOrDir, false);
if (context === null)
return null;
/** @type {?} */
let lView = context.lView;
/** @type {?} */
let parent;
ngDevMode && assertLView(lView);
while (lView[HOST] === null && (parent = (/** @type {?} */ (getLViewParent(lView))))) {
// As long as lView[HOST] is null we know we are part of sub-template such as `*ngIf`
lView = parent;
}
return lView[FLAGS] & 512 /* IsRoot */ ? null : (/** @type {?} */ (lView[CONTEXT]));
}
/**
* Retrieves all root components associated with a DOM element, directive or component instance.
* Root components are those which have been bootstrapped by Angular.
*
* \@publicApi
* \@globalApi ng
* @param {?} elementOrDir DOM element, component or directive instance
* for which to retrieve the root components.
* @return {?} Root components associated with the target object.
*
*/
export function getRootComponents(elementOrDir) {
return [...getRootContext(elementOrDir).components];
}
/**
* Retrieves an `Injector` associated with an element, component or directive instance.
*
* \@publicApi
* \@globalApi ng
* @param {?} elementOrDir DOM element, component or directive instance for which to
* retrieve the injector.
* @return {?} Injector associated with the element, component or directive instance.
*
*/
export function getInjector(elementOrDir) {
/** @type {?} */
const context = loadLContext(elementOrDir, false);
if (context === null)
return Injector.NULL;
/** @type {?} */
const tNode = (/** @type {?} */ (context.lView[TVIEW].data[context.nodeIndex]));
return new NodeInjector(tNode, context.lView);
}
/**
* Retrieve a set of injection tokens at a given DOM node.
*
* @param {?} element Element for which the injection tokens should be retrieved.
* @return {?}
*/
export function getInjectionTokens(element) {
/** @type {?} */
const context = loadLContext(element, false);
if (context === null)
return [];
/** @type {?} */
const lView = context.lView;
/** @type {?} */
const tView = lView[TVIEW];
/** @type {?} */
const tNode = (/** @type {?} */ (tView.data[context.nodeIndex]));
/** @type {?} */
const providerTokens = [];
/** @type {?} */
const startIndex = tNode.providerIndexes & 65535 /* ProvidersStartIndexMask */;
/** @type {?} */
const endIndex = tNode.directiveEnd;
for (let i = startIndex; i < endIndex; i++) {
/** @type {?} */
let value = tView.data[i];
if (isDirectiveDefHack(value)) {
// The fact that we sometimes store Type and sometimes DirectiveDef in this location is a
// design flaw. We should always store same type so that we can be monomorphic. The issue
// is that for Components/Directives we store the def instead the type. The correct behavior
// is that we should always be storing injectable type in this location.
value = value.type;
}
providerTokens.push(value);
}
return providerTokens;
}
/**
* Retrieves directive instances associated with a given DOM element. Does not include
* component instances.
*
* \@usageNotes
* Given the following DOM structure:
* ```
* <my-app>
* <button my-button></button>
* <my-comp></my-comp>
* </my-app>
* ```
* Calling `getDirectives` on `<button>` will return an array with an instance of the `MyButton`
* directive that is associated with the DOM element.
*
* Calling `getDirectives` on `<my-comp>` will return an empty array.
*
* \@publicApi
* \@globalApi ng
* @param {?} element DOM element for which to get the directives.
* @return {?} Array of directives associated with the element.
*
*/
export function getDirectives(element) {
/** @type {?} */
const context = (/** @type {?} */ (loadLContext(element)));
if (context.directives === undefined) {
context.directives = getDirectivesAtNodeIndex(context.nodeIndex, context.lView, false);
}
// The `directives` in this case are a named array called `LComponentView`. Clone the
// result so we don't expose an internal data structure in the user's console.
return context.directives === null ? [] : [...context.directives];
}
/**
* @param {?} target
* @param {?=} throwOnNotFound
* @return {?}
*/
export function loadLContext(target, throwOnNotFound = true) {
/** @type {?} */
const context = getLContext(target);
if (!context && throwOnNotFound) {
throw new Error(ngDevMode ? `Unable to find context associated with ${stringifyForError(target)}` :
'Invalid ng target');
}
return context;
}
/**
* Retrieve map of local references.
*
* The references are retrieved as a map of local reference name to element or directive instance.
*
* @param {?} target DOM element, component or directive instance for which to retrieve
* the local references.
* @return {?}
*/
export function getLocalRefs(target) {
/** @type {?} */
const context = loadLContext(target, false);
if (context === null)
return {};
if (context.localRefs === undefined) {
context.localRefs = discoverLocalRefs(context.lView, context.nodeIndex);
}
return context.localRefs || {};
}
/**
* Retrieves the host element of a component or directive instance.
* The host element is the DOM element that matched the selector of the directive.
*
* \@publicApi
* \@globalApi ng
* @param {?} componentOrDirective Component or directive instance for which the host
* element should be retrieved.
* @return {?} Host element of the target.
*
*/
export function getHostElement(componentOrDirective) {
return (/** @type {?} */ ((/** @type {?} */ ((/** @type {?} */ (getLContext(componentOrDirective))).native))));
}
/**
* Retrieves the rendered text for a given component.
*
* This function retrieves the host element of a component and
* and then returns the `textContent` for that element. This implies
* that the text returned will include re-projected content of
* the component as well.
*
* @param {?} component The component to return the content text for.
* @return {?}
*/
export function getRenderedText(component) {
/** @type {?} */
const hostElement = getHostElement(component);
return hostElement.textContent || '';
}
/**
* @param {?} node
* @return {?}
*/
export function loadLContextFromNode(node) {
if (!(node instanceof Node))
throw new Error('Expecting instance of DOM Element');
return (/** @type {?} */ (loadLContext(node)));
}
/**
* Event listener configuration returned from `getListeners`.
* \@publicApi
* @record
*/
export function Listener() { }
if (false) {
/**
* Name of the event listener.
* @type {?}
*/
Listener.prototype.name;
/**
* Element that the listener is bound to.
* @type {?}
*/
Listener.prototype.element;
/**
* Callback that is invoked when the event is triggered.
* @type {?}
*/
Listener.prototype.callback;
/**
* Whether the listener is using event capturing.
* @type {?}
*/
Listener.prototype.useCapture;
/**
* Type of the listener (e.g. a native DOM event or a custom \@Output).
* @type {?}
*/
Listener.prototype.type;
}
/**
* Retrieves a list of event listeners associated with a DOM element. The list does include host
* listeners, but it does not include event listeners defined outside of the Angular context
* (e.g. through `addEventListener`).
*
* \@usageNotes
* Given the following DOM structure:
* ```
* <my-app>
* <div (click)="doSomething()"></div>
* </my-app>
*
* ```
* Calling `getListeners` on `<div>` will return an object that looks as follows:
* ```
* {
* name: 'click',
* element: <div>,
* callback: () => doSomething(),
* useCapture: false
* }
* ```
*
* \@publicApi
* \@globalApi ng
* @param {?} element Element for which the DOM listeners should be retrieved.
* @return {?} Array of event listeners on the DOM element.
*
*/
export function getListeners(element) {
assertDomElement(element);
/** @type {?} */
const lContext = loadLContext(element, false);
if (lContext === null)
return [];
/** @type {?} */
const lView = lContext.lView;
/** @type {?} */
const tView = lView[TVIEW];
/** @type {?} */
const lCleanup = lView[CLEANUP];
/** @type {?} */
const tCleanup = tView.cleanup;
/** @type {?} */
const listeners = [];
if (tCleanup && lCleanup) {
for (let i = 0; i < tCleanup.length;) {
/** @type {?} */
const firstParam = tCleanup[i++];
/** @type {?} */
const secondParam = tCleanup[i++];
if (typeof firstParam === 'string') {
/** @type {?} */
const name = firstParam;
/** @type {?} */
const listenerElement = (/** @type {?} */ ((/** @type {?} */ (unwrapRNode(lView[secondParam])))));
/** @type {?} */
const callback = lCleanup[tCleanup[i++]];
/** @type {?} */
const useCaptureOrIndx = tCleanup[i++];
// if useCaptureOrIndx is boolean then report it as is.
// if useCaptureOrIndx is positive number then it in unsubscribe method
// if useCaptureOrIndx is negative number then it is a Subscription
/** @type {?} */
const type = (typeof useCaptureOrIndx === 'boolean' || useCaptureOrIndx >= 0) ? 'dom' : 'output';
/** @type {?} */
const useCapture = typeof useCaptureOrIndx === 'boolean' ? useCaptureOrIndx : false;
if (element == listenerElement) {
listeners.push({ element, name, callback, useCapture, type });
}
}
}
}
listeners.sort(sortListeners);
return listeners;
}
/**
* @param {?} a
* @param {?} b
* @return {?}
*/
function sortListeners(a, b) {
if (a.name == b.name)
return 0;
return a.name < b.name ? -1 : 1;
}
/**
* This function should not exist because it is megamorphic and only mostly correct.
*
* See call site for more info.
* @param {?} obj
* @return {?}
*/
function isDirectiveDefHack(obj) {
return obj.type !== undefined && obj.template !== undefined && obj.declaredInputs !== undefined;
}
/**
* Returns the attached `DebugNode` instance for an element in the DOM.
*
* @param {?} element DOM element which is owned by an existing component's view.
* @return {?}
*/
export function getDebugNode(element) {
/** @type {?} */
let debugNode = null;
/** @type {?} */
const lContext = loadLContextFromNode(element);
/** @type {?} */
const lView = lContext.lView;
/** @type {?} */
const nodeIndex = lContext.nodeIndex;
if (nodeIndex !== -1) {
/** @type {?} */
const valueInLView = lView[nodeIndex];
// this means that value in the lView is a component with its own
// data. In this situation the TNode is not accessed at the same spot.
/** @type {?} */
const tNode = isLView(valueInLView) ? ((/** @type {?} */ (valueInLView[T_HOST]))) :
getTNode(lView[TVIEW], nodeIndex - HEADER_OFFSET);
debugNode = buildDebugNode(tNode, lView, nodeIndex);
}
return debugNode;
}
/**
* Retrieve the component `LView` from component/element.
*
* NOTE: `LView` is a private and should not be leaked outside.
* Don't export this method to `ng.*` on window.
*
* @param {?} target DOM element or component instance for which to retrieve the LView.
* @return {?}
*/
export function getComponentLView(target) {
/** @type {?} */
const lContext = loadLContext(target);
/** @type {?} */
const nodeIndx = lContext.nodeIndex;
/** @type {?} */
const lView = lContext.lView;
/** @type {?} */
const componentLView = lView[nodeIndx];
ngDevMode && assertLView(componentLView);
return componentLView;
}
/**
* Asserts that a value is a DOM Element.
* @param {?} value
* @return {?}
*/
function assertDomElement(value) {
if (typeof Element !== 'undefined' && !(value instanceof Element)) {
throw new Error('Expecting instance of DOM Element');
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"discovery_utils.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/util/discovery_utils.ts"],"names":[],"mappings":";;;;;;;;;;;;AAQA,OAAO,EAAC,QAAQ,EAAC,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EAAC,WAAW,EAAC,MAAM,WAAW,CAAC;AACtC,OAAO,EAAC,iBAAiB,EAAE,uBAAuB,EAAE,wBAAwB,EAAE,WAAW,EAAC,MAAM,sBAAsB,CAAC;AACvH,OAAO,EAAC,YAAY,EAAC,MAAM,OAAO,CAAC;AACnC,OAAO,EAAC,cAAc,EAAY,MAAM,6BAA6B,CAAC;AAItE,OAAO,EAAC,OAAO,EAAC,MAAM,2BAA2B,CAAC;AAClD,OAAO,EAAC,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,EAAqB,MAAM,EAAE,KAAK,EAAC,MAAM,oBAAoB,CAAC;AAElH,OAAO,EAAC,iBAAiB,EAAC,MAAM,cAAc,CAAC;AAC/C,OAAO,EAAC,cAAc,EAAE,cAAc,EAAC,MAAM,wBAAwB,CAAC;AACtE,OAAO,EAAC,QAAQ,EAAE,WAAW,EAAC,MAAM,cAAc,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6BnD,MAAM,UAAU,YAAY,CAAI,OAAgB;IAC9C,gBAAgB,CAAC,OAAO,CAAC,CAAC;;UACpB,OAAO,GAAG,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC;IAC5C,IAAI,OAAO,KAAK,IAAI;QAAE,OAAO,IAAI,CAAC;IAElC,IAAI,OAAO,CAAC,SAAS,KAAK,SAAS,EAAE;QACnC,OAAO,CAAC,SAAS,GAAG,uBAAuB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;KAC/E;IAED,OAAO,mBAAA,OAAO,CAAC,SAAS,EAAK,CAAC;AAChC,CAAC;;;;;;;;;;;;;;AAeD,MAAM,UAAU,UAAU,CAAI,OAAgB;IAC5C,gBAAgB,CAAC,OAAO,CAAC,CAAC;;UACpB,OAAO,GAAG,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC;IAC5C,OAAO,OAAO,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,mBAAA,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,EAAK,CAAC;AAC/D,CAAC;;;;;;;;;;;;;;;;;AAiBD,MAAM,UAAU,kBAAkB,CAAI,YAAwB;;UACtD,OAAO,GAAG,YAAY,CAAC,YAAY,EAAE,KAAK,CAAC;IACjD,IAAI,OAAO,KAAK,IAAI;QAAE,OAAO,IAAI,CAAC;;QAE9B,KAAK,GAAG,OAAO,CAAC,KAAK;;QACrB,MAAkB;IACtB,SAAS,IAAI,WAAW,CAAC,KAAK,CAAC,CAAC;IAChC,OAAO,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,GAAG,mBAAA,cAAc,CAAC,KAAK,CAAC,EAAC,CAAC,EAAE;QAChE,qFAAqF;QACrF,KAAK,GAAG,MAAM,CAAC;KAChB;IACD,OAAO,KAAK,CAAC,KAAK,CAAC,mBAAoB,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,mBAAA,KAAK,CAAC,OAAO,CAAC,EAAK,CAAC;AACvE,CAAC;;;;;;;;;;;;AAaD,MAAM,UAAU,iBAAiB,CAAC,YAAwB;IACxD,OAAO,CAAC,GAAG,cAAc,CAAC,YAAY,CAAC,CAAC,UAAU,CAAC,CAAC;AACtD,CAAC;;;;;;;;;;;AAYD,MAAM,UAAU,WAAW,CAAC,YAAwB;;UAC5C,OAAO,GAAG,YAAY,CAAC,YAAY,EAAE,KAAK,CAAC;IACjD,IAAI,OAAO,KAAK,IAAI;QAAE,OAAO,QAAQ,CAAC,IAAI,CAAC;;UAErC,KAAK,GAAG,mBAAA,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAgB;IAC1E,OAAO,IAAI,YAAY,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;AAChD,CAAC;;;;;;;AAOD,MAAM,UAAU,kBAAkB,CAAC,OAAgB;;UAC3C,OAAO,GAAG,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC;IAC5C,IAAI,OAAO,KAAK,IAAI;QAAE,OAAO,EAAE,CAAC;;UAC1B,KAAK,GAAG,OAAO,CAAC,KAAK;;UACrB,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC;;UACpB,KAAK,GAAG,mBAAA,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAS;;UAC9C,cAAc,GAAU,EAAE;;UAC1B,UAAU,GAAG,KAAK,CAAC,eAAe,sCAA+C;;UACjF,QAAQ,GAAG,KAAK,CAAC,YAAY;IACnC,KAAK,IAAI,CAAC,GAAG,UAAU,EAAE,CAAC,GAAG,QAAQ,EAAE,CAAC,EAAE,EAAE;;YACtC,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;QACzB,IAAI,kBAAkB,CAAC,KAAK,CAAC,EAAE;YAC7B,yFAAyF;YACzF,0FAA0F;YAC1F,4FAA4F;YAC5F,wEAAwE;YACxE,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC;SACpB;QACD,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;KAC5B;IACD,OAAO,cAAc,CAAC;AACxB,CAAC;;;;;;;;;;;;;;;;;;;;;;;;AAyBD,MAAM,UAAU,aAAa,CAAC,OAAgB;;UACtC,OAAO,GAAG,mBAAA,YAAY,CAAC,OAAO,CAAC,EAAC;IAEtC,IAAI,OAAO,CAAC,UAAU,KAAK,SAAS,EAAE;QACpC,OAAO,CAAC,UAAU,GAAG,wBAAwB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;KACxF;IAED,qFAAqF;IACrF,8EAA8E;IAC9E,OAAO,OAAO,CAAC,UAAU,KAAK,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;AACpE,CAAC;;;;;;AAQD,MAAM,UAAU,YAAY,CAAC,MAAU,EAAE,kBAA2B,IAAI;;UAChE,OAAO,GAAG,WAAW,CAAC,MAAM,CAAC;IACnC,IAAI,CAAC,OAAO,IAAI,eAAe,EAAE;QAC/B,MAAM,IAAI,KAAK,CACX,SAAS,CAAC,CAAC,CAAC,0CAA0C,iBAAiB,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACvE,mBAAmB,CAAC,CAAC;KACtC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC;;;;;;;;;;AAUD,MAAM,UAAU,YAAY,CAAC,MAAU;;UAC/B,OAAO,GAAG,YAAY,CAAC,MAAM,EAAE,KAAK,CAAC;IAC3C,IAAI,OAAO,KAAK,IAAI;QAAE,OAAO,EAAE,CAAC;IAEhC,IAAI,OAAO,CAAC,SAAS,KAAK,SAAS,EAAE;QACnC,OAAO,CAAC,SAAS,GAAG,iBAAiB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC;KACzE;IAED,OAAO,OAAO,CAAC,SAAS,IAAI,EAAE,CAAC;AACjC,CAAC;;;;;;;;;;;;AAaD,MAAM,UAAU,cAAc,CAAC,oBAAwB;IACrD,OAAO,mBAAA,mBAAA,mBAAA,WAAW,CAAC,oBAAoB,CAAC,EAAC,CAAC,MAAM,EAAS,EAAW,CAAC;AACvE,CAAC;;;;;;;;;;;;AAYD,MAAM,UAAU,eAAe,CAAC,SAAc;;UACtC,WAAW,GAAG,cAAc,CAAC,SAAS,CAAC;IAC7C,OAAO,WAAW,CAAC,WAAW,IAAI,EAAE,CAAC;AACvC,CAAC;;;;;AAED,MAAM,UAAU,oBAAoB,CAAC,IAAU;IAC7C,IAAI,CAAC,CAAC,IAAI,YAAY,IAAI,CAAC;QAAE,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;IAClF,OAAO,mBAAA,YAAY,CAAC,IAAI,CAAC,EAAC,CAAC;AAC7B,CAAC;;;;;;AAMD,8BAaC;;;;;;IAXC,wBAAa;;;;;IAEb,2BAAiB;;;;;IAEjB,4BAA8B;;;;;IAE9B,8BAAoB;;;;;IAIpB,wBAAqB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAiCvB,MAAM,UAAU,YAAY,CAAC,OAAgB;IAC3C,gBAAgB,CAAC,OAAO,CAAC,CAAC;;UACpB,QAAQ,GAAG,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC;IAC7C,IAAI,QAAQ,KAAK,IAAI;QAAE,OAAO,EAAE,CAAC;;UAE3B,KAAK,GAAG,QAAQ,CAAC,KAAK;;UACtB,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC;;UACpB,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC;;UACzB,QAAQ,GAAG,KAAK,CAAC,OAAO;;UACxB,SAAS,GAAe,EAAE;IAChC,IAAI,QAAQ,IAAI,QAAQ,EAAE;QACxB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,CAAC,MAAM,GAAG;;kBAC9B,UAAU,GAAG,QAAQ,CAAC,CAAC,EAAE,CAAC;;kBAC1B,WAAW,GAAG,QAAQ,CAAC,CAAC,EAAE,CAAC;YACjC,IAAI,OAAO,UAAU,KAAK,QAAQ,EAAE;;sBAC5B,IAAI,GAAW,UAAU;;sBACzB,eAAe,GAAG,mBAAA,mBAAA,WAAW,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,EAAO,EAAW;;sBACnE,QAAQ,GAAwB,QAAQ,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,CAAC;;sBACvD,gBAAgB,GAAG,QAAQ,CAAC,CAAC,EAAE,CAAC;;;;;sBAIhC,IAAI,GACN,CAAC,OAAO,gBAAgB,KAAK,SAAS,IAAI,gBAAgB,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ;;sBACjF,UAAU,GAAG,OAAO,gBAAgB,KAAK,SAAS,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK;gBACnF,IAAI,OAAO,IAAI,eAAe,EAAE;oBAC9B,SAAS,CAAC,IAAI,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAC,CAAC,CAAC;iBAC7D;aACF;SACF;KACF;IACD,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAC9B,OAAO,SAAS,CAAC;AACnB,CAAC;;;;;;AAED,SAAS,aAAa,CAAC,CAAW,EAAE,CAAW;IAC7C,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI;QAAE,OAAO,CAAC,CAAC;IAC/B,OAAO,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AAClC,CAAC;;;;;;;;AAOD,SAAS,kBAAkB,CAAC,GAAQ;IAClC,OAAO,GAAG,CAAC,IAAI,KAAK,SAAS,IAAI,GAAG,CAAC,QAAQ,KAAK,SAAS,IAAI,GAAG,CAAC,cAAc,KAAK,SAAS,CAAC;AAClG,CAAC;;;;;;;AAOD,MAAM,UAAU,YAAY,CAAC,OAAgB;;QACvC,SAAS,GAAmB,IAAI;;UAE9B,QAAQ,GAAG,oBAAoB,CAAC,OAAO,CAAC;;UACxC,KAAK,GAAG,QAAQ,CAAC,KAAK;;UACtB,SAAS,GAAG,QAAQ,CAAC,SAAS;IACpC,IAAI,SAAS,KAAK,CAAC,CAAC,EAAE;;cACd,YAAY,GAAG,KAAK,CAAC,SAAS,CAAC;;;;cAG/B,KAAK,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,mBAAA,YAAY,CAAC,MAAM,CAAC,EAAS,CAAC,CAAC,CAAC;YACjC,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,SAAS,GAAG,aAAa,CAAC;QACvF,SAAS,GAAG,cAAc,CAAC,KAAK,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC;KACrD;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;;;;;;;;;;AAUD,MAAM,UAAU,iBAAiB,CAAC,MAAW;;UACrC,QAAQ,GAAG,YAAY,CAAC,MAAM,CAAC;;UAC/B,QAAQ,GAAG,QAAQ,CAAC,SAAS;;UAC7B,KAAK,GAAG,QAAQ,CAAC,KAAK;;UACtB,cAAc,GAAG,KAAK,CAAC,QAAQ,CAAC;IACtC,SAAS,IAAI,WAAW,CAAC,cAAc,CAAC,CAAC;IACzC,OAAO,cAAc,CAAC;AACxB,CAAC;;;;;;AAGD,SAAS,gBAAgB,CAAC,KAAU;IAClC,IAAI,OAAO,OAAO,KAAK,WAAW,IAAI,CAAC,CAAC,KAAK,YAAY,OAAO,CAAC,EAAE;QACjE,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;KACtD;AACH,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {Injector} from '../../di/injector';\nimport {assertLView} from '../assert';\nimport {discoverLocalRefs, getComponentAtNodeIndex, getDirectivesAtNodeIndex, getLContext} from '../context_discovery';\nimport {NodeInjector} from '../di';\nimport {buildDebugNode, DebugNode} from '../instructions/lview_debug';\nimport {LContext} from '../interfaces/context';\nimport {DirectiveDef} from '../interfaces/definition';\nimport {TElementNode, TNode, TNodeProviderIndexes} from '../interfaces/node';\nimport {isLView} from '../interfaces/type_checks';\nimport {CLEANUP, CONTEXT, FLAGS, HEADER_OFFSET, HOST, LView, LViewFlags, T_HOST, TVIEW} from '../interfaces/view';\n\nimport {stringifyForError} from './misc_utils';\nimport {getLViewParent, getRootContext} from './view_traversal_utils';\nimport {getTNode, unwrapRNode} from './view_utils';\n\n\n\n/**\n * Retrieves the component instance associated with a given DOM element.\n *\n * @usageNotes\n * Given the following DOM structure:\n * ```html\n * <my-app>\n *   <div>\n *     <child-comp></child-comp>\n *   </div>\n * </my-app>\n * ```\n * Calling `getComponent` on `<child-comp>` will return the instance of `ChildComponent`\n * associated with this DOM element.\n *\n * Calling the function on `<my-app>` will return the `MyApp` instance.\n *\n *\n * @param element DOM element from which the component should be retrieved.\n * @returns Component instance associated with the element or `null` if there\n *    is no component associated with it.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getComponent<T>(element: Element): T|null {\n  assertDomElement(element);\n  const context = loadLContext(element, false);\n  if (context === null) return null;\n\n  if (context.component === undefined) {\n    context.component = getComponentAtNodeIndex(context.nodeIndex, context.lView);\n  }\n\n  return context.component as T;\n}\n\n\n/**\n * If inside an embedded view (e.g. `*ngIf` or `*ngFor`), retrieves the context of the embedded\n * view that the element is part of. Otherwise retrieves the instance of the component whose view\n * owns the element (in this case, the result is the same as calling `getOwningComponent`).\n *\n * @param element Element for which to get the surrounding component instance.\n * @returns Instance of the component that is around the element or null if the element isn't\n *    inside any component.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getContext<T>(element: Element): T|null {\n  assertDomElement(element);\n  const context = loadLContext(element, false);\n  return context === null ? null : context.lView[CONTEXT] as T;\n}\n\n/**\n * Retrieves the component instance whose view contains the DOM element.\n *\n * For example, if `<child-comp>` is used in the template of `<app-comp>`\n * (i.e. a `ViewChild` of `<app-comp>`), calling `getOwningComponent` on `<child-comp>`\n * would return `<app-comp>`.\n *\n * @param elementOrDir DOM element, component or directive instance\n *    for which to retrieve the root components.\n * @returns Component instance whose view owns the DOM element or null if the element is not\n *    part of a component view.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getOwningComponent<T>(elementOrDir: Element|{}): T|null {\n  const context = loadLContext(elementOrDir, false);\n  if (context === null) return null;\n\n  let lView = context.lView;\n  let parent: LView|null;\n  ngDevMode && assertLView(lView);\n  while (lView[HOST] === null && (parent = getLViewParent(lView)!)) {\n    // As long as lView[HOST] is null we know we are part of sub-template such as `*ngIf`\n    lView = parent;\n  }\n  return lView[FLAGS] & LViewFlags.IsRoot ? null : lView[CONTEXT] as T;\n}\n\n/**\n * Retrieves all root components associated with a DOM element, directive or component instance.\n * Root components are those which have been bootstrapped by Angular.\n *\n * @param elementOrDir DOM element, component or directive instance\n *    for which to retrieve the root components.\n * @returns Root components associated with the target object.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getRootComponents(elementOrDir: Element|{}): {}[] {\n  return [...getRootContext(elementOrDir).components];\n}\n\n/**\n * Retrieves an `Injector` associated with an element, component or directive instance.\n *\n * @param elementOrDir DOM element, component or directive instance for which to\n *    retrieve the injector.\n * @returns Injector associated with the element, component or directive instance.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getInjector(elementOrDir: Element|{}): Injector {\n  const context = loadLContext(elementOrDir, false);\n  if (context === null) return Injector.NULL;\n\n  const tNode = context.lView[TVIEW].data[context.nodeIndex] as TElementNode;\n  return new NodeInjector(tNode, context.lView);\n}\n\n/**\n * Retrieve a set of injection tokens at a given DOM node.\n *\n * @param element Element for which the injection tokens should be retrieved.\n */\nexport function getInjectionTokens(element: Element): any[] {\n  const context = loadLContext(element, false);\n  if (context === null) return [];\n  const lView = context.lView;\n  const tView = lView[TVIEW];\n  const tNode = tView.data[context.nodeIndex] as TNode;\n  const providerTokens: any[] = [];\n  const startIndex = tNode.providerIndexes & TNodeProviderIndexes.ProvidersStartIndexMask;\n  const endIndex = tNode.directiveEnd;\n  for (let i = startIndex; i < endIndex; i++) {\n    let value = tView.data[i];\n    if (isDirectiveDefHack(value)) {\n      // The fact that we sometimes store Type and sometimes DirectiveDef in this location is a\n      // design flaw.  We should always store same type so that we can be monomorphic. The issue\n      // is that for Components/Directives we store the def instead the type. The correct behavior\n      // is that we should always be storing injectable type in this location.\n      value = value.type;\n    }\n    providerTokens.push(value);\n  }\n  return providerTokens;\n}\n\n/**\n * Retrieves directive instances associated with a given DOM element. Does not include\n * component instances.\n *\n * @usageNotes\n * Given the following DOM structure:\n * ```\n * <my-app>\n *   <button my-button></button>\n *   <my-comp></my-comp>\n * </my-app>\n * ```\n * Calling `getDirectives` on `<button>` will return an array with an instance of the `MyButton`\n * directive that is associated with the DOM element.\n *\n * Calling `getDirectives` on `<my-comp>` will return an empty array.\n *\n * @param element DOM element for which to get the directives.\n * @returns Array of directives associated with the element.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getDirectives(element: Element): {}[] {\n  const context = loadLContext(element)!;\n\n  if (context.directives === undefined) {\n    context.directives = getDirectivesAtNodeIndex(context.nodeIndex, context.lView, false);\n  }\n\n  // The `directives` in this case are a named array called `LComponentView`. Clone the\n  // result so we don't expose an internal data structure in the user's console.\n  return context.directives === null ? [] : [...context.directives];\n}\n\n/**\n * Returns LContext associated with a target passed as an argument.\n * Throws if a given target doesn't have associated LContext.\n */\nexport function loadLContext(target: {}): LContext;\nexport function loadLContext(target: {}, throwOnNotFound: false): LContext|null;\nexport function loadLContext(target: {}, throwOnNotFound: boolean = true): LContext|null {\n  const context = getLContext(target);\n  if (!context && throwOnNotFound) {\n    throw new Error(\n        ngDevMode ? `Unable to find context associated with ${stringifyForError(target)}` :\n                    'Invalid ng target');\n  }\n  return context;\n}\n\n/**\n * Retrieve map of local references.\n *\n * The references are retrieved as a map of local reference name to element or directive instance.\n *\n * @param target DOM element, component or directive instance for which to retrieve\n *    the local references.\n */\nexport function getLocalRefs(target: {}): {[key: string]: any} {\n  const context = loadLContext(target, false);\n  if (context === null) return {};\n\n  if (context.localRefs === undefined) {\n    context.localRefs = discoverLocalRefs(context.lView, context.nodeIndex);\n  }\n\n  return context.localRefs || {};\n}\n\n/**\n * Retrieves the host element of a component or directive instance.\n * The host element is the DOM element that matched the selector of the directive.\n *\n * @param componentOrDirective Component or directive instance for which the host\n *     element should be retrieved.\n * @returns Host element of the target.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getHostElement(componentOrDirective: {}): Element {\n  return getLContext(componentOrDirective)!.native as never as Element;\n}\n\n/**\n * Retrieves the rendered text for a given component.\n *\n * This function retrieves the host element of a component and\n * and then returns the `textContent` for that element. This implies\n * that the text returned will include re-projected content of\n * the component as well.\n *\n * @param component The component to return the content text for.\n */\nexport function getRenderedText(component: any): string {\n  const hostElement = getHostElement(component);\n  return hostElement.textContent || '';\n}\n\nexport function loadLContextFromNode(node: Node): LContext {\n  if (!(node instanceof Node)) throw new Error('Expecting instance of DOM Element');\n  return loadLContext(node)!;\n}\n\n/**\n * Event listener configuration returned from `getListeners`.\n * @publicApi\n */\nexport interface Listener {\n  /** Name of the event listener. */\n  name: string;\n  /** Element that the listener is bound to. */\n  element: Element;\n  /** Callback that is invoked when the event is triggered. */\n  callback: (value: any) => any;\n  /** Whether the listener is using event capturing. */\n  useCapture: boolean;\n  /**\n   * Type of the listener (e.g. a native DOM event or a custom @Output).\n   */\n  type: 'dom'|'output';\n}\n\n\n/**\n * Retrieves a list of event listeners associated with a DOM element. The list does include host\n * listeners, but it does not include event listeners defined outside of the Angular context\n * (e.g. through `addEventListener`).\n *\n * @usageNotes\n * Given the following DOM structure:\n * ```\n * <my-app>\n *   <div (click)=\"doSomething()\"></div>\n * </my-app>\n *\n * ```\n * Calling `getListeners` on `<div>` will return an object that looks as follows:\n * ```\n * {\n *   name: 'click',\n *   element: <div>,\n *   callback: () => doSomething(),\n *   useCapture: false\n * }\n * ```\n *\n * @param element Element for which the DOM listeners should be retrieved.\n * @returns Array of event listeners on the DOM element.\n *\n * @publicApi\n * @globalApi ng\n */\nexport function getListeners(element: Element): Listener[] {\n  assertDomElement(element);\n  const lContext = loadLContext(element, false);\n  if (lContext === null) return [];\n\n  const lView = lContext.lView;\n  const tView = lView[TVIEW];\n  const lCleanup = lView[CLEANUP];\n  const tCleanup = tView.cleanup;\n  const listeners: Listener[] = [];\n  if (tCleanup && lCleanup) {\n    for (let i = 0; i < tCleanup.length;) {\n      const firstParam = tCleanup[i++];\n      const secondParam = tCleanup[i++];\n      if (typeof firstParam === 'string') {\n        const name: string = firstParam;\n        const listenerElement = unwrapRNode(lView[secondParam]) as any as Element;\n        const callback: (value: any) => any = lCleanup[tCleanup[i++]];\n        const useCaptureOrIndx = tCleanup[i++];\n        // if useCaptureOrIndx is boolean then report it as is.\n        // if useCaptureOrIndx is positive number then it in unsubscribe method\n        // if useCaptureOrIndx is negative number then it is a Subscription\n        const type =\n            (typeof useCaptureOrIndx === 'boolean' || useCaptureOrIndx >= 0) ? 'dom' : 'output';\n        const useCapture = typeof useCaptureOrIndx === 'boolean' ? useCaptureOrIndx : false;\n        if (element == listenerElement) {\n          listeners.push({element, name, callback, useCapture, type});\n        }\n      }\n    }\n  }\n  listeners.sort(sortListeners);\n  return listeners;\n}\n\nfunction sortListeners(a: Listener, b: Listener) {\n  if (a.name == b.name) return 0;\n  return a.name < b.name ? -1 : 1;\n}\n\n/**\n * This function should not exist because it is megamorphic and only mostly correct.\n *\n * See call site for more info.\n */\nfunction isDirectiveDefHack(obj: any): obj is DirectiveDef<any> {\n  return obj.type !== undefined && obj.template !== undefined && obj.declaredInputs !== undefined;\n}\n\n/**\n * Returns the attached `DebugNode` instance for an element in the DOM.\n *\n * @param element DOM element which is owned by an existing component's view.\n */\nexport function getDebugNode(element: Element): DebugNode|null {\n  let debugNode: DebugNode|null = null;\n\n  const lContext = loadLContextFromNode(element);\n  const lView = lContext.lView;\n  const nodeIndex = lContext.nodeIndex;\n  if (nodeIndex !== -1) {\n    const valueInLView = lView[nodeIndex];\n    // this means that value in the lView is a component with its own\n    // data. In this situation the TNode is not accessed at the same spot.\n    const tNode = isLView(valueInLView) ? (valueInLView[T_HOST] as TNode) :\n                                          getTNode(lView[TVIEW], nodeIndex - HEADER_OFFSET);\n    debugNode = buildDebugNode(tNode, lView, nodeIndex);\n  }\n\n  return debugNode;\n}\n\n/**\n * Retrieve the component `LView` from component/element.\n *\n * NOTE: `LView` is a private and should not be leaked outside.\n *       Don't export this method to `ng.*` on window.\n *\n * @param target DOM element or component instance for which to retrieve the LView.\n */\nexport function getComponentLView(target: any): LView {\n  const lContext = loadLContext(target);\n  const nodeIndx = lContext.nodeIndex;\n  const lView = lContext.lView;\n  const componentLView = lView[nodeIndx];\n  ngDevMode && assertLView(componentLView);\n  return componentLView;\n}\n\n/** Asserts that a value is a DOM Element. */\nfunction assertDomElement(value: any) {\n  if (typeof Element !== 'undefined' && !(value instanceof Element)) {\n    throw new Error('Expecting instance of DOM Element');\n  }\n}\n"]}