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-02-26 16:58:15 -08:00
|
|
|
import {ChangeDetectorRef as viewEngine_ChangeDetectorRef} from '../change_detection/change_detector_ref';
|
2017-12-20 10:47:22 -08:00
|
|
|
import {Injector} from '../di/injector';
|
|
|
|
import {ComponentFactory as viewEngine_ComponentFactory, ComponentRef as viewEngine_ComponentRef} from '../linker/component_factory';
|
|
|
|
import {ElementRef as viewEngine_ElementRef} from '../linker/element_ref';
|
|
|
|
import {NgModuleRef as viewEngine_NgModuleRef} from '../linker/ng_module_factory';
|
|
|
|
import {TemplateRef as viewEngine_TemplateRef} from '../linker/template_ref';
|
|
|
|
import {ViewContainerRef as viewEngine_ViewContainerRef} from '../linker/view_container_ref';
|
|
|
|
import {EmbeddedViewRef as viewEngine_EmbeddedViewRef, ViewRef as viewEngine_ViewRef} from '../linker/view_ref';
|
|
|
|
import {Type} from '../type';
|
|
|
|
|
2018-02-28 22:18:34 -08:00
|
|
|
import {assertLessThan, assertNotNull} from './assert';
|
2018-01-30 15:37:01 -08:00
|
|
|
import {assertPreviousIsParent, getDirectiveInstance, getPreviousOrParentNode, getRenderer, renderEmbeddedTemplate} from './instructions';
|
2018-01-22 15:27:21 -08:00
|
|
|
import {ComponentTemplate, DirectiveDef} from './interfaces/definition';
|
2018-01-09 18:38:17 -08:00
|
|
|
import {LInjector} from './interfaces/injector';
|
2018-01-17 10:09:05 -08:00
|
|
|
import {LContainerNode, LElementNode, LNode, LNodeFlags, LViewNode} from './interfaces/node';
|
2018-01-17 09:45:40 -08:00
|
|
|
import {QueryReadType} from './interfaces/query';
|
2018-01-17 10:09:05 -08:00
|
|
|
import {Renderer3} from './interfaces/renderer';
|
2018-03-08 16:55:47 -08:00
|
|
|
import {LView} from './interfaces/view';
|
2018-01-17 09:45:40 -08:00
|
|
|
import {assertNodeOfPossibleTypes, assertNodeType} from './node_assert';
|
2018-01-17 10:09:05 -08:00
|
|
|
import {insertView} from './node_manipulation';
|
2017-12-14 15:03:46 -08:00
|
|
|
import {notImplemented, stringify} from './util';
|
2018-02-26 16:58:15 -08:00
|
|
|
import {EmbeddedViewRef, ViewRef, addDestroyable, createViewRef} from './view_ref';
|
2017-12-14 15:03:46 -08:00
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2017-12-18 15:14:09 +01:00
|
|
|
|
2017-12-15 14:59:17 +01:00
|
|
|
/**
|
|
|
|
* If a directive is diPublic, bloomAdd sets a property on the instance with this constant as
|
|
|
|
* the key and the directive's unique ID as the value. This allows us to map directives to their
|
|
|
|
* bloom filter bit for DI.
|
|
|
|
*/
|
|
|
|
const NG_ELEMENT_ID = '__NG_ELEMENT_ID__';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
const BLOOM_SIZE = 128;
|
|
|
|
|
|
|
|
/** 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.
|
|
|
|
*
|
|
|
|
* @param injector The node injector in which the directive should be registered
|
|
|
|
* @param type The directive to register
|
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function bloomAdd(injector: LInjector, type: Type<any>): void {
|
2017-12-15 14:59:17 +01:00
|
|
|
let id: number|undefined = (type as any)[NG_ELEMENT_ID];
|
|
|
|
|
|
|
|
// 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++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We only have BLOOM_SIZE (128) slots in our bloom filter (4 buckets * 32 bits each),
|
|
|
|
// so all unique IDs must be modulo-ed into a number from 0 - 127 to fit into the filter.
|
|
|
|
// This means that after 128, some directives will share slots, leading to some false positives
|
|
|
|
// when checking for a directive's presence.
|
|
|
|
const bloomBit = id % BLOOM_SIZE;
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
|
|
|
|
// 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]
|
|
|
|
if (bloomBit < 64) {
|
|
|
|
if (bloomBit < 32) {
|
|
|
|
// Then use the mask to flip on the bit (0-31) associated with the directive in that bucket
|
|
|
|
injector.bf0 |= mask;
|
|
|
|
} else {
|
|
|
|
injector.bf1 |= mask;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (bloomBit < 96) {
|
|
|
|
injector.bf2 |= mask;
|
|
|
|
} else {
|
|
|
|
injector.bf3 |= mask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-17 09:45:40 -08:00
|
|
|
export function getOrCreateNodeInjector(): LInjector {
|
|
|
|
ngDevMode && assertPreviousIsParent();
|
|
|
|
return getOrCreateNodeInjectorForNode(getPreviousOrParentNode() as LElementNode | LContainerNode);
|
|
|
|
}
|
|
|
|
|
2017-12-15 14:59:17 +01:00
|
|
|
/**
|
|
|
|
* Creates (or gets an existing) injector for a given element or container.
|
|
|
|
*
|
2017-12-19 15:01:05 -08:00
|
|
|
* @param node for which an injector should be retrieved / created.
|
|
|
|
* @returns Node injector
|
2017-12-15 14:59:17 +01:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function getOrCreateNodeInjectorForNode(node: LElementNode | LContainerNode): LInjector {
|
2017-12-15 14:59:17 +01:00
|
|
|
const nodeInjector = node.nodeInjector;
|
|
|
|
const parentInjector = node.parent && node.parent.nodeInjector;
|
|
|
|
if (nodeInjector != parentInjector) {
|
|
|
|
return nodeInjector !;
|
|
|
|
}
|
|
|
|
return node.nodeInjector = {
|
|
|
|
parent: parentInjector,
|
|
|
|
node: node,
|
|
|
|
bf0: 0,
|
|
|
|
bf1: 0,
|
|
|
|
bf2: 0,
|
|
|
|
bf3: 0,
|
|
|
|
cbf0: parentInjector == null ? 0 : parentInjector.cbf0 | parentInjector.bf0,
|
|
|
|
cbf1: parentInjector == null ? 0 : parentInjector.cbf1 | parentInjector.bf1,
|
|
|
|
cbf2: parentInjector == null ? 0 : parentInjector.cbf2 | parentInjector.bf2,
|
|
|
|
cbf3: parentInjector == null ? 0 : parentInjector.cbf3 | parentInjector.bf3,
|
|
|
|
injector: null,
|
|
|
|
templateRef: null,
|
|
|
|
viewContainerRef: null,
|
2018-02-26 16:58:15 -08:00
|
|
|
elementRef: null,
|
|
|
|
changeDetectorRef: null
|
2017-12-15 14:59:17 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-12-13 16:32:21 -08:00
|
|
|
/** Injection flags for DI. */
|
2017-12-01 14:23:03 -08:00
|
|
|
export const enum InjectFlags {
|
2017-12-14 15:03:46 -08:00
|
|
|
/** Dependency is not required. Null will be injected if there is no provider for the dependency.
|
|
|
|
*/
|
2017-12-01 14:23:03 -08:00
|
|
|
Optional = 1 << 0,
|
2017-12-13 16:32:21 -08:00
|
|
|
/** When resolving a dependency, include the node that is requesting injection. */
|
2017-12-01 14:23:03 -08:00
|
|
|
CheckSelf = 1 << 1,
|
2017-12-13 16:32:21 -08:00
|
|
|
/** When resolving a dependency, include ancestors of the node requesting injection. */
|
2017-12-01 14:23:03 -08:00
|
|
|
CheckParent = 1 << 2,
|
2017-12-13 16:32:21 -08:00
|
|
|
/** Default injection options: required, checks both self and ancestors. */
|
|
|
|
Default = CheckSelf | CheckParent,
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* Constructs an injection error with the given text and token.
|
|
|
|
*
|
|
|
|
* @param text The text of the error
|
|
|
|
* @param token The token associated with the error
|
|
|
|
* @returns The error that was created
|
|
|
|
*/
|
|
|
|
function createInjectionError(text: string, token: any) {
|
2017-12-01 14:23:03 -08:00
|
|
|
return new Error(`ElementInjector: ${text} [${stringify(token)}]`);
|
|
|
|
}
|
|
|
|
|
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-01-22 15:27:21 -08:00
|
|
|
export function diPublicInInjector(di: LInjector, def: DirectiveDef<any>): void {
|
2017-12-15 14:59:17 +01:00
|
|
|
bloomAdd(di, def.type);
|
|
|
|
}
|
|
|
|
|
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-01-23 10:57:48 -08:00
|
|
|
export function diPublic(def: DirectiveDef<any>): void {
|
2018-01-17 09:45:40 -08:00
|
|
|
diPublicInInjector(getOrCreateNodeInjector(), def);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-03-04 20:21:23 -08:00
|
|
|
* Searches for an instance of the given type up the injector tree and returns
|
2018-01-17 09:45:40 -08:00
|
|
|
* that instance if found.
|
|
|
|
*
|
|
|
|
* If not found, it will propagate up to the next parent injector until the token
|
|
|
|
* is found or the top is reached.
|
|
|
|
*
|
|
|
|
* 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-03-04 20:21:23 -08:00
|
|
|
* NOTE: use `directiveInject` with `@Directive`, `@Component`, and `@Pipe`. For
|
|
|
|
* all other injection use `inject` which does not walk the DOM render tree.
|
|
|
|
*
|
2018-01-17 09:45:40 -08:00
|
|
|
* @param token The directive type to search for
|
|
|
|
* @param flags Injection flags (e.g. CheckParent)
|
|
|
|
* @returns The instance found
|
|
|
|
*/
|
2018-03-04 20:21:23 -08:00
|
|
|
export function directiveInject<T>(token: Type<T>, flags?: InjectFlags, defaultValue?: T): T {
|
2018-01-17 10:09:05 -08:00
|
|
|
return getOrCreateInjectable<T>(getOrCreateNodeInjector(), token, flags, defaultValue);
|
2018-01-17 09:45:40 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates an ElementRef and stores it on the injector.
|
|
|
|
* Or, if the ElementRef already exists, retrieves the existing ElementRef.
|
|
|
|
*
|
|
|
|
* @returns The ElementRef instance to use
|
|
|
|
*/
|
|
|
|
export function injectElementRef(): viewEngine_ElementRef {
|
|
|
|
return getOrCreateElementRef(getOrCreateNodeInjector());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a TemplateRef and stores it on the injector. Or, if the TemplateRef already
|
|
|
|
* exists, retrieves the existing TemplateRef.
|
|
|
|
*
|
|
|
|
* @returns The TemplateRef instance to use
|
|
|
|
*/
|
|
|
|
export function injectTemplateRef<T>(): viewEngine_TemplateRef<T> {
|
|
|
|
return getOrCreateTemplateRef<T>(getOrCreateNodeInjector());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a ViewContainerRef and stores it on the injector. Or, if the ViewContainerRef
|
|
|
|
* already exists, retrieves the existing ViewContainerRef.
|
|
|
|
*
|
|
|
|
* @returns The ViewContainerRef instance to use
|
|
|
|
*/
|
|
|
|
export function injectViewContainerRef(): viewEngine_ViewContainerRef {
|
|
|
|
return getOrCreateContainerRef(getOrCreateNodeInjector());
|
|
|
|
}
|
|
|
|
|
2018-02-26 16:58:15 -08:00
|
|
|
/** Returns a ChangeDetectorRef (a.k.a. a ViewRef) */
|
|
|
|
export function injectChangeDetectorRef(): viewEngine_ChangeDetectorRef {
|
|
|
|
return getOrCreateChangeDetectorRef(getOrCreateNodeInjector(), null);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
export function injectAttribute(attrName: string): string|undefined {
|
|
|
|
ngDevMode && assertPreviousIsParent();
|
|
|
|
const lElement = getPreviousOrParentNode() as LElementNode;
|
|
|
|
ngDevMode && assertNodeType(lElement, LNodeFlags.Element);
|
|
|
|
const tElement = lElement.tNode !;
|
|
|
|
ngDevMode && assertNotNull(tElement, 'expecting tNode');
|
|
|
|
const attrs = tElement.attrs;
|
|
|
|
if (attrs) {
|
|
|
|
for (let i = 0; i < attrs.length; i = i + 2) {
|
|
|
|
if (attrs[i] == attrName) {
|
|
|
|
return attrs[i + 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2018-02-26 16:58:15 -08:00
|
|
|
/**
|
|
|
|
* Creates a ViewRef and stores it on the injector as ChangeDetectorRef (public alias).
|
|
|
|
* Or, if it already exists, retrieves the existing instance.
|
|
|
|
*
|
|
|
|
* @returns The ChangeDetectorRef to use
|
|
|
|
*/
|
|
|
|
export function getOrCreateChangeDetectorRef(
|
|
|
|
di: LInjector, context: any): viewEngine_ChangeDetectorRef {
|
|
|
|
if (di.changeDetectorRef) return di.changeDetectorRef;
|
|
|
|
|
|
|
|
const currentNode = di.node;
|
|
|
|
if (currentNode.data === null) {
|
|
|
|
// if data is null, this node is a regular element node (not a component)
|
|
|
|
return di.changeDetectorRef = getOrCreateHostChangeDetector(currentNode.view.node);
|
|
|
|
} else if ((currentNode.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element) {
|
|
|
|
// if it's an element node with data, it's a component and context will be set later
|
2018-03-08 16:55:47 -08:00
|
|
|
return di.changeDetectorRef = createViewRef(currentNode.data as LView, context);
|
2018-02-26 16:58:15 -08:00
|
|
|
}
|
|
|
|
return null !;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Gets or creates ChangeDetectorRef for the closest host component */
|
|
|
|
function getOrCreateHostChangeDetector(currentNode: LViewNode | LElementNode):
|
|
|
|
viewEngine_ChangeDetectorRef {
|
|
|
|
const hostNode = getClosestComponentAncestor(currentNode);
|
|
|
|
const hostInjector = hostNode.nodeInjector;
|
|
|
|
const existingRef = hostInjector && hostInjector.changeDetectorRef;
|
|
|
|
|
2018-03-08 16:55:47 -08:00
|
|
|
return existingRef ?
|
|
|
|
existingRef :
|
|
|
|
createViewRef(
|
|
|
|
hostNode.data as LView, hostNode.view.data[hostNode.flags >> LNodeFlags.INDX_SHIFT]);
|
2018-02-26 16:58:15 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If the node is an embedded view, traverses up the view tree to return the closest
|
|
|
|
* ancestor view that is attached to a component. If it's already a component node,
|
|
|
|
* returns itself.
|
|
|
|
*/
|
|
|
|
function getClosestComponentAncestor(node: LViewNode | LElementNode): LElementNode {
|
|
|
|
while ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.View) {
|
|
|
|
node = node.view.node;
|
|
|
|
}
|
|
|
|
return node as LElementNode;
|
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* Searches for an instance of the given directive type up the injector tree and returns
|
|
|
|
* that instance if found.
|
|
|
|
*
|
|
|
|
* Specifically, it gets the bloom filter bit associated with the directive (see bloomHashBit),
|
|
|
|
* checks that bit against the bloom filter structure to identify an injector that might have
|
|
|
|
* the directive (see bloomFindPossibleInjector), then searches the directives on that injector
|
|
|
|
* for a match.
|
|
|
|
*
|
|
|
|
* If not found, it will propagate up to the next parent injector until the token
|
|
|
|
* is found or the top is reached.
|
|
|
|
*
|
2017-12-15 14:59:17 +01:00
|
|
|
* @param di Node injector where the search should start
|
2017-12-13 14:28:36 -08:00
|
|
|
* @param token The directive type to search for
|
|
|
|
* @param flags Injection flags (e.g. CheckParent)
|
|
|
|
* @returns The instance found
|
|
|
|
*/
|
2018-01-17 10:09:05 -08:00
|
|
|
export function getOrCreateInjectable<T>(
|
|
|
|
di: LInjector, token: Type<T>, flags?: InjectFlags, defaultValue?: T): T {
|
2017-12-01 14:23:03 -08:00
|
|
|
const bloomHash = bloomHashBit(token);
|
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
|
|
|
|
// (diPublic). If there is no hash, fall back to the module injector.
|
2017-12-01 14:23:03 -08:00
|
|
|
if (bloomHash === null) {
|
|
|
|
const moduleInjector = di.injector;
|
|
|
|
if (!moduleInjector) {
|
2018-01-17 10:09:05 -08:00
|
|
|
if (defaultValue != null) {
|
|
|
|
return defaultValue;
|
|
|
|
}
|
2017-12-13 14:28:36 -08:00
|
|
|
throw createInjectionError('NotFound', token);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
moduleInjector.get(token);
|
|
|
|
} else {
|
2018-01-08 20:17:13 -08:00
|
|
|
let injector: LInjector|null = di;
|
2017-12-13 16:32:21 -08:00
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
while (injector) {
|
2017-12-13 16:32:21 -08:00
|
|
|
// Get the closest potential matching injector (upwards in the injector tree) that
|
|
|
|
// *potentially* has the token.
|
2017-12-01 14:23:03 -08:00
|
|
|
injector = bloomFindPossibleInjector(injector, bloomHash);
|
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.
|
2017-12-14 15:03:46 -08:00
|
|
|
if (!injector) {
|
|
|
|
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.
|
|
|
|
const node = injector.node;
|
|
|
|
|
|
|
|
// The size of the node's directive's list is stored in certain bits of the node's flags,
|
|
|
|
// so exact it with a mask and shift it back such that the bits reflect the real value.
|
|
|
|
const flags = node.flags;
|
|
|
|
const size = (flags & LNodeFlags.SIZE_MASK) >> LNodeFlags.SIZE_SHIFT;
|
|
|
|
|
|
|
|
if (size !== 0) {
|
|
|
|
// The start index of the directives list is also part of the node's flags, but there is
|
|
|
|
// nothing to the "left" of it so it doesn't need a mask.
|
|
|
|
const start = flags >> LNodeFlags.INDX_SHIFT;
|
|
|
|
|
2018-01-10 18:19:16 -08:00
|
|
|
const tData = node.view.tView.data;
|
2017-12-13 16:32:21 -08:00
|
|
|
for (let i = start, ii = start + size; i < ii; 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-01-22 15:27:21 -08:00
|
|
|
const directiveDef = tData[i] as DirectiveDef<any>;
|
2017-12-13 16:32:21 -08:00
|
|
|
if (directiveDef.diPublic && directiveDef.type == token) {
|
2018-01-30 15:37:01 -08:00
|
|
|
return getDirectiveInstance(node.view.data[i]);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// If we *didn't* find the directive for the token from the candidate injector, we had a false
|
|
|
|
// positive. Traverse up the tree and continue.
|
|
|
|
injector = injector.parent;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// No directive was found for the given token.
|
|
|
|
// TODO: implement optional, check-self, and check-parent.
|
2017-12-13 14:28:36 -08:00
|
|
|
throw createInjectionError('Not found', token);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* Given a directive type, this function returns the bit in an injector's bloom filter
|
|
|
|
* that should be used to determine whether or not the directive is present.
|
|
|
|
*
|
|
|
|
* When the directive was added to the bloom filter, it was given a unique ID that can be
|
|
|
|
* retrieved on the class. Since there are only BLOOM_SIZE slots per bloom filter, the directive's
|
|
|
|
* ID must be modulo-ed by BLOOM_SIZE to get the correct bloom bit (directives share slots after
|
|
|
|
* BLOOM_SIZE is reached).
|
|
|
|
*
|
|
|
|
* @param type The directive type
|
|
|
|
* @returns The bloom bit to check for the directive
|
|
|
|
*/
|
2017-12-20 10:47:22 -08:00
|
|
|
function bloomHashBit(type: Type<any>): number|null {
|
2017-12-01 14:23:03 -08:00
|
|
|
let id: number|undefined = (type as any)[NG_ELEMENT_ID];
|
|
|
|
return typeof id === 'number' ? id % BLOOM_SIZE : null;
|
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* Finds the closest injector that might have a certain directive.
|
|
|
|
*
|
|
|
|
* Each directive corresponds to a bit in an injector's bloom filter. Given the bloom bit to
|
2017-12-13 16:32:21 -08:00
|
|
|
* check and a starting injector, this function traverses up injectors until it finds an
|
2017-12-13 14:28:36 -08:00
|
|
|
* injector that contains a 1 for that bit in its bloom filter. A 1 indicates that the
|
|
|
|
* injector may have that directive. It only *may* have the directive because directives begin
|
|
|
|
* to share bloom filter bits after the BLOOM_SIZE is reached, and it could correspond to a
|
|
|
|
* different directive sharing the bit.
|
|
|
|
*
|
|
|
|
* Note: We can skip checking further injectors up the tree if an injector's cbf structure
|
|
|
|
* has a 0 for that bloom bit. Since cbf contains the merged value of all the parent
|
|
|
|
* injectors, a 0 in the bloom bit indicates that the parents definitely do not contain
|
|
|
|
* the directive and do not need to be checked.
|
|
|
|
*
|
2017-12-15 14:59:17 +01:00
|
|
|
* @param injector The starting node injector to check
|
2017-12-13 14:28:36 -08:00
|
|
|
* @param bloomBit The bit to check in each injector's bloom filter
|
|
|
|
* @returns An injector that might have the directive
|
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function bloomFindPossibleInjector(startInjector: LInjector, bloomBit: number): LInjector|
|
|
|
|
null {
|
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.
|
2017-12-01 14:23:03 -08:00
|
|
|
const mask = 1 << bloomBit;
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// Traverse up the injector tree until we find a potential match or until we know there *isn't* a
|
|
|
|
// match.
|
2018-01-08 20:17:13 -08:00
|
|
|
let injector: LInjector|null = startInjector;
|
2017-12-13 16:32:21 -08:00
|
|
|
while (injector) {
|
|
|
|
// Our bloom filter size is 128 bits, which is four 32-bit bloom filter buckets:
|
|
|
|
// bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127]
|
|
|
|
// Get the bloom filter value from the appropriate bucket based on the directive's bloomBit.
|
2017-12-14 15:03:46 -08:00
|
|
|
let value: number = bloomBit < 64 ? (bloomBit < 32 ? injector.bf0 : injector.bf1) :
|
|
|
|
(bloomBit < 96 ? injector.bf2 : injector.bf3);
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// If the bloom filter value has the bit corresponding to the directive's bloomBit flipped on,
|
|
|
|
// this injector is a potential match.
|
2017-12-01 14:23:03 -08:00
|
|
|
if ((value & mask) === mask) {
|
2017-12-13 16:32:21 -08:00
|
|
|
return injector;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// If the current injector does not have the directive, check the bloom filters for the ancestor
|
|
|
|
// injectors (cbf0 - cbf3). These filters capture *all* ancestor injectors.
|
2017-12-14 15:03:46 -08:00
|
|
|
value = bloomBit < 64 ? (bloomBit < 32 ? injector.cbf0 : injector.cbf1) :
|
|
|
|
(bloomBit < 96 ? injector.cbf2 : injector.cbf3);
|
2017-12-13 16:32:21 -08:00
|
|
|
|
|
|
|
// 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.
|
|
|
|
injector = (value & mask) ? injector.parent : null;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-01-17 09:45:40 -08:00
|
|
|
export class ReadFromInjectorFn<T> {
|
|
|
|
constructor(readonly read: (injector: LInjector, node: LNode, directiveIndex?: number) => T) {}
|
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
2017-12-15 14:59:17 +01:00
|
|
|
* Creates an ElementRef for a given node injector and stores it on the injector.
|
2017-12-12 14:42:28 +01:00
|
|
|
* Or, if the ElementRef already exists, retrieves the existing ElementRef.
|
2017-12-13 14:28:36 -08:00
|
|
|
*
|
2017-12-15 14:59:17 +01:00
|
|
|
* @param di The node injector where we should store a created ElementRef
|
2017-12-13 14:28:36 -08:00
|
|
|
* @returns The ElementRef instance to use
|
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function getOrCreateElementRef(di: LInjector): viewEngine_ElementRef {
|
2018-01-26 11:36:31 +01:00
|
|
|
return di.elementRef ||
|
|
|
|
(di.elementRef = new ElementRef(
|
|
|
|
((di.node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Container) ? null :
|
|
|
|
di.node.native));
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-01-17 09:45:40 -08:00
|
|
|
export const QUERY_READ_TEMPLATE_REF = <QueryReadType<viewEngine_TemplateRef<any>>>(
|
|
|
|
new ReadFromInjectorFn<viewEngine_TemplateRef<any>>(
|
|
|
|
(injector: LInjector) => getOrCreateTemplateRef(injector)) as any);
|
|
|
|
|
|
|
|
export const QUERY_READ_CONTAINER_REF = <QueryReadType<viewEngine_ViewContainerRef>>(
|
|
|
|
new ReadFromInjectorFn<viewEngine_ViewContainerRef>(
|
|
|
|
(injector: LInjector) => getOrCreateContainerRef(injector)) as any);
|
|
|
|
|
|
|
|
export const QUERY_READ_ELEMENT_REF =
|
|
|
|
<QueryReadType<viewEngine_ElementRef>>(new ReadFromInjectorFn<viewEngine_ElementRef>(
|
|
|
|
(injector: LInjector) => getOrCreateElementRef(injector)) as any);
|
|
|
|
|
|
|
|
export const QUERY_READ_FROM_NODE =
|
|
|
|
(new ReadFromInjectorFn<any>((injector: LInjector, node: LNode, directiveIdx: number) => {
|
|
|
|
ngDevMode && assertNodeOfPossibleTypes(node, LNodeFlags.Container, LNodeFlags.Element);
|
|
|
|
if (directiveIdx > -1) {
|
|
|
|
return node.view.data[directiveIdx];
|
|
|
|
} else if ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element) {
|
|
|
|
return getOrCreateElementRef(injector);
|
|
|
|
} else if ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Container) {
|
|
|
|
return getOrCreateTemplateRef(injector);
|
|
|
|
}
|
|
|
|
throw new Error('fail');
|
|
|
|
}) as any as QueryReadType<any>);
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/** A ref to a node's native element. */
|
2017-12-20 10:47:22 -08:00
|
|
|
class ElementRef implements viewEngine_ElementRef {
|
2017-12-01 14:23:03 -08:00
|
|
|
readonly nativeElement: any;
|
|
|
|
constructor(nativeElement: any) { this.nativeElement = nativeElement; }
|
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* Creates a ViewContainerRef and stores it on the injector. Or, if the ViewContainerRef
|
|
|
|
* already exists, retrieves the existing ViewContainerRef.
|
|
|
|
*
|
|
|
|
* @returns The ViewContainerRef instance to use
|
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function getOrCreateContainerRef(di: LInjector): viewEngine_ViewContainerRef {
|
|
|
|
return di.viewContainerRef ||
|
|
|
|
(di.viewContainerRef = new ViewContainerRef(di.node as LContainerNode));
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 14:28:36 -08:00
|
|
|
/**
|
|
|
|
* A ref to a container that enables adding and removing views from that container
|
|
|
|
* imperatively.
|
|
|
|
*/
|
2017-12-20 10:47:22 -08:00
|
|
|
class ViewContainerRef implements viewEngine_ViewContainerRef {
|
|
|
|
element: viewEngine_ElementRef;
|
|
|
|
injector: Injector;
|
|
|
|
parentInjector: Injector;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-17 10:09:05 -08:00
|
|
|
constructor(private _node: LContainerNode) {}
|
2017-12-01 14:23:03 -08:00
|
|
|
|
|
|
|
clear(): void { throw notImplemented(); }
|
2017-12-20 10:47:22 -08:00
|
|
|
get(index: number): viewEngine_ViewRef|null { throw notImplemented(); }
|
2017-12-01 14:23:03 -08:00
|
|
|
length: number;
|
|
|
|
createEmbeddedView<C>(
|
2017-12-20 10:47:22 -08:00
|
|
|
templateRef: viewEngine_TemplateRef<C>, context?: C|undefined,
|
|
|
|
index?: number|undefined): viewEngine_EmbeddedViewRef<C> {
|
2018-01-17 10:09:05 -08:00
|
|
|
const viewRef = templateRef.createEmbeddedView(context !);
|
|
|
|
this.insert(viewRef, index);
|
|
|
|
return viewRef;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
createComponent<C>(
|
2017-12-20 10:47:22 -08:00
|
|
|
componentFactory: viewEngine_ComponentFactory<C>, index?: number|undefined,
|
|
|
|
injector?: Injector|undefined, projectableNodes?: any[][]|undefined,
|
|
|
|
ngModule?: viewEngine_NgModuleRef<any>|undefined): viewEngine_ComponentRef<C> {
|
2017-12-01 14:23:03 -08:00
|
|
|
throw notImplemented();
|
|
|
|
}
|
2017-12-20 10:47:22 -08:00
|
|
|
insert(viewRef: viewEngine_ViewRef, index?: number|undefined): viewEngine_ViewRef {
|
2018-01-17 10:09:05 -08:00
|
|
|
if (index == null) {
|
|
|
|
index = this._node.data.views.length;
|
|
|
|
} else {
|
|
|
|
// +1 because it's legal to insert at the end.
|
|
|
|
ngDevMode && assertLessThan(index, this._node.data.views.length + 1, 'index');
|
|
|
|
}
|
|
|
|
const lView = (viewRef as EmbeddedViewRef<any>)._lViewNode;
|
|
|
|
insertView(this._node, lView, index);
|
|
|
|
|
2018-03-08 12:10:20 +01:00
|
|
|
// TODO(pk): this is a temporary index adjustment so imperativelly inserted (through
|
|
|
|
// ViewContainerRef) views
|
|
|
|
// are not removed in the containerRefreshEnd instruction.
|
|
|
|
// The final fix will consist of creating a dedicated container node for views inserted through
|
|
|
|
// ViewContainerRef.
|
|
|
|
// Such container should not be trimmed as it is the case in the containerRefreshEnd
|
|
|
|
// instruction.
|
|
|
|
this._node.data.nextIndex = this._node.data.views.length;
|
|
|
|
|
2018-01-17 10:09:05 -08:00
|
|
|
// If the view is dynamic (has a template), it needs to be counted both at the container
|
|
|
|
// level and at the node above the container.
|
|
|
|
if (lView.data.template !== null) {
|
|
|
|
// Increment the container view count.
|
|
|
|
this._node.data.dynamicViewCount++;
|
|
|
|
|
|
|
|
// Look for the parent node and increment its dynamic view count.
|
|
|
|
if (this._node.parent !== null && this._node.parent.data !== null) {
|
|
|
|
ngDevMode &&
|
|
|
|
assertNodeOfPossibleTypes(this._node.parent, LNodeFlags.View, LNodeFlags.Element);
|
|
|
|
this._node.parent.data.dynamicViewCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return viewRef;
|
2017-12-14 15:03:46 -08:00
|
|
|
}
|
2017-12-20 10:47:22 -08:00
|
|
|
move(viewRef: viewEngine_ViewRef, currentIndex: number): viewEngine_ViewRef {
|
2017-12-14 15:03:46 -08:00
|
|
|
throw notImplemented();
|
|
|
|
}
|
2017-12-20 10:47:22 -08:00
|
|
|
indexOf(viewRef: viewEngine_ViewRef): number { throw notImplemented(); }
|
2017-12-01 14:23:03 -08:00
|
|
|
remove(index?: number|undefined): void { throw notImplemented(); }
|
2017-12-20 10:47:22 -08:00
|
|
|
detach(index?: number|undefined): viewEngine_ViewRef|null { throw notImplemented(); }
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-01-17 10:09:05 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a TemplateRef and stores it on the injector. Or, if the TemplateRef already
|
|
|
|
* exists, retrieves the existing TemplateRef.
|
|
|
|
*
|
|
|
|
* @param di The node injector where we should store a created TemplateRef
|
|
|
|
* @returns The TemplateRef instance to use
|
|
|
|
*/
|
|
|
|
export function getOrCreateTemplateRef<T>(di: LInjector): viewEngine_TemplateRef<T> {
|
|
|
|
ngDevMode && assertNodeType(di.node, LNodeFlags.Container);
|
|
|
|
const data = (di.node as LContainerNode).data;
|
|
|
|
return di.templateRef || (di.templateRef = new TemplateRef<any>(
|
|
|
|
getOrCreateElementRef(di), data.template !, getRenderer()));
|
|
|
|
}
|
|
|
|
|
|
|
|
class TemplateRef<T> implements viewEngine_TemplateRef<T> {
|
|
|
|
readonly elementRef: viewEngine_ElementRef;
|
|
|
|
private _template: ComponentTemplate<T>;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
elementRef: viewEngine_ElementRef, template: ComponentTemplate<T>,
|
|
|
|
private _renderer: Renderer3) {
|
|
|
|
this.elementRef = elementRef;
|
|
|
|
this._template = template;
|
|
|
|
}
|
|
|
|
|
|
|
|
createEmbeddedView(context: T): viewEngine_EmbeddedViewRef<T> {
|
|
|
|
let viewNode: LViewNode = renderEmbeddedTemplate(null, this._template, context, this._renderer);
|
2018-02-26 16:58:15 -08:00
|
|
|
return addDestroyable(new EmbeddedViewRef(viewNode, this._template, context));
|
2018-01-17 10:09:05 -08:00
|
|
|
}
|
|
|
|
}
|