| 
									
										
										
										
											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-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'; | 
					
						
							| 
									
										
										
										
											2018-05-31 15:50:02 -07:00
										 |  |  | import {NgModuleDef, NgModuleDefInternal} from '../metadata/ng_module'; | 
					
						
							| 
									
										
										
										
											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-07-23 17:01:22 -07:00
										 |  |  | import {BaseDef, ComponentDefFeature, ComponentDefInternal, ComponentQuery, ComponentTemplate, ComponentType, DirectiveDefFeature, DirectiveDefInternal, DirectiveDefListOrFactory, DirectiveType, DirectiveTypesOrFactory, PipeDefInternal, 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. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-07-10 10:43:07 +02:00
										 |  |  |   factory: () => T; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Static attributes to set on host element. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Even indices: attribute name | 
					
						
							|  |  |  |    * Odd indices: attribute value | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   attributes?: string[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of input names. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * The format is in: `{[actualPropertyName: string]:(string|[string, string])}`. | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * Given: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * class MyComponent { | 
					
						
							|  |  |  |    *   @Input() | 
					
						
							|  |  |  |    *   publicInput1: string; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    *   @Input('publicInput2') | 
					
						
							|  |  |  |    *   declaredInput2: string; | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * is described as: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   publicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   declaredInput2: ['declaredInput2', 'publicInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   minifiedPublicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   minifiedDeclaredInput2: [ 'publicInput2', 'declaredInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified, public, and declared names | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * of properties. | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * | 
					
						
							|  |  |  |    * NOTE: | 
					
						
							|  |  |  |    *  - Because declared and public name are usually same we only generate the array | 
					
						
							|  |  |  |    *    `['declared', 'public']` format when they differ. | 
					
						
							|  |  |  |    *  - The reason why this API and `outputs` API is not the same is that `NgOnChanges` has | 
					
						
							|  |  |  |    *    inconsistent behavior in that it uses declared names rather than minified or public. For | 
					
						
							|  |  |  |    *    this reason `NgOnChanges` will be deprecated and removed in future version and this | 
					
						
							|  |  |  |    *    API will be simplified to be consistent with `output`. | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |   inputs?: {[P in keyof T]?: string | [string, string]}; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * 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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 10:43:07 +02:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Function to create instances of content queries associated with a given directive. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   contentQueries?: (() => void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Refreshes content queries associated with directives in a given view */ | 
					
						
							|  |  |  |   contentQueriesRefresh?: ((directiveIndex: number, queryIndex: number) => void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * 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>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 17:58:42 +02:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Additional set of instructions specific to view query processing. This could be seen as a | 
					
						
							|  |  |  |    * set of instruction to be inserted into the template function. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Query-related instructions need to be pulled out to a specific function as a timing of | 
					
						
							|  |  |  |    * execution is different as compared to all other instructions (after change detection hooks but | 
					
						
							|  |  |  |    * before view hooks). | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   viewQuery?: ComponentQuery<T>| null; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * 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-06-25 11:11:22 +02:00
										 |  |  |   const declaredInputs: {[key: string]: string} = {} as any; | 
					
						
							| 
									
										
										
										
											2018-05-31 15:50:02 -07:00
										 |  |  |   const def: ComponentDefInternal<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-07-10 10:43:07 +02:00
										 |  |  |     contentQueries: componentDefinition.contentQueries || null, | 
					
						
							|  |  |  |     contentQueriesRefresh: componentDefinition.contentQueriesRefresh || null, | 
					
						
							| 
									
										
										
										
											2018-02-16 12:09:47 -08:00
										 |  |  |     attributes: componentDefinition.attributes || null, | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |     inputs: invertObject(componentDefinition.inputs, declaredInputs), | 
					
						
							|  |  |  |     declaredInputs: declaredInputs, | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |     outputs: invertObject(componentDefinition.outputs), | 
					
						
							|  |  |  |     rendererType: resolveRendererType2(componentDefinition.rendererType) || null, | 
					
						
							| 
									
										
										
										
											2018-05-31 15:50:02 -07:00
										 |  |  |     exportAs: componentDefinition.exportAs || null, | 
					
						
							| 
									
										
										
										
											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-06-19 17:58:42 +02:00
										 |  |  |     selectors: componentDefinition.selectors, | 
					
						
							|  |  |  |     viewQuery: componentDefinition.viewQuery || null, | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |     features: componentDefinition.features || null, | 
					
						
							| 
									
										
										
										
											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-05-31 15:50:02 -07:00
										 |  |  |     DirectiveDefInternal<any>|ComponentDefInternal<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-06-26 10:43:06 -07:00
										 |  |  | export function extractPipeDef(type: PipeType<any>): PipeDefInternal<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-05-31 15:50:02 -07:00
										 |  |  | export function defineNgModule<T>(def: {type: T} & Partial<NgModuleDef<T, any, any, any>>): never { | 
					
						
							|  |  |  |   const res: NgModuleDefInternal<T> = { | 
					
						
							|  |  |  |     type: def.type, | 
					
						
							|  |  |  |     bootstrap: def.bootstrap || [], | 
					
						
							|  |  |  |     declarations: def.declarations || [], | 
					
						
							|  |  |  |     imports: def.imports || [], | 
					
						
							|  |  |  |     exports: def.exports || [], | 
					
						
							|  |  |  |     transitiveCompileScopes: null, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   return res as never; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  | const EMPTY = {}; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  * Inverts an inputs or outputs lookup such that the keys, which were the | 
					
						
							|  |  |  |  * minified keys, are part of the values, and the values are parsed so that | 
					
						
							|  |  |  |  * the publicName of the property is the new key | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  * e.g. for | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  * ```
 | 
					
						
							|  |  |  |  * class Comp { | 
					
						
							|  |  |  |  *   @Input() | 
					
						
							|  |  |  |  *   propName1: string; | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  *   @Input('publicName') | 
					
						
							|  |  |  |  *   propName2: number; | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * will be serialized as | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  * { | 
					
						
							|  |  |  |  *   a0: 'propName1', | 
					
						
							|  |  |  |  *   b1: ['publicName', 'propName2'], | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * becomes | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * { | 
					
						
							|  |  |  |  *  'propName1': 'a0', | 
					
						
							|  |  |  |  *  'publicName': 'b1' | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Optionally the function can take `secondary` which will result in: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * { | 
					
						
							|  |  |  |  *  'propName1': 'a0', | 
					
						
							|  |  |  |  *  'propName2': 'b1' | 
					
						
							|  |  |  |  * } | 
					
						
							| 
									
										
										
										
											2018-03-12 11:18:50 -07:00
										 |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | function invertObject(obj: any, secondary?: any): any { | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   if (obj == null) return EMPTY; | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |   const newLookup: any = {}; | 
					
						
							|  |  |  |   for (const minifiedKey in obj) { | 
					
						
							|  |  |  |     if (obj.hasOwnProperty(minifiedKey)) { | 
					
						
							|  |  |  |       let publicName = obj[minifiedKey]; | 
					
						
							|  |  |  |       let declaredName = publicName; | 
					
						
							|  |  |  |       if (Array.isArray(publicName)) { | 
					
						
							|  |  |  |         declaredName = publicName[1]; | 
					
						
							|  |  |  |         publicName = publicName[0]; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       newLookup[publicName] = minifiedKey; | 
					
						
							|  |  |  |       if (secondary) { | 
					
						
							|  |  |  |         (secondary[declaredName] = minifiedKey); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |   return newLookup; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-23 17:01:22 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Create a base definition | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * # Example | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * class ShouldBeInherited { | 
					
						
							|  |  |  |  *   static ngBaseDef = defineBase({ | 
					
						
							|  |  |  |  *      ... | 
					
						
							|  |  |  |  *   }) | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * @param baseDefinition The base definition parameters | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function defineBase<T>(baseDefinition: { | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * A map of input names. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The format is in: `{[actualPropertyName: string]:(string|[string, string])}`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Given: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * class MyComponent { | 
					
						
							|  |  |  |    *   @Input() | 
					
						
							|  |  |  |    *   publicInput1: string; | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    *   @Input('publicInput2') | 
					
						
							|  |  |  |    *   declaredInput2: string; | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * is described as: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   publicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   declaredInput2: ['declaredInput2', 'publicInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   minifiedPublicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   minifiedDeclaredInput2: [ 'declaredInput2', 'publicInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified, public, and declared names | 
					
						
							|  |  |  |    * of properties. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * NOTE: | 
					
						
							|  |  |  |    *  - Because declared and public name are usually same we only generate the array | 
					
						
							|  |  |  |    *    `['declared', 'public']` format when they differ. | 
					
						
							|  |  |  |    *  - The reason why this API and `outputs` API is not the same is that `NgOnChanges` has | 
					
						
							|  |  |  |    *    inconsistent behavior in that it uses declared names rather than minified or public. For | 
					
						
							|  |  |  |    *    this reason `NgOnChanges` will be deprecated and removed in future version and this | 
					
						
							|  |  |  |    *    API will be simplified to be consistent with `outputs`. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   inputs?: {[P in keyof T]?: string | [string, 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}; | 
					
						
							|  |  |  | }): BaseDef<T> { | 
					
						
							|  |  |  |   const declaredInputs: {[P in keyof T]: P} = {} as any; | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     inputs: invertObject(baseDefinition.inputs, declaredInputs), | 
					
						
							|  |  |  |     declaredInputs: declaredInputs, | 
					
						
							|  |  |  |     outputs: invertObject(baseDefinition.outputs), | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * The format is in: `{[actualPropertyName: string]:(string|[string, string])}`. | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * Given: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * class MyComponent { | 
					
						
							|  |  |  |    *   @Input() | 
					
						
							|  |  |  |    *   publicInput1: string; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    *   @Input('publicInput2') | 
					
						
							|  |  |  |    *   declaredInput2: string; | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * is described as: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   publicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   declaredInput2: ['declaredInput2', 'publicInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Which the minifier may translate to: | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * { | 
					
						
							|  |  |  |    *   minifiedPublicInput1: 'publicInput1', | 
					
						
							|  |  |  |    *   minifiedDeclaredInput2: [ 'publicInput2', 'declaredInput2'], | 
					
						
							|  |  |  |    * } | 
					
						
							|  |  |  |    * ```
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * This allows the render to re-construct the minified, public, and declared names | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    * of properties. | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * | 
					
						
							|  |  |  |    * NOTE: | 
					
						
							|  |  |  |    *  - Because declared and public name are usually same we only generate the array | 
					
						
							|  |  |  |    *    `['declared', 'public']` format when they differ. | 
					
						
							|  |  |  |    *  - The reason why this API and `outputs` API is not the same is that `NgOnChanges` has | 
					
						
							|  |  |  |    *    inconsistent behavior in that it uses declared names rather than minified or public. For | 
					
						
							|  |  |  |    *    this reason `NgOnChanges` will be deprecated and removed in future version and this | 
					
						
							|  |  |  |    *    API will be simplified to be consistent with `output`. | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |   inputs?: {[P in keyof T]?: string | [string, string]}; | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * 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. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2018-06-18 08:05:06 -07:00
										 |  |  |    * See: {@link NgOnChangesFeature}, {@link PublicFeature}, {@link InheritDefinitionFeature} | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |    */ | 
					
						
							|  |  |  |   features?: DirectiveDefFeature[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Function executed by the parent template to allow child directive to apply host bindings. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   hostBindings?: (directiveIndex: number, elementIndex: number) => void; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 10:43:07 +02:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Function to create instances of content queries associated with a given directive. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   contentQueries?: (() => void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Refreshes content queries associated with directives in a given view */ | 
					
						
							|  |  |  |   contentQueriesRefresh?: ((directiveIndex: number, queryIndex: number) => void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-20 14:44:22 -07:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * 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 { | 
					
						
							| 
									
										
										
										
											2018-06-26 10:43:06 -07:00
										 |  |  |   return (<PipeDefInternal<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
										 |  |  | } |