| 
									
										
										
										
											2017-12-20 10:47:22 -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-01-17 10:09:05 -08:00
										 |  |  | import {SimpleChange} from '../change_detection/change_detection_util'; | 
					
						
							| 
									
										
										
										
											2018-02-23 13:17:20 -08:00
										 |  |  | import {ChangeDetectionStrategy} from '../change_detection/constants'; | 
					
						
							| 
									
										
										
										
											2018-01-27 13:07:03 -08:00
										 |  |  | import {PipeTransform} from '../change_detection/pipe_transform'; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | import {Provider} from '../core'; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | import {OnChanges, SimpleChanges} from '../metadata/lifecycle_hooks'; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | import {RendererType2} from '../render/api'; | 
					
						
							|  |  |  | import {Type} from '../type'; | 
					
						
							|  |  |  | import {resolveRendererType2} from '../view/util'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 09:45:40 -08:00
										 |  |  | import {diPublic} from './di'; | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  | import {ComponentDef, ComponentDefFeature, ComponentTemplate, ComponentType, DirectiveDef, DirectiveDefFeature, DirectiveDefListOrFactory, DirectiveType, DirectiveTypesOrFactory, PipeDef, PipeType, PipeTypesOrFactory} from './interfaces/definition'; | 
					
						
							| 
									
										
										
										
											2018-03-29 16:41:45 -07:00
										 |  |  | import {CssSelectorList, SelectorFlags} from './interfaces/projection'; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Create a component definition object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * # Example | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * class MyDirective { | 
					
						
							|  |  |  |  *   // Generated by Angular Template Compiler
 | 
					
						
							|  |  |  |  *   // [Symbol] syntax will not be supported by TypeScript until v2.7
 | 
					
						
							| 
									
										
										
										
											2018-01-27 13:07:03 -08:00
										 |  |  |  *   static ngComponentDef = defineComponent({ | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |  *     ... | 
					
						
							|  |  |  |  *   }); | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | export function defineComponent<T>(componentDefinition: { | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Directive type, needed to configure the injector. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   type: Type<T>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 16:41:45 -07:00
										 |  |  |   /** The selectors that will be used to match nodes to this component. */ | 
					
						
							|  |  |  |   selectors: CssSelectorList; | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Factory method used to create an instance of directive. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   factory: () => T | ({0: T} & any[]); /* trying to say T | [T, ...any] */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Static attributes to set on host element. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Even indices: attribute name | 
					
						
							|  |  |  |    * Odd indices: attribute value | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   attributes?: string[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of input names. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The format is in: `{[actualPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: `{[minifiedPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified and non-minified names | 
					
						
							|  |  |  |    * of properties. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   inputs?: {[P in keyof T]?: string}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of output names. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The format is in: `{[actualPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: `{[minifiedPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified and non-minified names | 
					
						
							|  |  |  |    * of properties. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   outputs?: {[P in keyof T]?: string}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Function executed by the parent template to allow child directive to apply host bindings. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   hostBindings?: (directiveIndex: number, elementIndex: number) => void; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Defines the name that can be used in the template to assign this directive to a variable. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * See: {@link Directive.exportAs} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   exportAs?: string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Template function use for rendering DOM. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This function has following structure. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * function Template<T>(ctx:T, creationMode: boolean) { | 
					
						
							|  |  |  |    *   if (creationMode) { | 
					
						
							|  |  |  |    *     // Contains creation mode instructions.
 | 
					
						
							|  |  |  |    *   } | 
					
						
							|  |  |  |    *   // Contains binding update instructions
 | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Common instructions are: | 
					
						
							|  |  |  |    * Creation mode instructions: | 
					
						
							|  |  |  |    *  - `elementStart`, `elementEnd` | 
					
						
							|  |  |  |    *  - `text` | 
					
						
							|  |  |  |    *  - `container` | 
					
						
							|  |  |  |    *  - `listener` | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Binding update instructions: | 
					
						
							|  |  |  |    * - `bind` | 
					
						
							|  |  |  |    * - `elementAttribute` | 
					
						
							|  |  |  |    * - `elementProperty` | 
					
						
							|  |  |  |    * - `elementClass` | 
					
						
							|  |  |  |    * - `elementStyle` | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   template: ComponentTemplate<T>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A list of optional features to apply. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * See: {@link NgOnChangesFeature}, {@link PublicFeature} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   features?: ComponentDefFeature[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   rendererType?: RendererType2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   changeDetection?: ChangeDetectionStrategy; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Defines the set of injectable objects that are visible to a Directive and its light DOM | 
					
						
							|  |  |  |    * children. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   providers?: Provider[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Defines the set of injectable objects that are visible to its view DOM children. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   viewProviders?: Provider[]; | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Registry of directives and components that may be found in this component's view. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  |    * The property is either an array of `DirectiveDef`s or a function which returns the array of | 
					
						
							|  |  |  |    * `DirectiveDef`s. The function is necessary to be able to support forward declarations. | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |   directives?: DirectiveTypesOrFactory | null; | 
					
						
							| 
									
										
										
										
											2018-03-27 15:53:48 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Registry of pipes that may be found in this component's view. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The property is either an array of `PipeDefs`s or a function which returns the array of | 
					
						
							|  |  |  |    * `PipeDefs`s. The function is necessary to be able to support forward declarations. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |   pipes?: PipeTypesOrFactory | null; | 
					
						
							| 
									
										
										
										
											2018-04-14 09:18:38 -07:00
										 |  |  | }): never { | 
					
						
							| 
									
										
										
										
											2018-01-23 18:39:09 -08:00
										 |  |  |   const type = componentDefinition.type; | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |   const pipeTypes = componentDefinition.pipes !; | 
					
						
							|  |  |  |   const directiveTypes = componentDefinition.directives !; | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  |   const def = <ComponentDef<any>>{ | 
					
						
							| 
									
										
										
										
											2018-01-23 18:39:09 -08:00
										 |  |  |     type: type, | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |     diPublic: null, | 
					
						
							| 
									
										
										
										
											2018-03-16 16:42:13 -07:00
										 |  |  |     factory: componentDefinition.factory, | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |     template: componentDefinition.template || null !, | 
					
						
							| 
									
										
										
										
											2018-03-16 16:42:13 -07:00
										 |  |  |     hostBindings: componentDefinition.hostBindings || null, | 
					
						
							| 
									
										
										
										
											2018-02-16 12:09:47 -08:00
										 |  |  |     attributes: componentDefinition.attributes || null, | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |     inputs: invertObject(componentDefinition.inputs), | 
					
						
							|  |  |  |     outputs: invertObject(componentDefinition.outputs), | 
					
						
							|  |  |  |     rendererType: resolveRendererType2(componentDefinition.rendererType) || null, | 
					
						
							| 
									
										
										
										
											2018-01-08 21:57:50 -08:00
										 |  |  |     exportAs: componentDefinition.exportAs, | 
					
						
							| 
									
										
										
										
											2018-01-23 18:39:09 -08:00
										 |  |  |     onInit: type.prototype.ngOnInit || null, | 
					
						
							|  |  |  |     doCheck: type.prototype.ngDoCheck || null, | 
					
						
							|  |  |  |     afterContentInit: type.prototype.ngAfterContentInit || null, | 
					
						
							|  |  |  |     afterContentChecked: type.prototype.ngAfterContentChecked || null, | 
					
						
							|  |  |  |     afterViewInit: type.prototype.ngAfterViewInit || null, | 
					
						
							|  |  |  |     afterViewChecked: type.prototype.ngAfterViewChecked || null, | 
					
						
							| 
									
										
										
										
											2018-02-23 13:17:20 -08:00
										 |  |  |     onDestroy: type.prototype.ngOnDestroy || null, | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  |     onPush: componentDefinition.changeDetection === ChangeDetectionStrategy.OnPush, | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |     directiveDefs: directiveTypes ? | 
					
						
							|  |  |  |         () => (typeof directiveTypes === 'function' ? directiveTypes() : directiveTypes) | 
					
						
							|  |  |  |                   .map(extractDirectiveDef) : | 
					
						
							|  |  |  |         null, | 
					
						
							|  |  |  |     pipeDefs: pipeTypes ? | 
					
						
							|  |  |  |         () => (typeof pipeTypes === 'function' ? pipeTypes() : pipeTypes).map(extractPipeDef) : | 
					
						
							|  |  |  |         null, | 
					
						
							| 
									
										
										
										
											2018-03-29 16:41:45 -07:00
										 |  |  |     selectors: componentDefinition.selectors | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  |   const feature = componentDefinition.features; | 
					
						
							|  |  |  |   feature && feature.forEach((fn) => fn(def)); | 
					
						
							| 
									
										
										
										
											2018-04-14 09:18:38 -07:00
										 |  |  |   return def as never; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  | export function extractDirectiveDef(type: DirectiveType<any>& ComponentType<any>): | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  |     DirectiveDef<any>|ComponentDef<any> { | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |   const def = type.ngComponentDef || type.ngDirectiveDef; | 
					
						
							|  |  |  |   if (ngDevMode && !def) { | 
					
						
							|  |  |  |     throw new Error(`'${type.name}' is neither 'ComponentType' or 'DirectiveType'.`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return def; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  | export function extractPipeDef(type: PipeType<any>): PipeDef<any> { | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |   const def = type.ngPipeDef; | 
					
						
							|  |  |  |   if (ngDevMode && !def) { | 
					
						
							|  |  |  |     throw new Error(`'${type.name}' is not a 'PipeType'.`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return def; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | const PRIVATE_PREFIX = '__ngOnChanges_'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type OnChangesExpando = OnChanges & { | 
					
						
							|  |  |  |   __ngOnChanges_: SimpleChanges|null|undefined; | 
					
						
							|  |  |  |   [key: string]: any; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Creates an NgOnChangesFeature function for a component's features list. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * It accepts an optional map of minified input property names to original property names, | 
					
						
							|  |  |  |  * if any input properties have a public alias. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The NgOnChangesFeature function that is returned decorates a component with support for | 
					
						
							|  |  |  |  * the ngOnChanges lifecycle hook, so it should be included in any component that implements | 
					
						
							|  |  |  |  * that hook. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Example usage: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * static ngComponentDef = defineComponent({ | 
					
						
							|  |  |  |  *   ... | 
					
						
							|  |  |  |  *   inputs: {name: 'publicName'}, | 
					
						
							|  |  |  |  *   features: [NgOnChangesFeature({name: 'name'})] | 
					
						
							|  |  |  |  * }); | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param inputPropertyNames Map of input property names, if they are aliased | 
					
						
							|  |  |  |  * @returns DirectiveDefFeature | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function NgOnChangesFeature(inputPropertyNames?: {[key: string]: string}): | 
					
						
							|  |  |  |     DirectiveDefFeature { | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  |   return function(definition: DirectiveDef<any>): void { | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |     const inputs = definition.inputs; | 
					
						
							|  |  |  |     const proto = definition.type.prototype; | 
					
						
							|  |  |  |     for (let pubKey in inputs) { | 
					
						
							|  |  |  |       const minKey = inputs[pubKey]; | 
					
						
							|  |  |  |       const propertyName = inputPropertyNames && inputPropertyNames[minKey] || pubKey; | 
					
						
							|  |  |  |       const privateMinKey = PRIVATE_PREFIX + minKey; | 
					
						
							| 
									
										
										
										
											2018-05-27 14:19:58 +02:00
										 |  |  |       const originalProperty = Object.getOwnPropertyDescriptor(proto, minKey); | 
					
						
							|  |  |  |       const getter = originalProperty && originalProperty.get; | 
					
						
							|  |  |  |       const setter = originalProperty && originalProperty.set; | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |       // create a getter and setter for property
 | 
					
						
							|  |  |  |       Object.defineProperty(proto, minKey, { | 
					
						
							| 
									
										
										
										
											2018-05-27 14:19:58 +02:00
										 |  |  |         get: getter || | 
					
						
							|  |  |  |             (setter ? undefined : function(this: OnChangesExpando) { return this[privateMinKey]; }), | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |         set: function(this: OnChangesExpando, value: any) { | 
					
						
							|  |  |  |           let simpleChanges = this[PRIVATE_PREFIX]; | 
					
						
							| 
									
										
										
										
											2018-05-27 14:19:58 +02:00
										 |  |  |           if (!simpleChanges) { | 
					
						
							|  |  |  |             // Place where we will store SimpleChanges if there is a change
 | 
					
						
							|  |  |  |             Object.defineProperty( | 
					
						
							|  |  |  |                 this, PRIVATE_PREFIX, {value: simpleChanges = {}, writable: true}); | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-05-27 14:19:58 +02:00
										 |  |  |           const isFirstChange = !this.hasOwnProperty(privateMinKey); | 
					
						
							| 
									
										
										
										
											2018-05-27 15:26:41 +02:00
										 |  |  |           const currentChange: SimpleChange|undefined = simpleChanges[propertyName]; | 
					
						
							|  |  |  |           if (currentChange) { | 
					
						
							|  |  |  |             currentChange.currentValue = value; | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             simpleChanges[propertyName] = | 
					
						
							|  |  |  |                 new SimpleChange(this[privateMinKey], value, isFirstChange); | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-05-27 14:19:58 +02:00
										 |  |  |           if (isFirstChange) { | 
					
						
							|  |  |  |             // Create a place where the actual value will be stored and make it non-enumerable
 | 
					
						
							|  |  |  |             Object.defineProperty(this, privateMinKey, {value, writable: true}); | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             this[privateMinKey] = value; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           setter && setter.call(this, value); | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-26 13:38:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |     // If an onInit hook is defined, it will need to wrap the ngOnChanges call
 | 
					
						
							|  |  |  |     // so the call order is changes-init-check in creation mode. In subsequent
 | 
					
						
							|  |  |  |     // change detection runs, only the check wrapper will be called.
 | 
					
						
							|  |  |  |     if (definition.onInit != null) { | 
					
						
							|  |  |  |       definition.onInit = onChangesWrapper(definition.onInit); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-01-26 13:38:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |     definition.doCheck = onChangesWrapper(definition.doCheck); | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2018-01-29 15:57:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   function onChangesWrapper(delegateHook: (() => void) | null) { | 
					
						
							|  |  |  |     return function(this: OnChangesExpando) { | 
					
						
							|  |  |  |       let simpleChanges = this[PRIVATE_PREFIX]; | 
					
						
							|  |  |  |       if (simpleChanges != null) { | 
					
						
							|  |  |  |         this.ngOnChanges(simpleChanges); | 
					
						
							|  |  |  |         this[PRIVATE_PREFIX] = null; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       delegateHook && delegateHook.apply(this); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-22 15:27:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-13 23:02:29 -07:00
										 |  |  | export function PublicFeature<T>(definition: DirectiveDef<T>) { | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   definition.diPublic = diPublic; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const EMPTY = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Swaps the keys and values of an object. */ | 
					
						
							|  |  |  | function invertObject(obj: any): any { | 
					
						
							|  |  |  |   if (obj == null) return EMPTY; | 
					
						
							|  |  |  |   const newObj: any = {}; | 
					
						
							|  |  |  |   for (let minifiedKey in obj) { | 
					
						
							|  |  |  |     newObj[obj[minifiedKey]] = minifiedKey; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return newObj; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Create a directive definition object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * # Example | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * class MyDirective { | 
					
						
							|  |  |  |  *   // Generated by Angular Template Compiler
 | 
					
						
							|  |  |  |  *   // [Symbol] syntax will not be supported by TypeScript until v2.7
 | 
					
						
							| 
									
										
										
										
											2018-01-27 13:07:03 -08:00
										 |  |  |  *   static ngDirectiveDef = defineDirective({ | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |  *     ... | 
					
						
							|  |  |  |  *   }); | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | export const defineDirective = defineComponent as any as<T>(directiveDefinition: { | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Directive type, needed to configure the injector. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   type: Type<T>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 16:41:45 -07:00
										 |  |  |   /** The selectors that will be used to match nodes to this directive. */ | 
					
						
							|  |  |  |   selectors: CssSelectorList; | 
					
						
							| 
									
										
										
										
											2018-03-25 21:32:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Factory method used to create an instance of directive. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   factory: () => T | ({0: T} & any[]); /* trying to say T | [T, ...any] */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Static attributes to set on host element. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Even indices: attribute name | 
					
						
							|  |  |  |    * Odd indices: attribute value | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   attributes?: string[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of input names. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The format is in: `{[actualPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: `{[minifiedPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified and non-minified names | 
					
						
							|  |  |  |    * of properties. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   inputs?: {[P in keyof T]?: string}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of output names. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The format is in: `{[actualPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: `{[minifiedPropertyName: string]:string}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified and non-minified names | 
					
						
							|  |  |  |    * of properties. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   outputs?: {[P in keyof T]?: string}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A list of optional features to apply. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * See: {@link NgOnChangesFeature}, {@link PublicFeature} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   features?: DirectiveDefFeature[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Function executed by the parent template to allow child directive to apply host bindings. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   hostBindings?: (directiveIndex: number, elementIndex: number) => void; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Defines the name that can be used in the template to assign this directive to a variable. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * See: {@link Directive.exportAs} | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   exportAs?: string; | 
					
						
							| 
									
										
										
										
											2018-04-14 09:18:38 -07:00
										 |  |  | }) => never; | 
					
						
							| 
									
										
										
										
											2018-01-27 13:07:03 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Create a pipe definition object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * # Example | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * class MyPipe implements PipeTransform { | 
					
						
							|  |  |  |  *   // Generated by Angular Template Compiler
 | 
					
						
							|  |  |  |  *   static ngPipeDef = definePipe({ | 
					
						
							|  |  |  |  *     ... | 
					
						
							|  |  |  |  *   }); | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2018-03-27 15:53:48 -07:00
										 |  |  |  * @param pipeDef Pipe definition generated by the compiler | 
					
						
							| 
									
										
										
										
											2018-01-27 13:07:03 -08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-03-27 15:53:48 -07:00
										 |  |  | export function definePipe<T>(pipeDef: { | 
					
						
							|  |  |  |   /** Name of the pipe. Used for matching pipes in template to pipe defs. */ | 
					
						
							|  |  |  |   name: string, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Pipe class reference. Needed to extract pipe lifecycle hooks. */ | 
					
						
							|  |  |  |   type: Type<T>, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** A factory for creating a pipe instance. */ | 
					
						
							|  |  |  |   factory: () => T, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Whether the pipe is pure. */ | 
					
						
							|  |  |  |   pure?: boolean | 
					
						
							| 
									
										
										
										
											2018-04-14 09:18:38 -07:00
										 |  |  | }): never { | 
					
						
							|  |  |  |   return (<PipeDef<T>>{ | 
					
						
							| 
									
										
										
										
											2018-03-27 15:53:48 -07:00
										 |  |  |     name: pipeDef.name, | 
					
						
							| 
									
										
										
										
											2018-05-13 18:29:45 +02:00
										 |  |  |     factory: pipeDef.factory, | 
					
						
							| 
									
										
										
										
											2018-03-27 15:53:48 -07:00
										 |  |  |     pure: pipeDef.pure !== false, | 
					
						
							|  |  |  |     onDestroy: pipeDef.type.prototype.ngOnDestroy || null | 
					
						
							| 
									
										
										
										
											2018-04-14 09:18:38 -07:00
										 |  |  |   }) as never; | 
					
						
							| 
									
										
										
										
											2018-02-16 12:09:47 -08:00
										 |  |  | } |