| 
									
										
										
										
											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-04-05 10:29:47 +02:00
										 |  |  | import {assertGreaterThan, assertLessThan, assertNotNull} from './assert'; | 
					
						
							| 
									
										
										
										
											2018-04-04 21:21:12 -07:00
										 |  |  | import {addToViewTree, assertPreviousIsParent, createLContainer, createLNodeObject, getDirectiveInstance, getPreviousOrParentNode, getRenderer, isComponent, renderEmbeddedTemplate, resolveDirective} from './instructions'; | 
					
						
							| 
									
										
										
										
											2018-04-11 14:15:30 +02:00
										 |  |  | import {ComponentTemplate, DirectiveDef, DirectiveDefList, PipeDefList} from './interfaces/definition'; | 
					
						
							| 
									
										
										
										
											2018-01-09 18:38:17 -08:00
										 |  |  | import {LInjector} from './interfaces/injector'; | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  | import {LContainerNode, LElementNode, LNode, LNodeType, LViewNode, TNodeFlags} 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-03-15 17:33:35 +01:00
										 |  |  | import {insertView, removeView} 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. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  | const BLOOM_SIZE = 256; | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @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++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -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.
 | 
					
						
							|  |  |  |   // This means that after 255, some directives will share slots, leading to some false positives
 | 
					
						
							| 
									
										
										
										
											2017-12-15 14:59:17 +01:00
										 |  |  |   // 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
 | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |   // e.g: bf0 = [0 - 31], bf1 = [32 - 63], bf2 = [64 - 95], bf3 = [96 - 127], etc
 | 
					
						
							|  |  |  |   if (bloomBit < 128) { | 
					
						
							|  |  |  |     // Then use the mask to flip on the bit (0-31) associated with the directive in that bucket
 | 
					
						
							|  |  |  |     bloomBit < 64 ? (bloomBit < 32 ? (injector.bf0 |= mask) : (injector.bf1 |= mask)) : | 
					
						
							|  |  |  |                     (bloomBit < 96 ? (injector.bf2 |= mask) : (injector.bf3 |= mask)); | 
					
						
							| 
									
										
										
										
											2017-12-15 14:59:17 +01:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |     bloomBit < 192 ? (bloomBit < 160 ? (injector.bf4 |= mask) : (injector.bf5 |= mask)) : | 
					
						
							|  |  |  |                      (bloomBit < 224 ? (injector.bf6 |= mask) : (injector.bf7 |= mask)); | 
					
						
							| 
									
										
										
										
											2017-12-15 14:59:17 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |     bf4: 0, | 
					
						
							|  |  |  |     bf5: 0, | 
					
						
							|  |  |  |     bf6: 0, | 
					
						
							|  |  |  |     bf7: 0, | 
					
						
							| 
									
										
										
										
											2017-12-15 14:59:17 +01:00
										 |  |  |     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, | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |     cbf4: parentInjector == null ? 0 : parentInjector.cbf4 | parentInjector.bf4, | 
					
						
							|  |  |  |     cbf5: parentInjector == null ? 0 : parentInjector.cbf5 | parentInjector.bf5, | 
					
						
							|  |  |  |     cbf6: parentInjector == null ? 0 : parentInjector.cbf6 | parentInjector.bf6, | 
					
						
							|  |  |  |     cbf7: parentInjector == null ? 0 : parentInjector.cbf7 | parentInjector.bf7, | 
					
						
							| 
									
										
										
										
											2017-12-15 14:59:17 +01:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |   ngDevMode && assertNodeType(lElement, LNodeType.Element); | 
					
						
							| 
									
										
										
										
											2018-02-28 22:18:34 -08:00
										 |  |  |   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; | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  |   if (isComponent(currentNode.tNode !)) { | 
					
						
							| 
									
										
										
										
											2018-03-08 16:55:47 -08:00
										 |  |  |     return di.changeDetectorRef = createViewRef(currentNode.data as LView, context); | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  |   } else if (currentNode.type === LNodeType.Element) { | 
					
						
							|  |  |  |     return di.changeDetectorRef = getOrCreateHostChangeDetector(currentNode.view.node); | 
					
						
							| 
									
										
										
										
											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-21 15:10:34 -07:00
										 |  |  |   return existingRef ? | 
					
						
							|  |  |  |       existingRef : | 
					
						
							|  |  |  |       createViewRef( | 
					
						
							|  |  |  |           hostNode.data as LView, | 
					
						
							| 
									
										
										
										
											2018-04-12 14:52:00 -07:00
										 |  |  |           hostNode.view | 
					
						
							|  |  |  |               .directives ![hostNode.tNode !.flags >> TNodeFlags.DirectiveStartingIndexShift]); | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |   while (node.type === LNodeType.View) { | 
					
						
							| 
									
										
										
										
											2018-02-26 16:58:15 -08:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |       const flags = node.tNode !.flags; | 
					
						
							| 
									
										
										
										
											2018-04-12 14:52:00 -07:00
										 |  |  |       const count = flags & TNodeFlags.DirectiveCountMask; | 
					
						
							| 
									
										
										
										
											2017-12-13 16:32:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-12 14:52:00 -07:00
										 |  |  |       if (count !== 0) { | 
					
						
							|  |  |  |         const start = flags >> TNodeFlags.DirectiveStartingIndexShift; | 
					
						
							|  |  |  |         const end = start + count; | 
					
						
							| 
									
										
										
										
											2018-03-21 15:10:34 -07:00
										 |  |  |         const defs = node.view.tView.directives !; | 
					
						
							| 
									
										
										
										
											2018-04-12 14:52:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (let i = start; i < end; i++) { | 
					
						
							| 
									
										
										
										
											2017-12-13 16:32:21 -08:00
										 |  |  |           // 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-03-21 15:10:34 -07:00
										 |  |  |           const directiveDef = defs[i] as DirectiveDef<any>; | 
					
						
							| 
									
										
										
										
											2018-04-12 14:52:00 -07:00
										 |  |  |           if (directiveDef.type === token && directiveDef.diPublic) { | 
					
						
							| 
									
										
										
										
											2018-03-21 15:10:34 -07:00
										 |  |  |             return getDirectiveInstance(node.view.directives ![i]); | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  |       let instance: T|null; | 
					
						
							|  |  |  |       if (injector === di && (instance = searchMatchesQueuedForCreation<T>(node, token))) { | 
					
						
							|  |  |  |         return instance; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // The def wasn't found anywhere on this node, so it might be a false positive.
 | 
					
						
							|  |  |  |       // Traverse up the tree and continue searching.
 | 
					
						
							| 
									
										
										
										
											2017-12-13 16:32:21 -08:00
										 |  |  |       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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-04 21:21:12 -07:00
										 |  |  | function searchMatchesQueuedForCreation<T>(node: LNode, token: any): T|null { | 
					
						
							|  |  |  |   const matches = node.view.tView.currentMatches; | 
					
						
							|  |  |  |   if (matches) { | 
					
						
							|  |  |  |     for (let i = 0; i < matches.length; i += 2) { | 
					
						
							|  |  |  |       const def = matches[i] as DirectiveDef<any>; | 
					
						
							|  |  |  |       if (def.type === token) { | 
					
						
							|  |  |  |         return resolveDirective(def, i + 1, matches, node.view.tView); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return null; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -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.
 | 
					
						
							| 
									
										
										
										
											2017-12-13 16:32:21 -08:00
										 |  |  |     // Get the bloom filter value from the appropriate bucket based on the directive's bloomBit.
 | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |     let value: number; | 
					
						
							|  |  |  |     if (bloomBit < 128) { | 
					
						
							|  |  |  |       value = bloomBit < 64 ? (bloomBit < 32 ? injector.bf0 : injector.bf1) : | 
					
						
							|  |  |  |                               (bloomBit < 96 ? injector.bf2 : injector.bf3); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       value = bloomBit < 192 ? (bloomBit < 160 ? injector.bf4 : injector.bf5) : | 
					
						
							|  |  |  |                                (bloomBit < 224 ? injector.bf6 : injector.bf7); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2018-03-14 13:29:48 -07:00
										 |  |  |     // injectors (cbf0 - cbf7). These filters capture *all* ancestor injectors.
 | 
					
						
							|  |  |  |     if (bloomBit < 128) { | 
					
						
							|  |  |  |       value = bloomBit < 64 ? (bloomBit < 32 ? injector.cbf0 : injector.cbf1) : | 
					
						
							|  |  |  |                               (bloomBit < 96 ? injector.cbf2 : injector.cbf3); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       value = bloomBit < 192 ? (bloomBit < 160 ? injector.cbf4 : injector.cbf5) : | 
					
						
							|  |  |  |                                (bloomBit < 224 ? injector.cbf6 : injector.cbf7); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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-03-20 19:06:49 -07:00
										 |  |  |   return di.elementRef || (di.elementRef = new ElementRef( | 
					
						
							|  |  |  |                                di.node.type === LNodeType.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) => { | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |       ngDevMode && assertNodeOfPossibleTypes(node, LNodeType.Container, LNodeType.Element); | 
					
						
							| 
									
										
										
										
											2018-01-17 09:45:40 -08:00
										 |  |  |       if (directiveIdx > -1) { | 
					
						
							| 
									
										
										
										
											2018-03-21 15:10:34 -07:00
										 |  |  |         return node.view.directives ![directiveIdx]; | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |       } else if (node.type === LNodeType.Element) { | 
					
						
							| 
									
										
										
										
											2018-01-17 09:45:40 -08:00
										 |  |  |         return getOrCreateElementRef(injector); | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |       } else if (node.type === LNodeType.Container) { | 
					
						
							| 
									
										
										
										
											2018-01-17 09:45:40 -08:00
										 |  |  |         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 { | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   if (!di.viewContainerRef) { | 
					
						
							|  |  |  |     const vcRefHost = di.node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ngDevMode && assertNodeOfPossibleTypes(vcRefHost, LNodeType.Container, LNodeType.Element); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-03 10:18:25 +02:00
										 |  |  |     const lContainer = createLContainer(vcRefHost.parent !, vcRefHost.view); | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     const lContainerNode: LContainerNode = createLNodeObject( | 
					
						
							|  |  |  |         LNodeType.Container, vcRefHost.view, vcRefHost.parent !, undefined, lContainer, null); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-03 10:18:25 +02:00
										 |  |  |     vcRefHost.dynamicLContainerNode = lContainerNode; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     addToViewTree(vcRefHost.view, lContainer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     di.viewContainerRef = new ViewContainerRef(lContainerNode); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return di.viewContainerRef; | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   private _viewRefs: viewEngine_ViewRef[] = []; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   element: viewEngine_ElementRef; | 
					
						
							|  |  |  |   injector: Injector; | 
					
						
							|  |  |  |   parentInjector: Injector; | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   constructor(private _lContainerNode: LContainerNode) {} | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   clear(): void { | 
					
						
							|  |  |  |     const lContainer = this._lContainerNode.data; | 
					
						
							|  |  |  |     while (lContainer.views.length) { | 
					
						
							|  |  |  |       this.remove(0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   get(index: number): viewEngine_ViewRef|null { return this._viewRefs[index] || null; } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   get length(): number { | 
					
						
							|  |  |  |     const lContainer = this._lContainerNode.data; | 
					
						
							|  |  |  |     return lContainer.views.length; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   createEmbeddedView<C>(templateRef: viewEngine_TemplateRef<C>, context?: C, index?: number): | 
					
						
							|  |  |  |       viewEngine_EmbeddedViewRef<C> { | 
					
						
							|  |  |  |     const viewRef = templateRef.createEmbeddedView(context || <any>{}); | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |     this.insert(viewRef, index); | 
					
						
							|  |  |  |     return viewRef; | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   insert(viewRef: viewEngine_ViewRef, index?: number): viewEngine_ViewRef { | 
					
						
							|  |  |  |     const lViewNode = (viewRef as EmbeddedViewRef<any>)._lViewNode; | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |     const adjustedIdx = this._adjustIndex(index); | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     insertView(this._lContainerNode, lViewNode, adjustedIdx); | 
					
						
							| 
									
										
										
										
											2018-04-03 10:18:25 +02:00
										 |  |  |     // invalidate cache of next sibling RNode (we do similar operation in the containerRefreshEnd
 | 
					
						
							|  |  |  |     // instruction)
 | 
					
						
							|  |  |  |     this._lContainerNode.native = undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     this._viewRefs.splice(adjustedIdx, 0, viewRef); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     (lViewNode as{parent: LNode}).parent = this._lContainerNode; | 
					
						
							| 
									
										
										
										
											2018-03-08 12:10:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     if (lViewNode.data.template !== null) { | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |       // Increment the container view count.
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |       this._lContainerNode.data.dynamicViewCount++; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // Look for the parent node and increment its dynamic view count.
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |       if (this._lContainerNode.parent !== null && this._lContainerNode.parent.data !== null) { | 
					
						
							|  |  |  |         ngDevMode && assertNodeOfPossibleTypes( | 
					
						
							|  |  |  |                          this._lContainerNode.parent, LNodeType.View, LNodeType.Element); | 
					
						
							|  |  |  |         this._lContainerNode.parent.data.dynamicViewCount++; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return viewRef; | 
					
						
							| 
									
										
										
										
											2017-12-14 15:03:46 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   move(viewRef: viewEngine_ViewRef, newIndex: number): viewEngine_ViewRef { | 
					
						
							|  |  |  |     const index = this.indexOf(viewRef); | 
					
						
							|  |  |  |     this.detach(index); | 
					
						
							|  |  |  |     this.insert(viewRef, this._adjustIndex(newIndex)); | 
					
						
							|  |  |  |     return viewRef; | 
					
						
							| 
									
										
										
										
											2017-12-14 15:03:46 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   indexOf(viewRef: viewEngine_ViewRef): number { return this._viewRefs.indexOf(viewRef); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   remove(index?: number): void { | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |     this.detach(index); | 
					
						
							|  |  |  |     // TODO(ml): proper destroy of the ViewRef, i.e. recursively destroy the LviewNode and its
 | 
					
						
							|  |  |  |     // children, delete DOM nodes and QueryList, trigger hooks (onDestroy), destroy the renderer,
 | 
					
						
							|  |  |  |     // detach projected nodes
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   detach(index?: number): viewEngine_ViewRef|null { | 
					
						
							|  |  |  |     const adjustedIdx = this._adjustIndex(index, -1); | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     removeView(this._lContainerNode, adjustedIdx); | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |     return this._viewRefs.splice(adjustedIdx, 1)[0] || null; | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |   private _adjustIndex(index?: number, shift: number = 0) { | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     if (index == null) { | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |       return this._lContainerNode.data.views.length + shift; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ngDevMode) { | 
					
						
							|  |  |  |       assertGreaterThan(index, -1, 'index must be positive'); | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |       // +1 because it's legal to insert at the end.
 | 
					
						
							| 
									
										
										
										
											2018-04-05 10:29:47 +02:00
										 |  |  |       assertLessThan(index, this._lContainerNode.data.views.length + 1 + shift, 'index'); | 
					
						
							| 
									
										
										
										
											2018-03-15 17:33:35 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return index; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											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> { | 
					
						
							| 
									
										
										
										
											2018-03-20 19:06:49 -07:00
										 |  |  |   ngDevMode && assertNodeType(di.node, LNodeType.Container); | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |   const data = (di.node as LContainerNode).data; | 
					
						
							| 
									
										
										
										
											2018-04-11 14:15:30 +02:00
										 |  |  |   const tView = di.node.view.tView; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |   return di.templateRef || (di.templateRef = new TemplateRef<any>( | 
					
						
							| 
									
										
										
										
											2018-04-11 14:15:30 +02:00
										 |  |  |                                 getOrCreateElementRef(di), data.template !, getRenderer(), | 
					
						
							|  |  |  |                                 tView.directiveRegistry, tView.pipeRegistry)); | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TemplateRef<T> implements viewEngine_TemplateRef<T> { | 
					
						
							|  |  |  |   readonly elementRef: viewEngine_ElementRef; | 
					
						
							|  |  |  |   private _template: ComponentTemplate<T>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor( | 
					
						
							|  |  |  |       elementRef: viewEngine_ElementRef, template: ComponentTemplate<T>, | 
					
						
							| 
									
										
										
										
											2018-04-11 14:15:30 +02:00
										 |  |  |       private _renderer: Renderer3, private _directives: DirectiveDefList|null, | 
					
						
							|  |  |  |       private _pipes: PipeDefList|null) { | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |     this.elementRef = elementRef; | 
					
						
							|  |  |  |     this._template = template; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   createEmbeddedView(context: T): viewEngine_EmbeddedViewRef<T> { | 
					
						
							| 
									
										
										
										
											2018-04-11 14:15:30 +02:00
										 |  |  |     const viewNode = renderEmbeddedTemplate( | 
					
						
							|  |  |  |         null, this._template, context, this._renderer, this._directives, this._pipes); | 
					
						
							| 
									
										
										
										
											2018-02-26 16:58:15 -08:00
										 |  |  |     return addDestroyable(new EmbeddedViewRef(viewNode, this._template, context)); | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | } |