2017-12-01 14:23:03 -08:00
|
|
|
/**
|
|
|
|
* @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
|
|
|
|
*/
|
|
|
|
|
2017-12-20 10:47:22 -08:00
|
|
|
// We are temporarily importing the existing viewEngine_from core so we can be sure we are
|
2017-12-13 14:28:36 -08:00
|
|
|
// correctly implementing its interfaces for backwards compatibility.
|
2018-07-27 09:56:35 -07:00
|
|
|
|
2018-08-29 16:34:44 -07:00
|
|
|
import {getInjectableDef, getInjectorDef} from '../di/defs';
|
2018-07-27 09:56:35 -07:00
|
|
|
import {InjectionToken} from '../di/injection_token';
|
2018-10-12 18:49:00 -07:00
|
|
|
import {InjectFlags, Injector, inject, setCurrentInjector} from '../di/injector';
|
2017-12-20 10:47:22 -08:00
|
|
|
import {Type} from '../type';
|
|
|
|
|
2018-09-28 21:26:45 -07:00
|
|
|
import {assertDefined} from './assert';
|
2018-08-29 16:34:44 -07:00
|
|
|
import {getComponentDef, getDirectiveDef, getPipeDef} from './definition';
|
2018-09-21 18:38:13 -07:00
|
|
|
import {NG_ELEMENT_ID} from './fields';
|
2018-10-12 18:49:00 -07:00
|
|
|
import {_getViewData, getPreviousOrParentTNode, resolveDirective, setEnvironment} from './instructions';
|
2018-09-21 12:12:06 -07:00
|
|
|
import {DirectiveDef} from './interfaces/definition';
|
2018-10-08 16:04:46 -07:00
|
|
|
import {InjectorLocationFlags, PARENT_INJECTOR, TNODE,} from './interfaces/injector';
|
2018-09-28 21:26:45 -07:00
|
|
|
import {AttributeMarker, TContainerNode, TElementContainerNode, TElementNode, TNode, TNodeFlags, TNodeType} from './interfaces/node';
|
2018-10-12 18:49:00 -07:00
|
|
|
import {DECLARATION_VIEW, HOST_NODE, INJECTOR, LViewData, TData, TVIEW, TView} from './interfaces/view';
|
2018-09-28 21:26:45 -07:00
|
|
|
import {assertNodeOfPossibleTypes} from './node_assert';
|
2017-12-15 14:59:17 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of slots in each bloom filter (used by DI). The larger this number, the fewer
|
|
|
|
* directives that will share slots, and thus, the fewer false positives when checking for
|
|
|
|
* the existence of a directive.
|
|
|
|
*/
|
2018-03-14 13:29:48 -07:00
|
|
|
const BLOOM_SIZE = 256;
|
2018-07-27 12:44:58 -07:00
|
|
|
const BLOOM_MASK = BLOOM_SIZE - 1;
|
2017-12-15 14:59:17 +01:00
|
|
|
|
|
|
|
/** Counter used to generate unique IDs for directives. */
|
|
|
|
let nextNgElementId = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Registers this directive as present in its node's injector by flipping the directive's
|
|
|
|
* corresponding bit in the injector's bloom filter.
|
|
|
|
*
|
2018-10-02 21:12:26 -07:00
|
|
|
* @param injectorIndex The index of the node injector where this token should be registered
|
|
|
|
* @param tView The TView for the injector's bloom filters
|
|
|
|
* @param type The directive token to register
|
2017-12-15 14:59:17 +01:00
|
|
|
*/
|
2018-10-02 21:12:26 -07:00
|
|
|
export function bloomAdd(injectorIndex: number, tView: TView, type: Type<any>): void {
|
|
|
|
if (tView.firstTemplatePass) {
|
|
|
|
let id: number|undefined = (type as any)[NG_ELEMENT_ID];
|
2017-12-15 14:59:17 +01:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// Set a unique ID on the directive type, so if something tries to inject the directive,
|
|
|
|
// we can easily retrieve the ID and hash it into the bloom bit that should be checked.
|
|
|
|
if (id == null) {
|
|
|
|
id = (type as any)[NG_ELEMENT_ID] = nextNgElementId++;
|
|
|
|
}
|
2017-12-15 14:59:17 +01:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// We only have BLOOM_SIZE (256) slots in our bloom filter (8 buckets * 32 bits each),
|
|
|
|
// so all unique IDs must be modulo-ed into a number from 0 - 255 to fit into the filter.
|
|
|
|
const bloomBit = id & BLOOM_MASK;
|
2017-12-15 14:59:17 +01:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// Create a mask that targets the specific bit associated with the directive.
|
|
|
|
// JS bit operations are 32 bits, so this will be a number between 2^0 and 2^31, corresponding
|
|
|
|
// to bit positions 0 - 31 in a 32 bit integer.
|
|
|
|
const mask = 1 << bloomBit;
|
2017-12-15 14:59:17 +01:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// Use the raw bloomBit number to determine which bloom filter bucket we should check
|
|
|
|
// e.g: bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127], etc
|
|
|
|
const b7 = bloomBit & 0x80;
|
|
|
|
const b6 = bloomBit & 0x40;
|
|
|
|
const b5 = bloomBit & 0x20;
|
|
|
|
const tData = tView.data as number[];
|
2018-07-27 12:44:58 -07:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
if (b7) {
|
|
|
|
b6 ? (b5 ? (tData[injectorIndex + 7] |= mask) : (tData[injectorIndex + 6] |= mask)) :
|
|
|
|
(b5 ? (tData[injectorIndex + 5] |= mask) : (tData[injectorIndex + 4] |= mask));
|
|
|
|
} else {
|
|
|
|
b6 ? (b5 ? (tData[injectorIndex + 3] |= mask) : (tData[injectorIndex + 2] |= mask)) :
|
|
|
|
(b5 ? (tData[injectorIndex + 1] |= mask) : (tData[injectorIndex] |= mask));
|
|
|
|
}
|
2017-12-15 14:59:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
export function getOrCreateNodeInjector(): number {
|
2018-07-26 17:22:41 +02:00
|
|
|
return getOrCreateNodeInjectorForNode(
|
2018-09-17 14:32:45 -07:00
|
|
|
getPreviousOrParentTNode() as TElementNode | TElementContainerNode | TContainerNode,
|
|
|
|
_getViewData());
|
2018-01-17 09:45:40 -08:00
|
|
|
}
|
|
|
|
|
2017-12-15 14:59:17 +01:00
|
|
|
/**
|
|
|
|
* Creates (or gets an existing) injector for a given element or container.
|
|
|
|
*
|
2018-09-17 14:32:45 -07:00
|
|
|
* @param tNode for which an injector should be retrieved / created.
|
|
|
|
* @param hostView View where the node is stored
|
2017-12-19 15:01:05 -08:00
|
|
|
* @returns Node injector
|
2017-12-15 14:59:17 +01:00
|
|
|
*/
|
2018-07-26 17:22:41 +02:00
|
|
|
export function getOrCreateNodeInjectorForNode(
|
2018-10-02 21:12:26 -07:00
|
|
|
tNode: TElementNode | TContainerNode | TElementContainerNode, hostView: LViewData): number {
|
|
|
|
const existingInjectorIndex = getInjectorIndex(tNode, hostView);
|
|
|
|
if (existingInjectorIndex !== -1) {
|
|
|
|
return existingInjectorIndex;
|
|
|
|
}
|
2018-09-28 21:26:45 -07:00
|
|
|
|
|
|
|
const tView = hostView[TVIEW];
|
|
|
|
if (tView.firstTemplatePass) {
|
|
|
|
tNode.injectorIndex = hostView.length;
|
2018-10-04 13:28:39 -07:00
|
|
|
setUpBloom(tView.data, tNode); // foundation for node bloom
|
|
|
|
setUpBloom(hostView, null); // foundation for cumulative bloom
|
|
|
|
setUpBloom(tView.blueprint, null);
|
2018-10-02 21:12:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const parentLoc = getParentInjectorLocation(tNode, hostView);
|
|
|
|
const parentIndex = parentLoc & InjectorLocationFlags.InjectorIndexMask;
|
|
|
|
const parentView: LViewData = getParentInjectorView(parentLoc, hostView);
|
|
|
|
|
|
|
|
const parentData = parentView[TVIEW].data as any;
|
|
|
|
const injectorIndex = tNode.injectorIndex;
|
|
|
|
|
2018-10-04 13:28:39 -07:00
|
|
|
// If a parent injector can't be found, its location is set to -1.
|
|
|
|
// In that case, we don't need to set up a cumulative bloom
|
|
|
|
if (parentLoc !== -1) {
|
|
|
|
for (let i = 0; i < PARENT_INJECTOR; i++) {
|
|
|
|
const bloomIndex = parentIndex + i;
|
|
|
|
// Creates a cumulative bloom filter that merges the parent's bloom filter
|
|
|
|
// and its own cumulative bloom (which contains tokens for all ancestors)
|
|
|
|
hostView[injectorIndex + i] = parentView[bloomIndex] | parentData[bloomIndex];
|
|
|
|
}
|
2017-12-15 14:59:17 +01:00
|
|
|
}
|
2018-09-28 21:26:45 -07:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
hostView[injectorIndex + PARENT_INJECTOR] = parentLoc;
|
|
|
|
return injectorIndex;
|
2017-12-15 14:59:17 +01:00
|
|
|
}
|
|
|
|
|
2018-10-04 13:28:39 -07:00
|
|
|
function setUpBloom(arr: any[], footer: TNode | null) {
|
|
|
|
arr.push(0, 0, 0, 0, 0, 0, 0, 0, footer);
|
|
|
|
}
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
export function getInjectorIndex(tNode: TNode, hostView: LViewData): number {
|
2018-09-28 21:26:45 -07:00
|
|
|
if (tNode.injectorIndex === -1 ||
|
2018-10-02 21:12:26 -07:00
|
|
|
// If the injector index is the same as its parent's injector index, then the index has been
|
|
|
|
// copied down from the parent node. No injector has been created yet on this node.
|
|
|
|
(tNode.parent && tNode.parent.injectorIndex === tNode.injectorIndex) ||
|
|
|
|
// After the first template pass, the injector index might exist but the parent values
|
|
|
|
// might not have been calculated yet for this instance
|
|
|
|
hostView[tNode.injectorIndex + PARENT_INJECTOR] == null) {
|
|
|
|
return -1;
|
2018-09-28 21:26:45 -07:00
|
|
|
} else {
|
2018-10-02 21:12:26 -07:00
|
|
|
return tNode.injectorIndex;
|
2018-09-28 21:26:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
/**
|
|
|
|
* Finds the index of the parent injector, with a view offset if applicable. Used to set the
|
|
|
|
* parent injector initially.
|
|
|
|
*/
|
|
|
|
export function getParentInjectorLocation(tNode: TNode, view: LViewData): number {
|
2018-09-28 21:26:45 -07:00
|
|
|
if (tNode.parent && tNode.parent.injectorIndex !== -1) {
|
2018-10-02 21:12:26 -07:00
|
|
|
return tNode.parent.injectorIndex; // view offset is 0
|
2018-09-28 21:26:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// For most cases, the parent injector index can be found on the host node (e.g. for component
|
|
|
|
// or container), so this loop will be skipped, but we must keep the loop here to support
|
2018-10-02 21:12:26 -07:00
|
|
|
// the rarer case of deeply nested <ng-template> tags or inline views.
|
2018-09-28 21:26:45 -07:00
|
|
|
let hostTNode = view[HOST_NODE];
|
2018-10-02 21:12:26 -07:00
|
|
|
let viewOffset = 1;
|
2018-09-28 21:26:45 -07:00
|
|
|
while (hostTNode && hostTNode.injectorIndex === -1) {
|
|
|
|
view = view[DECLARATION_VIEW] !;
|
|
|
|
hostTNode = view[HOST_NODE] !;
|
2018-10-02 21:12:26 -07:00
|
|
|
viewOffset++;
|
2018-09-28 21:26:45 -07:00
|
|
|
}
|
2018-10-02 21:12:26 -07:00
|
|
|
return hostTNode ?
|
|
|
|
hostTNode.injectorIndex | (viewOffset << InjectorLocationFlags.ViewOffsetShift) :
|
|
|
|
-1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unwraps a parent injector location number to find the view offset from the current injector,
|
|
|
|
* then walks up the declaration view tree until the view is found that contains the parent
|
|
|
|
* injector.
|
|
|
|
*
|
|
|
|
* @param location The location of the parent injector, which contains the view offset
|
|
|
|
* @param startView The LViewData instance from which to start walking up the view tree
|
|
|
|
* @returns The LViewData instance that contains the parent injector
|
|
|
|
*/
|
|
|
|
export function getParentInjectorView(location: number, startView: LViewData): LViewData {
|
|
|
|
let viewOffset = location >> InjectorLocationFlags.ViewOffsetShift;
|
|
|
|
let parentView = startView;
|
|
|
|
// For most cases, the parent injector can be found on the host node (e.g. for component
|
|
|
|
// or container), but we must keep the loop here to support the rarer case of deeply nested
|
|
|
|
// <ng-template> tags or inline views, where the parent injector might live many views
|
|
|
|
// above the child injector.
|
|
|
|
while (viewOffset > 0) {
|
|
|
|
parentView = parentView[DECLARATION_VIEW] !;
|
|
|
|
viewOffset--;
|
|
|
|
}
|
|
|
|
return parentView;
|
2018-09-28 21:26:45 -07:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2017-12-15 14:59:17 +01:00
|
|
|
/**
|
|
|
|
* Makes a directive public to the DI system by adding it to an injector's bloom filter.
|
|
|
|
*
|
|
|
|
* @param di The node injector in which a directive will be added
|
|
|
|
* @param def The definition of the directive to be made public
|
|
|
|
*/
|
2018-10-02 21:12:26 -07:00
|
|
|
export function diPublicInInjector(
|
2018-09-21 12:12:06 -07:00
|
|
|
injectorIndex: number, view: LViewData, def: DirectiveDef<any>): void {
|
2018-10-02 21:12:26 -07:00
|
|
|
bloomAdd(injectorIndex, view[TVIEW], def.type);
|
2017-12-15 14:59:17 +01:00
|
|
|
}
|
|
|
|
|
2018-01-17 09:45:40 -08:00
|
|
|
/**
|
|
|
|
* Makes a directive public to the DI system by adding it to an injector's bloom filter.
|
|
|
|
*
|
|
|
|
* @param def The definition of the directive to be made public
|
|
|
|
*/
|
2018-09-21 12:12:06 -07:00
|
|
|
export function diPublic(def: DirectiveDef<any>): void {
|
2018-10-02 21:12:26 -07:00
|
|
|
diPublicInInjector(getOrCreateNodeInjector(), _getViewData(), def);
|
2018-01-17 09:45:40 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-07-27 09:47:34 -07:00
|
|
|
* Returns the value associated to the given token from the injectors.
|
2018-01-17 09:45:40 -08:00
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* `directiveInject` is intended to be used for directive, component and pipe factories.
|
|
|
|
* All other injection use `inject` which does not walk the node injector tree.
|
2018-01-17 09:45:40 -08:00
|
|
|
*
|
|
|
|
* Usage example (in factory function):
|
|
|
|
*
|
|
|
|
* class SomeDirective {
|
|
|
|
* constructor(directive: DirectiveA) {}
|
|
|
|
*
|
|
|
|
* static ngDirectiveDef = defineDirective({
|
|
|
|
* type: SomeDirective,
|
2018-03-04 20:21:23 -08:00
|
|
|
* factory: () => new SomeDirective(directiveInject(DirectiveA))
|
2018-01-17 09:45:40 -08:00
|
|
|
* });
|
|
|
|
* }
|
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* @param token the type or token to inject
|
|
|
|
* @param flags Injection flags
|
|
|
|
* @returns the value from the injector or `null` when not found
|
2018-01-17 09:45:40 -08:00
|
|
|
*/
|
2018-07-27 09:56:35 -07:00
|
|
|
export function directiveInject<T>(token: Type<T>| InjectionToken<T>): T;
|
|
|
|
export function directiveInject<T>(token: Type<T>| InjectionToken<T>, flags: InjectFlags): T;
|
|
|
|
export function directiveInject<T>(
|
|
|
|
token: Type<T>| InjectionToken<T>, flags = InjectFlags.Default): T|null {
|
2018-10-08 16:04:46 -07:00
|
|
|
const hostTNode =
|
|
|
|
getPreviousOrParentTNode() as TElementNode | TContainerNode | TElementContainerNode;
|
|
|
|
return getOrCreateInjectable<T>(hostTNode, _getViewData(), token, flags);
|
2018-01-17 09:45:40 -08:00
|
|
|
}
|
|
|
|
|
2018-02-28 22:18:34 -08:00
|
|
|
/**
|
|
|
|
* Inject static attribute value into directive constructor.
|
|
|
|
*
|
|
|
|
* This method is used with `factory` functions which are generated as part of
|
|
|
|
* `defineDirective` or `defineComponent`. The method retrieves the static value
|
|
|
|
* of an attribute. (Dynamic attributes are not supported since they are not resolved
|
|
|
|
* at the time of injection and can change over time.)
|
|
|
|
*
|
|
|
|
* # Example
|
|
|
|
* Given:
|
|
|
|
* ```
|
|
|
|
* @Component(...)
|
|
|
|
* class MyComponent {
|
|
|
|
* constructor(@Attribute('title') title: string) { ... }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
* When instantiated with
|
|
|
|
* ```
|
|
|
|
* <my-component title="Hello"></my-component>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Then factory method generated is:
|
|
|
|
* ```
|
|
|
|
* MyComponent.ngComponentDef = defineComponent({
|
|
|
|
* factory: () => new MyComponent(injectAttribute('title'))
|
|
|
|
* ...
|
|
|
|
* })
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @experimental
|
|
|
|
*/
|
2018-05-04 15:58:42 +02:00
|
|
|
export function injectAttribute(attrNameToInject: string): string|undefined {
|
2018-09-05 16:15:37 -07:00
|
|
|
const tNode = getPreviousOrParentTNode();
|
2018-08-28 14:13:32 +02:00
|
|
|
ngDevMode && assertNodeOfPossibleTypes(
|
2018-09-05 16:15:37 -07:00
|
|
|
tNode, TNodeType.Container, TNodeType.Element, TNodeType.ElementContainer);
|
2018-08-28 14:13:32 +02:00
|
|
|
ngDevMode && assertDefined(tNode, 'expecting tNode');
|
|
|
|
const attrs = tNode.attrs;
|
2018-02-28 22:18:34 -08:00
|
|
|
if (attrs) {
|
|
|
|
for (let i = 0; i < attrs.length; i = i + 2) {
|
2018-06-06 13:38:20 -07:00
|
|
|
const attrName = attrs[i];
|
2018-06-08 15:25:39 -07:00
|
|
|
if (attrName === AttributeMarker.SelectOnly) break;
|
2018-05-04 15:58:42 +02:00
|
|
|
if (attrName == attrNameToInject) {
|
|
|
|
return attrs[i + 1] as string;
|
2018-02-28 22:18:34 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2018-08-16 17:43:29 +02:00
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
2018-07-27 09:47:34 -07:00
|
|
|
* Returns the value associated to the given token from the injectors.
|
2017-12-13 14:28:36 -08:00
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* Look for the injector providing the token by walking up the node injector tree and then
|
|
|
|
* the module injector tree.
|
2017-12-13 14:28:36 -08:00
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* @param nodeInjector Node injector where the search should start
|
|
|
|
* @param token The token to look for
|
|
|
|
* @param flags Injection flags
|
|
|
|
* @returns the value from the injector or `null` when not found
|
2017-12-13 14:28:36 -08:00
|
|
|
*/
|
2018-04-23 18:24:40 -07:00
|
|
|
export function getOrCreateInjectable<T>(
|
2018-10-08 16:04:46 -07:00
|
|
|
hostTNode: TElementNode | TContainerNode | TElementContainerNode, hostView: LViewData,
|
|
|
|
token: Type<T>| InjectionToken<T>, flags: InjectFlags = InjectFlags.Default): T|null {
|
2018-09-21 18:38:13 -07:00
|
|
|
const bloomHash = bloomHashBitOrFactory(token);
|
|
|
|
// If the ID stored here is a function, this is a special object like ElementRef or TemplateRef
|
|
|
|
// so just call the factory function to create it.
|
|
|
|
if (typeof bloomHash === 'function') return bloomHash();
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// If the token has a bloom hash, then it is a directive that is public to the injection system
|
2018-07-27 09:56:35 -07:00
|
|
|
// (diPublic) otherwise fall back to the module injector.
|
2018-09-21 18:38:13 -07:00
|
|
|
if (bloomHash != null) {
|
2018-10-08 16:04:46 -07:00
|
|
|
const startInjectorIndex = getInjectorIndex(hostTNode, hostView);
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
let injectorIndex = startInjectorIndex;
|
|
|
|
let injectorView = hostView;
|
2018-10-08 16:04:46 -07:00
|
|
|
let parentLocation: number = -1;
|
|
|
|
|
|
|
|
// If we should skip this injector or if an injector doesn't exist on this node (e.g. all
|
|
|
|
// directives on this node are private), start by searching the parent injector.
|
|
|
|
if (flags & InjectFlags.SkipSelf || injectorIndex === -1) {
|
|
|
|
parentLocation = injectorIndex === -1 ? getParentInjectorLocation(hostTNode, hostView) :
|
|
|
|
injectorView[injectorIndex + PARENT_INJECTOR];
|
|
|
|
|
|
|
|
if (shouldNotSearchParent(flags, parentLocation)) {
|
|
|
|
injectorIndex = -1;
|
|
|
|
} else {
|
|
|
|
injectorIndex = parentLocation & InjectorLocationFlags.InjectorIndexMask;
|
|
|
|
injectorView = getParentInjectorView(parentLocation, injectorView);
|
|
|
|
}
|
2018-10-02 21:12:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
while (injectorIndex !== -1) {
|
|
|
|
// Traverse up the injector tree until we find a potential match or until we know there
|
|
|
|
// *isn't* a match. Outer loop is necessary in case we get a false positive injector.
|
|
|
|
while (injectorIndex !== -1) {
|
|
|
|
// Check the current injector. If it matches, stop searching for an injector.
|
|
|
|
if (injectorHasToken(bloomHash, injectorIndex, injectorView[TVIEW].data)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-10-08 16:04:46 -07:00
|
|
|
parentLocation = injectorView[injectorIndex + PARENT_INJECTOR];
|
|
|
|
if (shouldNotSearchParent(flags, parentLocation)) {
|
2018-10-02 21:12:26 -07:00
|
|
|
injectorIndex = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the ancestor bloom filter value has the bit corresponding to the directive, traverse
|
|
|
|
// up to find the specific injector. If the ancestor bloom filter does not have the bit, we
|
|
|
|
// can abort.
|
|
|
|
if (injectorHasToken(bloomHash, injectorIndex, injectorView)) {
|
|
|
|
injectorIndex = parentLocation & InjectorLocationFlags.InjectorIndexMask;
|
|
|
|
injectorView = getParentInjectorView(parentLocation, injectorView);
|
|
|
|
} else {
|
|
|
|
injectorIndex = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// If no injector is found, we *know* that there is no ancestor injector that contains the
|
|
|
|
// token, so we abort.
|
2018-10-02 21:12:26 -07:00
|
|
|
if (injectorIndex === -1) {
|
2017-12-14 15:03:46 -08:00
|
|
|
break;
|
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// At this point, we have an injector which *may* contain the token, so we step through the
|
|
|
|
// directives associated with the injector's corresponding node to get the directive instance.
|
2018-10-02 21:12:26 -07:00
|
|
|
let instance: T|null;
|
|
|
|
if (instance = searchDirectivesOnInjector<T>(injectorIndex, injectorView, token)) {
|
|
|
|
return instance;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
2018-04-04 21:21:12 -07:00
|
|
|
// If we *didn't* find the directive for the token and we are searching the current node's
|
|
|
|
// injector, it's possible the directive is on this node and hasn't been created yet.
|
2018-10-02 21:12:26 -07:00
|
|
|
if (injectorIndex === startInjectorIndex && hostView === injectorView &&
|
2018-09-17 14:32:45 -07:00
|
|
|
(instance = searchMatchesQueuedForCreation<T>(token, injectorView[TVIEW]))) {
|
2018-04-04 21:21:12 -07:00
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
2018-04-23 18:24:40 -07:00
|
|
|
// The def wasn't found anywhere on this node, so it was a false positive.
|
2018-10-02 21:12:26 -07:00
|
|
|
// Traverse up the tree and continue searching.
|
|
|
|
injectorIndex = parentLocation & InjectorLocationFlags.InjectorIndexMask;
|
|
|
|
injectorView = getParentInjectorView(parentLocation, injectorView);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
const moduleInjector = hostView[INJECTOR];
|
2018-07-27 09:56:35 -07:00
|
|
|
const formerInjector = setCurrentInjector(moduleInjector);
|
|
|
|
try {
|
|
|
|
return inject(token, flags);
|
|
|
|
} finally {
|
|
|
|
setCurrentInjector(formerInjector);
|
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-09-17 14:32:45 -07:00
|
|
|
function searchMatchesQueuedForCreation<T>(token: any, hostTView: TView): T|null {
|
|
|
|
const matches = hostTView.currentMatches;
|
2018-04-04 21:21:12 -07:00
|
|
|
if (matches) {
|
|
|
|
for (let i = 0; i < matches.length; i += 2) {
|
2018-09-21 12:12:06 -07:00
|
|
|
const def = matches[i] as DirectiveDef<any>;
|
2018-04-04 21:21:12 -07:00
|
|
|
if (def.type === token) {
|
2018-10-08 16:04:46 -07:00
|
|
|
return resolveDirective(def, i + 1, matches);
|
2018-04-04 21:21:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
function searchDirectivesOnInjector<T>(
|
|
|
|
injectorIndex: number, injectorView: LViewData, token: Type<T>| InjectionToken<T>) {
|
|
|
|
const tNode = injectorView[TVIEW].data[injectorIndex + TNODE] as TNode;
|
|
|
|
const nodeFlags = tNode.flags;
|
|
|
|
const count = nodeFlags & TNodeFlags.DirectiveCountMask;
|
|
|
|
|
|
|
|
if (count !== 0) {
|
|
|
|
const start = nodeFlags >> TNodeFlags.DirectiveStartingIndexShift;
|
|
|
|
const end = start + count;
|
2018-10-08 16:04:46 -07:00
|
|
|
const defs = injectorView[TVIEW].data;
|
2018-10-02 21:12:26 -07:00
|
|
|
|
|
|
|
for (let i = start; i < end; i++) {
|
|
|
|
// Get the definition for the directive at this index and, if it is injectable (diPublic),
|
|
|
|
// and matches the given token, return the directive instance.
|
2018-09-21 12:12:06 -07:00
|
|
|
const directiveDef = defs[i] as DirectiveDef<any>;
|
2018-10-02 21:12:26 -07:00
|
|
|
if (directiveDef.type === token && directiveDef.diPublic) {
|
2018-10-08 16:04:46 -07:00
|
|
|
return injectorView[i];
|
2018-10-02 21:12:26 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
2018-07-27 09:47:34 -07:00
|
|
|
* Returns the bit in an injector's bloom filter that should be used to determine whether or not
|
|
|
|
* the directive might be provided by the injector.
|
2017-12-13 14:28:36 -08:00
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* When a directive is public, it is added to the bloom filter and given a unique ID that can be
|
|
|
|
* retrieved on the Type. When the directive isn't public or the token is not a directive `null`
|
|
|
|
* is returned as the node injector can not possibly provide that token.
|
2017-12-13 14:28:36 -08:00
|
|
|
*
|
2018-07-27 09:47:34 -07:00
|
|
|
* @param token the injection token
|
|
|
|
* @returns the matching bit to check in the bloom filter or `null` if the token is not known.
|
2017-12-13 14:28:36 -08:00
|
|
|
*/
|
2018-10-02 21:12:26 -07:00
|
|
|
export function bloomHashBitOrFactory(token: Type<any>| InjectionToken<any>): number|Function|
|
|
|
|
undefined {
|
|
|
|
const tokenId: number|undefined = (token as any)[NG_ELEMENT_ID];
|
2018-09-21 18:38:13 -07:00
|
|
|
return typeof tokenId === 'number' ? tokenId & BLOOM_MASK : tokenId;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
export function injectorHasToken(
|
|
|
|
bloomHash: number, injectorIndex: number, injectorView: LViewData | TData) {
|
2017-12-13 16:32:21 -08:00
|
|
|
// Create a mask that targets the specific bit associated with the directive we're looking for.
|
2017-12-14 15:50:01 -08:00
|
|
|
// JS bit operations are 32 bits, so this will be a number between 2^0 and 2^31, corresponding
|
|
|
|
// to bit positions 0 - 31 in a 32 bit integer.
|
2018-10-02 21:12:26 -07:00
|
|
|
const mask = 1 << bloomHash;
|
|
|
|
const b7 = bloomHash & 0x80;
|
|
|
|
const b6 = bloomHash & 0x40;
|
|
|
|
const b5 = bloomHash & 0x20;
|
2018-07-27 12:44:58 -07:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// Our bloom filter size is 256 bits, which is eight 32-bit bloom filter buckets:
|
|
|
|
// bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127], etc.
|
|
|
|
// Get the bloom filter value from the appropriate bucket based on the directive's bloomBit.
|
|
|
|
let value: number;
|
2017-12-13 16:32:21 -08:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
if (b7) {
|
|
|
|
value = b6 ? (b5 ? injectorView[injectorIndex + 7] : injectorView[injectorIndex + 6]) :
|
|
|
|
(b5 ? injectorView[injectorIndex + 5] : injectorView[injectorIndex + 4]);
|
|
|
|
} else {
|
|
|
|
value = b6 ? (b5 ? injectorView[injectorIndex + 3] : injectorView[injectorIndex + 2]) :
|
|
|
|
(b5 ? injectorView[injectorIndex + 1] : injectorView[injectorIndex]);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-07-27 12:44:58 -07:00
|
|
|
|
2018-10-02 21:12:26 -07:00
|
|
|
// If the bloom filter value has the bit corresponding to the directive's bloomBit flipped on,
|
|
|
|
// this injector is a potential match.
|
|
|
|
return !!(value & mask);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-10-08 16:04:46 -07:00
|
|
|
/** Returns true if flags prevent parent injector from being searched for tokens */
|
|
|
|
function shouldNotSearchParent(flags: InjectFlags, parentLocation: number): boolean|number {
|
|
|
|
return flags & InjectFlags.Self ||
|
|
|
|
(flags & InjectFlags.Host && (parentLocation >> InjectorLocationFlags.ViewOffsetShift) > 0);
|
2018-04-23 18:24:40 -07:00
|
|
|
}
|
|
|
|
|
2018-08-06 14:09:38 -07:00
|
|
|
export class NodeInjector implements Injector {
|
2018-10-02 21:12:26 -07:00
|
|
|
private _injectorIndex: number;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
private _tNode: TElementNode|TContainerNode|TElementContainerNode,
|
|
|
|
private _hostView: LViewData) {
|
|
|
|
this._injectorIndex = getOrCreateNodeInjectorForNode(_tNode, _hostView);
|
|
|
|
}
|
2018-07-25 18:11:39 +02:00
|
|
|
|
|
|
|
get(token: any): any {
|
2018-10-02 21:12:26 -07:00
|
|
|
setEnvironment(this._tNode, this._hostView);
|
2018-10-08 16:04:46 -07:00
|
|
|
return getOrCreateInjectable(this._tNode, this._hostView, token);
|
2018-07-25 18:11:39 +02:00
|
|
|
}
|
|
|
|
}
|
2018-07-16 16:36:31 -07:00
|
|
|
export function getFactoryOf<T>(type: Type<any>): ((type?: Type<T>) => T)|null {
|
|
|
|
const typeAny = type as any;
|
2018-08-29 16:34:44 -07:00
|
|
|
const def = getComponentDef<T>(typeAny) || getDirectiveDef<T>(typeAny) ||
|
|
|
|
getPipeDef<T>(typeAny) || getInjectableDef<T>(typeAny) || getInjectorDef<T>(typeAny);
|
|
|
|
if (!def || def.factory === undefined) {
|
2018-07-16 16:36:31 -07:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return def.factory;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getInheritedFactory<T>(type: Type<any>): (type: Type<T>) => T {
|
|
|
|
const proto = Object.getPrototypeOf(type.prototype).constructor as Type<any>;
|
|
|
|
const factory = getFactoryOf<T>(proto);
|
2018-08-10 11:15:59 +01:00
|
|
|
if (factory !== null) {
|
|
|
|
return factory;
|
|
|
|
} else {
|
|
|
|
// There is no factory defined. Either this was improper usage of inheritance
|
|
|
|
// (no Angular decorator on the superclass) or there is no constructor at all
|
|
|
|
// in the inheritance chain. Since the two cases cannot be distinguished, the
|
|
|
|
// latter has to be assumed.
|
|
|
|
return (t) => new t();
|
2018-07-16 16:36:31 -07:00
|
|
|
}
|
|
|
|
}
|