| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-09-13 16:07:23 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-26 12:27:40 +02:00
										 |  |  | import {global} from '../util'; | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-28 15:54:38 -08:00
										 |  |  | import {assertDataInRange, assertDefined, assertGreaterThan, assertLessThan} from './assert'; | 
					
						
							| 
									
										
										
										
											2018-12-13 15:51:47 -08:00
										 |  |  | import {ACTIVE_INDEX, LCONTAINER_LENGTH, LContainer} from './interfaces/container'; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | import {LContext, MONKEY_PATCH_KEY_NAME} from './interfaces/context'; | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  | import {ComponentDef, DirectiveDef} from './interfaces/definition'; | 
					
						
							|  |  |  | import {NO_PARENT_INJECTOR, RelativeInjectorLocation, RelativeInjectorLocationFlags} from './interfaces/injector'; | 
					
						
							| 
									
										
										
										
											2018-12-13 11:14:33 +01:00
										 |  |  | import {TContainerNode, TElementNode, TNode, TNodeFlags, TNodeType} from './interfaces/node'; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | import {RComment, RElement, RText} from './interfaces/renderer'; | 
					
						
							| 
									
										
										
										
											2018-10-11 13:13:57 -07:00
										 |  |  | import {StylingContext} from './interfaces/styling'; | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | import {CONTEXT, DECLARATION_VIEW, FLAGS, HEADER_OFFSET, HOST, HOST_NODE, LView, LViewFlags, PARENT, RootContext, TData, TVIEW, TView} from './interfaces/view'; | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2018-09-13 16:07:23 -07:00
										 |  |  |  * Returns whether the values are different from a change detection stand point. | 
					
						
							| 
									
										
										
										
											2018-08-01 10:57:16 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Constraints are relaxed in checkNoChanges mode. See `devModeEqual` for details. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function isDifferent(a: any, b: any): boolean { | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  |   // NaN is the only value that is not equal to itself so the first
 | 
					
						
							|  |  |  |   // test checks if both a and b are not NaN
 | 
					
						
							|  |  |  |   return !(a !== a && b !== b) && a !== b; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function stringify(value: any): string { | 
					
						
							|  |  |  |   if (typeof value == 'function') return value.name || value; | 
					
						
							|  |  |  |   if (typeof value == 'string') return value; | 
					
						
							|  |  |  |   if (value == null) return ''; | 
					
						
							| 
									
										
										
										
											2018-12-13 18:16:03 +01:00
										 |  |  |   if (typeof value == 'object' && typeof value.type == 'function') | 
					
						
							|  |  |  |     return value.type.name || value.type; | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  |   return '' + value; | 
					
						
							| 
									
										
										
										
											2017-12-13 14:28:36 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 17:55:55 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Flattens an array in non-recursive way. Input arrays are not modified. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function flatten(list: any[]): any[] { | 
					
						
							|  |  |  |   const result: any[] = []; | 
					
						
							|  |  |  |   let i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (i < list.length) { | 
					
						
							|  |  |  |     const item = list[i]; | 
					
						
							|  |  |  |     if (Array.isArray(item)) { | 
					
						
							|  |  |  |       if (item.length > 0) { | 
					
						
							|  |  |  |         list = item.concat(list.slice(i + 1)); | 
					
						
							|  |  |  |         i = 0; | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         i++; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       result.push(item); | 
					
						
							|  |  |  |       i++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-11 09:56:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | /** Retrieves a value from any `LView` or `TData`. */ | 
					
						
							|  |  |  | export function loadInternal<T>(view: LView | TData, index: number): T { | 
					
						
							|  |  |  |   ngDevMode && assertDataInRange(view, index + HEADER_OFFSET); | 
					
						
							|  |  |  |   return view[index + HEADER_OFFSET]; | 
					
						
							| 
									
										
										
										
											2018-07-11 09:56:47 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * Takes the value of a slot in `LView` and returns the element node. | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Normally, element nodes are stored flat, but if the node has styles/classes on it, | 
					
						
							|  |  |  |  * it might be wrapped in a styling context. Or if that node has a directive that injects | 
					
						
							|  |  |  |  * ViewContainerRef, it may be wrapped in an LContainer. Or if that node is a component, | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * it will be wrapped in LView. It could even have all three, so we keep looping | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |  * until we find something that isn't an array. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * @param value The initial value in `LView` | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function readElementValue(value: RElement | StylingContext | LContainer | LView): RElement { | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   while (Array.isArray(value)) { | 
					
						
							|  |  |  |     value = value[HOST] as any; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 18:49:00 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Retrieves an element value from the provided `viewData`, by unwrapping | 
					
						
							|  |  |  |  * from any containers, component views, or style contexts. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getNativeByIndex(index: number, lView: LView): RElement { | 
					
						
							|  |  |  |   return readElementValue(lView[index + HEADER_OFFSET]); | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getNativeByTNode(tNode: TNode, hostView: LView): RElement|RText|RComment { | 
					
						
							| 
									
										
										
										
											2018-09-17 14:32:45 -07:00
										 |  |  |   return readElementValue(hostView[tNode.index]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getTNode(index: number, view: LView): TNode { | 
					
						
							| 
									
										
										
										
											2018-11-28 15:54:38 -08:00
										 |  |  |   ngDevMode && assertGreaterThan(index, -1, 'wrong index for TNode'); | 
					
						
							|  |  |  |   ngDevMode && assertLessThan(index, view[TVIEW].data.length, 'wrong index for TNode'); | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   return view[TVIEW].data[index + HEADER_OFFSET] as TNode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getComponentViewByIndex(nodeIndex: number, hostView: LView): LView { | 
					
						
							|  |  |  |   // Could be an LView or an LContainer. If LContainer, unwrap to find LView.
 | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   const slotValue = hostView[nodeIndex]; | 
					
						
							|  |  |  |   return slotValue.length >= HEADER_OFFSET ? slotValue : slotValue[HOST]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 16:07:23 -07:00
										 |  |  | export function isContentQueryHost(tNode: TNode): boolean { | 
					
						
							|  |  |  |   return (tNode.flags & TNodeFlags.hasContentQuery) !== 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function isComponent(tNode: TNode): boolean { | 
					
						
							|  |  |  |   return (tNode.flags & TNodeFlags.isComponent) === TNodeFlags.isComponent; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-11 13:13:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  | export function isComponentDef<T>(def: DirectiveDef<T>): def is ComponentDef<T> { | 
					
						
							|  |  |  |   return (def as ComponentDef<T>).template !== null; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 18:49:00 -07:00
										 |  |  | export function isLContainer(value: RElement | RComment | LContainer | StylingContext): boolean { | 
					
						
							| 
									
										
										
										
											2018-10-11 13:13:57 -07:00
										 |  |  |   // Styling contexts are also arrays, but their first index contains an element node
 | 
					
						
							| 
									
										
										
										
											2018-12-13 15:51:47 -08:00
										 |  |  |   return Array.isArray(value) && value.length === LCONTAINER_LENGTH; | 
					
						
							| 
									
										
										
										
											2018-10-11 13:13:57 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function isRootView(target: LView): boolean { | 
					
						
							| 
									
										
										
										
											2018-10-17 11:01:35 +02:00
										 |  |  |   return (target[FLAGS] & LViewFlags.IsRoot) !== 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * Retrieve the root view from any component by walking the parent `LView` until | 
					
						
							|  |  |  |  * reaching the root `LView`. | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @param component any component | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getRootView(target: LView | {}): LView { | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  |   ngDevMode && assertDefined(target, 'component'); | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |   let lView = Array.isArray(target) ? (target as LView) : readPatchedLView(target) !; | 
					
						
							|  |  |  |   while (lView && !(lView[FLAGS] & LViewFlags.IsRoot)) { | 
					
						
							|  |  |  |     lView = lView[PARENT] !; | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |   return lView; | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getRootContext(viewOrComponent: LView | {}): RootContext { | 
					
						
							| 
									
										
										
										
											2018-10-26 12:27:40 +02:00
										 |  |  |   const rootView = getRootView(viewOrComponent); | 
					
						
							|  |  |  |   ngDevMode && | 
					
						
							|  |  |  |       assertDefined(rootView[CONTEXT], 'RootView has no context. Perhaps it is disconnected?'); | 
					
						
							|  |  |  |   return rootView[CONTEXT] as RootContext; | 
					
						
							| 
									
										
										
										
											2018-08-28 16:49:52 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the monkey-patch value data present on the target (which could be | 
					
						
							|  |  |  |  * a component, directive or a DOM node). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function readPatchedData(target: any): LView|LContext|null { | 
					
						
							| 
									
										
										
										
											2018-11-28 15:54:38 -08:00
										 |  |  |   ngDevMode && assertDefined(target, 'Target expected'); | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   return target[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function readPatchedLView(target: any): LView|null { | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   const value = readPatchedData(target); | 
					
						
							|  |  |  |   if (value) { | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |     return Array.isArray(value) ? value : (value as LContext).lView; | 
					
						
							| 
									
										
										
										
											2018-10-12 15:02:54 -07:00
										 |  |  |   } | 
					
						
							|  |  |  |   return null; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function hasParentInjector(parentLocation: RelativeInjectorLocation): boolean { | 
					
						
							|  |  |  |   return parentLocation !== NO_PARENT_INJECTOR; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function getParentInjectorIndex(parentLocation: RelativeInjectorLocation): number { | 
					
						
							|  |  |  |   return (parentLocation as any as number) & RelativeInjectorLocationFlags.InjectorIndexMask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function getParentInjectorViewOffset(parentLocation: RelativeInjectorLocation): number { | 
					
						
							|  |  |  |   return (parentLocation as any as number) >> RelativeInjectorLocationFlags.ViewOffsetShift; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Unwraps a parent injector location number to find the view offset from the current injector, | 
					
						
							|  |  |  |  * then walks up the declaration view tree until the view is found that contains the parent | 
					
						
							|  |  |  |  * injector. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param location The location of the parent injector, which contains the view offset | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * @param startView The LView instance from which to start walking up the view tree | 
					
						
							|  |  |  |  * @returns The LView instance that contains the parent injector | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  | export function getParentInjectorView(location: RelativeInjectorLocation, startView: LView): LView { | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |   let viewOffset = getParentInjectorViewOffset(location); | 
					
						
							|  |  |  |   let parentView = startView; | 
					
						
							|  |  |  |   // For most cases, the parent injector can be found on the host node (e.g. for component
 | 
					
						
							|  |  |  |   // or container), but we must keep the loop here to support the rarer case of deeply nested
 | 
					
						
							|  |  |  |   // <ng-template> tags or inline views, where the parent injector might live many views
 | 
					
						
							|  |  |  |   // above the child injector.
 | 
					
						
							|  |  |  |   while (viewOffset > 0) { | 
					
						
							|  |  |  |     parentView = parentView[DECLARATION_VIEW] !; | 
					
						
							|  |  |  |     viewOffset--; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return parentView; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Unwraps a parent injector location number to find the view offset from the current injector, | 
					
						
							|  |  |  |  * then walks up the declaration view tree until the TNode of the parent injector is found. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param location The location of the parent injector, which contains the view offset | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |  * @param startView The LView instance from which to start walking up the view tree | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |  * @param startTNode The TNode instance of the starting element | 
					
						
							|  |  |  |  * @returns The TNode of the parent injector | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function getParentInjectorTNode( | 
					
						
							| 
									
										
										
										
											2018-11-21 21:14:06 -08:00
										 |  |  |     location: RelativeInjectorLocation, startView: LView, startTNode: TNode): TElementNode| | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |     TContainerNode|null { | 
					
						
							|  |  |  |   if (startTNode.parent && startTNode.parent.injectorIndex !== -1) { | 
					
						
							|  |  |  |     // view offset is 0
 | 
					
						
							|  |  |  |     const injectorIndex = startTNode.parent.injectorIndex; | 
					
						
							|  |  |  |     let parentTNode = startTNode.parent; | 
					
						
							|  |  |  |     while (parentTNode.parent != null && injectorIndex == parentTNode.injectorIndex) { | 
					
						
							|  |  |  |       parentTNode = parentTNode.parent; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return parentTNode; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let viewOffset = getParentInjectorViewOffset(location); | 
					
						
							| 
									
										
										
										
											2018-11-14 17:04:22 +01:00
										 |  |  |   // view offset is 1
 | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |   let parentView = startView; | 
					
						
							|  |  |  |   let parentTNode = startView[HOST_NODE] as TElementNode; | 
					
						
							| 
									
										
										
										
											2018-11-14 17:04:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // view offset is superior to 1
 | 
					
						
							|  |  |  |   while (viewOffset > 1) { | 
					
						
							| 
									
										
										
										
											2018-10-18 09:23:18 +02:00
										 |  |  |     parentView = parentView[DECLARATION_VIEW] !; | 
					
						
							|  |  |  |     parentTNode = parentView[HOST_NODE] as TElementNode; | 
					
						
							|  |  |  |     viewOffset--; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return parentTNode; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-26 12:27:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export const defaultScheduler = | 
					
						
							|  |  |  |     (typeof requestAnimationFrame !== 'undefined' && requestAnimationFrame ||  // browser only
 | 
					
						
							|  |  |  |      setTimeout                                                                // everything else
 | 
					
						
							| 
									
										
										
										
											2018-11-13 09:36:30 +01:00
										 |  |  |      ).bind(global); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Equivalent to ES6 spread, add each item to an array. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param items The items to add | 
					
						
							|  |  |  |  * @param arr The array to which you want to add the items | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function addAllToArray(items: any[], arr: any[]) { | 
					
						
							|  |  |  |   for (let i = 0; i < items.length; i++) { | 
					
						
							|  |  |  |     arr.push(items[i]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-13 11:14:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Given a current view, finds the nearest component's host (LElement). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param lView LView for which we want a host element node | 
					
						
							|  |  |  |  * @returns The host node | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-12-20 13:30:08 -08:00
										 |  |  | export function findComponentView(lView: LView): LView { | 
					
						
							| 
									
										
										
										
											2018-12-13 11:14:33 +01:00
										 |  |  |   let rootTNode = lView[HOST_NODE]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (rootTNode && rootTNode.type === TNodeType.View) { | 
					
						
							| 
									
										
										
										
											2018-12-20 13:30:08 -08:00
										 |  |  |     ngDevMode && assertDefined(lView[DECLARATION_VIEW], 'lView[DECLARATION_VIEW]'); | 
					
						
							|  |  |  |     lView = lView[DECLARATION_VIEW] !; | 
					
						
							| 
									
										
										
										
											2018-12-13 11:14:33 +01:00
										 |  |  |     rootTNode = lView[HOST_NODE]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return lView; | 
					
						
							|  |  |  | } |