| 
									
										
										
										
											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'; | 
					
						
							|  |  |  | 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-01-23 18:39:09 -08:00
										 |  |  | import {ComponentDef, ComponentDefArgs, DirectiveDef, DirectiveDefArgs} from './interfaces/definition'; | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |  *   static [COMPONENT_DEF_SYMBOL] = defineComponent({ | 
					
						
							|  |  |  |  *     ... | 
					
						
							|  |  |  |  *   }); | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function defineComponent<T>(componentDefinition: ComponentDefArgs<T>): ComponentDef<T> { | 
					
						
							| 
									
										
										
										
											2018-01-23 18:39:09 -08:00
										 |  |  |   const type = componentDefinition.type; | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08: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, | 
					
						
							|  |  |  |     n: componentDefinition.factory, | 
					
						
							|  |  |  |     tag: (componentDefinition as ComponentDefArgs<T>).tag || null !, | 
					
						
							|  |  |  |     template: (componentDefinition as ComponentDefArgs<T>).template || null !, | 
					
						
							| 
									
										
										
										
											2017-12-20 16:26:07 -08:00
										 |  |  |     h: componentDefinition.hostBindings || noop, | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |     inputs: invertObject(componentDefinition.inputs), | 
					
						
							|  |  |  |     outputs: invertObject(componentDefinition.outputs), | 
					
						
							|  |  |  |     methods: invertObject(componentDefinition.methods), | 
					
						
							|  |  |  |     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, | 
					
						
							|  |  |  |     onDestroy: type.prototype.ngOnDestroy || null | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  |   const feature = componentDefinition.features; | 
					
						
							|  |  |  |   feature && feature.forEach((fn) => fn(def)); | 
					
						
							|  |  |  |   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-01-29 15:57:58 -08:00
										 |  |  | export function NgOnChangesFeature(definition: DirectiveDef<any>): void { | 
					
						
							|  |  |  |   const inputs = definition.inputs; | 
					
						
							|  |  |  |   const proto = definition.type.prototype; | 
					
						
							|  |  |  |   // Place where we will store SimpleChanges if there is a change
 | 
					
						
							|  |  |  |   Object.defineProperty(proto, PRIVATE_PREFIX, {value: undefined, writable: true}); | 
					
						
							|  |  |  |   for (let pubKey in inputs) { | 
					
						
							|  |  |  |     const minKey = inputs[pubKey]; | 
					
						
							|  |  |  |     const privateMinKey = PRIVATE_PREFIX + minKey; | 
					
						
							|  |  |  |     // Create a place where the actual value will be stored and make it non-enumerable
 | 
					
						
							|  |  |  |     Object.defineProperty(proto, privateMinKey, {value: undefined, writable: true}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const existingDesc = Object.getOwnPropertyDescriptor(proto, minKey); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // create a getter and setter for property
 | 
					
						
							|  |  |  |     Object.defineProperty(proto, minKey, { | 
					
						
							|  |  |  |       get: function(this: OnChangesExpando) { | 
					
						
							|  |  |  |         return (existingDesc && existingDesc.get) ? existingDesc.get.call(this) : | 
					
						
							|  |  |  |                                                     this[privateMinKey]; | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       set: function(this: OnChangesExpando, value: any) { | 
					
						
							|  |  |  |         let simpleChanges = this[PRIVATE_PREFIX]; | 
					
						
							|  |  |  |         let isFirstChange = simpleChanges === undefined; | 
					
						
							|  |  |  |         if (simpleChanges == null) { | 
					
						
							|  |  |  |           simpleChanges = this[PRIVATE_PREFIX] = {}; | 
					
						
							| 
									
										
										
										
											2018-01-17 10:09:05 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-01-29 15:57:58 -08:00
										 |  |  |         simpleChanges[pubKey] = new SimpleChange(this[privateMinKey], value, isFirstChange); | 
					
						
							|  |  |  |         (existingDesc && existingDesc.set) ? existingDesc.set.call(this, value) : | 
					
						
							|  |  |  |                                              this[privateMinKey] = value; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-01-26 13:38:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-29 15:57:58 -08: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-01-29 15:57:58 -08:00
										 |  |  |   definition.doCheck = onChangesWrapper(definition.doCheck); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | export function PublicFeature<T>(definition: DirectiveDef<T>) { | 
					
						
							|  |  |  |   definition.diPublic = diPublic; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const EMPTY = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-20 16:26:07 -08:00
										 |  |  | function noop() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-20 10:47:22 -08:00
										 |  |  | /** 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
 | 
					
						
							|  |  |  |  *   static [DIRECTIVE_DEF_SYMBOL] = defineDirective({ | 
					
						
							|  |  |  |  *     ... | 
					
						
							|  |  |  |  *   }); | 
					
						
							|  |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export const defineDirective = defineComponent as<T>(directiveDefinition: DirectiveDefArgs<T>) => | 
					
						
							|  |  |  |     DirectiveDef<T>; |