157 lines
5.3 KiB
TypeScript
157 lines
5.3 KiB
TypeScript
|
/**
|
||
|
* @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 {ComponentFactory, ComponentRef as IComponentRef, ElementRef as IElementRef, EmbeddedViewRef as IEmbeddedViewRef, Injector, NgModuleRef as INgModuleRef, TemplateRef as ITemplateRef, Type, ViewContainerRef as IViewContainerRef, ViewRef as IViewRef} from '../core';
|
||
|
import {BLOOM_SIZE, NG_ELEMENT_ID, getOrCreateNodeInjector} from './instructions';
|
||
|
import {LContainer, LNodeFlags, LNodeInjector} from './interfaces';
|
||
|
import {ComponentTemplate} from './public_interfaces';
|
||
|
import {stringify} from './util';
|
||
|
|
||
|
export const enum InjectFlags {
|
||
|
Optional = 1 << 0,
|
||
|
CheckSelf = 1 << 1,
|
||
|
CheckParent = 1 << 2,
|
||
|
Default = CheckSelf | CheckParent
|
||
|
}
|
||
|
|
||
|
function createError(text: string, token: any) {
|
||
|
return new Error(`ElementInjector: ${text} [${stringify(token)}]`);
|
||
|
}
|
||
|
|
||
|
export function inject<T>(token: Type<T>, flags?: InjectFlags): T {
|
||
|
const di = getOrCreateNodeInjector();
|
||
|
const bloomHash = bloomHashBit(token);
|
||
|
if (bloomHash === null) {
|
||
|
const moduleInjector = di.injector;
|
||
|
if (!moduleInjector) {
|
||
|
throw createError('NotFound', token);
|
||
|
}
|
||
|
moduleInjector.get(token);
|
||
|
} else {
|
||
|
let injector: LNodeInjector|null = di;
|
||
|
while (injector) {
|
||
|
injector = bloomFindPossibleInjector(injector, bloomHash);
|
||
|
if (injector) {
|
||
|
const node = injector.node;
|
||
|
const flags = node.flags;
|
||
|
let size = flags & LNodeFlags.SIZE_MASK;
|
||
|
if (size !== 0) {
|
||
|
size = size >> LNodeFlags.SIZE_SHIFT;
|
||
|
const start = flags >> LNodeFlags.INDX_SHIFT;
|
||
|
const directives = node.view.directives;
|
||
|
if (directives) {
|
||
|
for (let i = start, ii = start + size; i < ii; i++) {
|
||
|
const def = directives[(i << 1) | 1];
|
||
|
if (def.diPublic && def.type == token) {
|
||
|
return directives[i << 1];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
injector = injector.parent;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
throw createError('Not found', token);
|
||
|
}
|
||
|
|
||
|
function bloomHashBit(type: Type<any>): number|null {
|
||
|
let id: number|undefined = (type as any)[NG_ELEMENT_ID];
|
||
|
return typeof id === 'number' ? id % BLOOM_SIZE : null;
|
||
|
}
|
||
|
|
||
|
export function bloomFindPossibleInjector(injector: LNodeInjector, bloomBit: number): LNodeInjector|
|
||
|
null {
|
||
|
const mask = 1 << bloomBit;
|
||
|
let di: LNodeInjector|null = injector;
|
||
|
while (di) {
|
||
|
// See if the current injector may have the value.
|
||
|
let value: number =
|
||
|
bloomBit < 64 ? (bloomBit < 32 ? di.bf0 : di.bf1) : (bloomBit < 96 ? di.bf2 : di.bf3);
|
||
|
if ((value & mask) === mask) {
|
||
|
return di;
|
||
|
}
|
||
|
// See if the parent injectors may have the value
|
||
|
value =
|
||
|
bloomBit < 64 ? (bloomBit < 32 ? di.cbf0 : di.cbf1) : (bloomBit < 96 ? di.cbf2 : di.cbf3);
|
||
|
// Only go to parent if parent may have value otherwise exit.
|
||
|
di = (value & mask) ? di.parent : null;
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
|
||
|
export function injectElementRef(): IElementRef {
|
||
|
let di = getOrCreateNodeInjector();
|
||
|
return di.elementRef || (di.elementRef = new ElementRef(di.node.native));
|
||
|
}
|
||
|
|
||
|
class ElementRef implements IElementRef {
|
||
|
readonly nativeElement: any;
|
||
|
constructor(nativeElement: any) { this.nativeElement = nativeElement; }
|
||
|
}
|
||
|
|
||
|
|
||
|
export function injectTemplateRef(): ITemplateRef<any> {
|
||
|
let di = getOrCreateNodeInjector();
|
||
|
const data = (di.node as LContainer).data;
|
||
|
if (data === null || data.template === null) {
|
||
|
throw createError('Directive not used in structural way.', null);
|
||
|
}
|
||
|
return di.templateRef ||
|
||
|
(di.templateRef = new TemplateRef<any>(injectElementRef(), data.template));
|
||
|
}
|
||
|
|
||
|
class TemplateRef<T> implements ITemplateRef<T> {
|
||
|
readonly elementRef: IElementRef;
|
||
|
|
||
|
constructor(elementRef: IElementRef, template: ComponentTemplate<T>) {
|
||
|
this.elementRef = elementRef;
|
||
|
}
|
||
|
|
||
|
createEmbeddedView(context: T): IEmbeddedViewRef<T> { throw notImplemented(); }
|
||
|
}
|
||
|
|
||
|
export function injectViewContainerRef(): IViewContainerRef {
|
||
|
let di = getOrCreateNodeInjector();
|
||
|
return di.viewContainerRef || (di.viewContainerRef = new ViewContainerRef(di.node as LContainer));
|
||
|
}
|
||
|
|
||
|
class ViewContainerRef implements IViewContainerRef {
|
||
|
element: IElementRef;
|
||
|
injector: Injector;
|
||
|
parentInjector: Injector;
|
||
|
|
||
|
constructor(node: LContainer) {}
|
||
|
|
||
|
clear(): void { throw notImplemented(); }
|
||
|
get(index: number): IViewRef|null { throw notImplemented(); }
|
||
|
length: number;
|
||
|
createEmbeddedView<C>(
|
||
|
templateRef: ITemplateRef<C>, context?: C|undefined,
|
||
|
index?: number|undefined): IEmbeddedViewRef<C> {
|
||
|
throw notImplemented();
|
||
|
}
|
||
|
createComponent<C>(
|
||
|
componentFactory: ComponentFactory<C>, index?: number|undefined,
|
||
|
injector?: Injector|undefined, projectableNodes?: any[][]|undefined,
|
||
|
ngModule?: INgModuleRef<any>|undefined): IComponentRef<C> {
|
||
|
throw notImplemented();
|
||
|
}
|
||
|
insert(viewRef: IViewRef, index?: number|undefined): IViewRef { throw notImplemented(); }
|
||
|
move(viewRef: IViewRef, currentIndex: number): IViewRef { throw notImplemented(); }
|
||
|
indexOf(viewRef: IViewRef): number { throw notImplemented(); }
|
||
|
remove(index?: number|undefined): void { throw notImplemented(); }
|
||
|
detach(index?: number|undefined): IViewRef|null { throw notImplemented(); }
|
||
|
}
|
||
|
|
||
|
|
||
|
function notImplemented() {
|
||
|
return new Error('Method not implemented.');
|
||
|
}
|