| 
									
										
										
										
											2018-09-24 14:51:54 -07: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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | import {Injector} from '../di/injector'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import {assertDefined} from './assert'; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | import {discoverDirectives, discoverLocalRefs, getContext, isComponentInstance} from './context_discovery'; | 
					
						
							| 
									
										
										
										
											2018-10-02 14:42:34 +02:00
										 |  |  | import {NodeInjector} from './di'; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | import {LContext} from './interfaces/context'; | 
					
						
							|  |  |  | import {TElementNode, TNode, TNodeFlags} from './interfaces/node'; | 
					
						
							| 
									
										
										
										
											2018-10-02 14:42:34 +02:00
										 |  |  | import {CONTEXT, FLAGS, LViewData, LViewFlags, PARENT, RootContext, TVIEW} from './interfaces/view'; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | import {getComponentViewByIndex, readPatchedLViewData} from './util'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * NOTE: The following functions might not be ideal for core usage in Angular... | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Each function below is designed | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the component instance associated with the target. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * If a DOM is used then it will return the component that | 
					
						
							|  |  |  |  *    owns the view where the element is situated. | 
					
						
							|  |  |  |  * If a component instance is used then it will return the | 
					
						
							|  |  |  |  *    instance of the parent component depending on where | 
					
						
							|  |  |  |  *    the component instance is exists in a template. | 
					
						
							|  |  |  |  * If a directive instance is used then it will return the | 
					
						
							|  |  |  |  *    component that contains that directive in it's template. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getComponent<T = {}>(target: {}): T|null { | 
					
						
							|  |  |  |   const context = loadContext(target) !; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (context.component === undefined) { | 
					
						
							|  |  |  |     let lViewData = context.lViewData; | 
					
						
							|  |  |  |     while (lViewData) { | 
					
						
							|  |  |  |       const ctx = lViewData ![CONTEXT] !as{}; | 
					
						
							|  |  |  |       if (ctx && isComponentInstance(ctx)) { | 
					
						
							|  |  |  |         context.component = ctx; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       lViewData = lViewData ![PARENT] !; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (context.component === undefined) { | 
					
						
							|  |  |  |       context.component = null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return context.component as T; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the host component instance associated with the target. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This will only return a component instance of the DOM node | 
					
						
							|  |  |  |  * contains an instance of a component on it. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getHostComponent<T = {}>(target: {}): T|null { | 
					
						
							|  |  |  |   const context = loadContext(target); | 
					
						
							| 
									
										
										
										
											2018-10-04 20:40:39 -07:00
										 |  |  |   const tNode = context.lViewData[TVIEW].data[context.nodeIndex] as TNode; | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  |   if (tNode.flags & TNodeFlags.isComponent) { | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |     const componentView = getComponentViewByIndex(context.nodeIndex, context.lViewData); | 
					
						
							|  |  |  |     return componentView[CONTEXT] as any as T; | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  |   } | 
					
						
							|  |  |  |   return null; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the `RootContext` instance that is associated with | 
					
						
							|  |  |  |  * the application where the target is situated. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getRootContext(target: {}): RootContext { | 
					
						
							|  |  |  |   const context = loadContext(target) !; | 
					
						
							|  |  |  |   const rootLViewData = getRootView(context.lViewData); | 
					
						
							|  |  |  |   return rootLViewData[CONTEXT] as RootContext; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns a list of all the components in the application | 
					
						
							|  |  |  |  * that are have been bootstrapped. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getRootComponents(target: {}): any[] { | 
					
						
							|  |  |  |   return [...getRootContext(target).components]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the injector instance that is associated with | 
					
						
							|  |  |  |  * the element, component or directive. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-10-02 14:42:34 +02:00
										 |  |  | export function getInjector(target: {}): Injector { | 
					
						
							|  |  |  |   const context = loadContext(target); | 
					
						
							| 
									
										
										
										
											2018-10-04 20:40:39 -07:00
										 |  |  |   const tNode = context.lViewData[TVIEW].data[context.nodeIndex] as TElementNode; | 
					
						
							| 
									
										
										
										
											2018-10-02 14:42:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return new NodeInjector(tNode, context.lViewData); | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns a list of all the directives that are associated | 
					
						
							|  |  |  |  * with the underlying target element. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getDirectives(target: {}): Array<{}> { | 
					
						
							|  |  |  |   const context = loadContext(target) !; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (context.directives === undefined) { | 
					
						
							| 
									
										
										
										
											2018-10-05 21:23:41 -07:00
										 |  |  |     context.directives = discoverDirectives(context.nodeIndex, context.lViewData, false); | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return context.directives || []; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 10:30:29 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Returns LContext associated with a target passed as an argument. | 
					
						
							|  |  |  |  * Throws if a given target doesn't have associated LContext. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function loadContext(target: {}): LContext { | 
					
						
							| 
									
										
										
										
											2018-09-24 14:51:54 -07:00
										 |  |  |   const context = getContext(target); | 
					
						
							|  |  |  |   if (!context) { | 
					
						
							|  |  |  |     throw new Error( | 
					
						
							|  |  |  |         ngDevMode ? 'Unable to find the given context data for the given target' : | 
					
						
							|  |  |  |                     'Invalid ng target'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Retrieve the root view from any component by walking the parent `LViewData` until | 
					
						
							|  |  |  |  * reaching the root `LViewData`. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param componentOrView any component or view | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getRootView(componentOrView: LViewData | {}): LViewData { | 
					
						
							|  |  |  |   let lViewData: LViewData; | 
					
						
							|  |  |  |   if (Array.isArray(componentOrView)) { | 
					
						
							|  |  |  |     ngDevMode && assertDefined(componentOrView, 'lViewData'); | 
					
						
							|  |  |  |     lViewData = componentOrView as LViewData; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     ngDevMode && assertDefined(componentOrView, 'component'); | 
					
						
							|  |  |  |     lViewData = readPatchedLViewData(componentOrView) !; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   while (lViewData && !(lViewData[FLAGS] & LViewFlags.IsRoot)) { | 
					
						
							|  |  |  |     lViewData = lViewData[PARENT] !; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return lViewData; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-26 15:21:58 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  *  Retrieve map of local references (local reference name => element or directive instance). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getLocalRefs(target: {}): {[key: string]: any} { | 
					
						
							|  |  |  |   const context = loadContext(target) !; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (context.localRefs === undefined) { | 
					
						
							| 
									
										
										
										
											2018-10-04 20:40:39 -07:00
										 |  |  |     context.localRefs = discoverLocalRefs(context.lViewData, context.nodeIndex); | 
					
						
							| 
									
										
										
										
											2018-09-26 15:21:58 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return context.localRefs || {}; | 
					
						
							| 
									
										
										
										
											2018-10-04 20:40:39 -07:00
										 |  |  | } |