| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | import {RegExpWrapper, StringWrapper} from 'angular2/src/facade/lang'; | 
					
						
							|  |  |  | import {List, ListWrapper} from 'angular2/src/facade/collection'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import {DirectiveIndex} from './directive_record'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import {ProtoRecord} from './proto_record'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  | // The names of these fields must be kept in sync with abstract_change_detector.ts or change
 | 
					
						
							|  |  |  | // detection will fail.
 | 
					
						
							|  |  |  | const _ALREADY_CHECKED_ACCESSOR = "alreadyChecked"; | 
					
						
							|  |  |  | const _CONTEXT_ACCESSOR = "context"; | 
					
						
							| 
									
										
										
										
											2015-07-30 10:59:52 -07:00
										 |  |  | const _FIRST_PROTO_IN_CURRENT_BINDING = "firstProtoInCurrentBinding"; | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  | const _DIRECTIVES_ACCESSOR = "directiveRecords"; | 
					
						
							|  |  |  | const _DISPATCHER_ACCESSOR = "dispatcher"; | 
					
						
							|  |  |  | const _LOCALS_ACCESSOR = "locals"; | 
					
						
							|  |  |  | const _MODE_ACCESSOR = "mode"; | 
					
						
							|  |  |  | const _PIPES_ACCESSOR = "pipes"; | 
					
						
							|  |  |  | const _PROTOS_ACCESSOR = "protos"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // `context` is always first.
 | 
					
						
							|  |  |  | export const CONTEXT_INDEX = 0; | 
					
						
							|  |  |  | const _FIELD_PREFIX = 'this.'; | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | var _whiteSpaceRegExp = RegExpWrapper.create("\\W", "g"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns `s` with all non-identifier characters removed. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export function sanitizeName(s: string): string { | 
					
						
							|  |  |  |   return StringWrapper.replaceAll(s, _whiteSpaceRegExp, ''); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class responsible for providing field and local variable names for change detector classes. | 
					
						
							|  |  |  |  * Also provides some convenience functions, for example, declaring variables, destroying pipes, | 
					
						
							|  |  |  |  * and dehydrating the detector. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export class CodegenNameUtil { | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Record names sanitized for use as fields. | 
					
						
							|  |  |  |    * See [sanitizeName] for details. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   _sanitizedNames: List<string>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   constructor(private records: List<ProtoRecord>, private directiveRecords: List<any>, | 
					
						
							|  |  |  |               private utilName: string) { | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     this._sanitizedNames = ListWrapper.createFixedSize(this.records.length + 1); | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |     this._sanitizedNames[CONTEXT_INDEX] = _CONTEXT_ACCESSOR; | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     for (var i = 0, iLen = this.records.length; i < iLen; ++i) { | 
					
						
							|  |  |  |       this._sanitizedNames[i + 1] = sanitizeName(`${this.records[i].name}${i}`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   _addFieldPrefix(name: string): string { return `${_FIELD_PREFIX}${name}`; } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   getDispatcherName(): string { return this._addFieldPrefix(_DISPATCHER_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getPipesAccessorName(): string { return this._addFieldPrefix(_PIPES_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getProtosName(): string { return this._addFieldPrefix(_PROTOS_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getDirectivesAccessorName(): string { return this._addFieldPrefix(_DIRECTIVES_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getLocalsAccessorName(): string { return this._addFieldPrefix(_LOCALS_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getAlreadyCheckedName(): string { return this._addFieldPrefix(_ALREADY_CHECKED_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getModeName(): string { return this._addFieldPrefix(_MODE_ACCESSOR); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-30 10:59:52 -07:00
										 |  |  |   getFirstProtoInCurrentBinding(): string { | 
					
						
							|  |  |  |     return this._addFieldPrefix(_FIRST_PROTO_IN_CURRENT_BINDING); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getLocalName(idx: int): string { return `l_${this._sanitizedNames[idx]}`; } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getChangeName(idx: int): string { return `c_${this._sanitizedNames[idx]}`; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generate a statement initializing local variables used when detecting changes. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   genInitLocals(): string { | 
					
						
							|  |  |  |     var declarations = []; | 
					
						
							|  |  |  |     var assignments = []; | 
					
						
							|  |  |  |     for (var i = 0, iLen = this.getFieldCount(); i < iLen; ++i) { | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |       if (i == CONTEXT_INDEX) { | 
					
						
							|  |  |  |         declarations.push(`${this.getLocalName(i)} = ${this.getFieldName(i)}`); | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2015-07-28 15:57:05 -07:00
										 |  |  |         var rec = this.records[i - 1]; | 
					
						
							|  |  |  |         if (rec.argumentToPureFunction) { | 
					
						
							|  |  |  |           var changeName = this.getChangeName(i); | 
					
						
							|  |  |  |           declarations.push(`${this.getLocalName(i)},${changeName}`); | 
					
						
							|  |  |  |           assignments.push(changeName); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           declarations.push(`${this.getLocalName(i)}`); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |     var assignmentsCode = | 
					
						
							|  |  |  |         ListWrapper.isEmpty(assignments) ? '' : `${ListWrapper.join(assignments, '=')} = false;`; | 
					
						
							|  |  |  |     return `var ${ListWrapper.join(declarations, ',')};${assignmentsCode}`; | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   getFieldCount(): int { return this._sanitizedNames.length; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   getFieldName(idx: int): string { return this._addFieldPrefix(this._sanitizedNames[idx]); } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getAllFieldNames(): List<string> { | 
					
						
							|  |  |  |     var fieldList = []; | 
					
						
							|  |  |  |     for (var k = 0, kLen = this.getFieldCount(); k < kLen; ++k) { | 
					
						
							| 
									
										
										
										
											2015-07-28 15:57:05 -07:00
										 |  |  |       if (k === 0 || this.records[k - 1].shouldBeChecked()) { | 
					
						
							|  |  |  |         fieldList.push(this.getFieldName(k)); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-28 15:57:05 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     for (var i = 0, iLen = this.records.length; i < iLen; ++i) { | 
					
						
							|  |  |  |       var rec = this.records[i]; | 
					
						
							|  |  |  |       if (rec.isPipeRecord()) { | 
					
						
							|  |  |  |         fieldList.push(this.getPipeName(rec.selfIndex)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-28 15:57:05 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |     for (var j = 0, jLen = this.directiveRecords.length; j < jLen; ++j) { | 
					
						
							|  |  |  |       var dRec = this.directiveRecords[j]; | 
					
						
							|  |  |  |       fieldList.push(this.getDirectiveName(dRec.directiveIndex)); | 
					
						
							|  |  |  |       if (dRec.isOnPushChangeDetection()) { | 
					
						
							|  |  |  |         fieldList.push(this.getDetectorName(dRec.directiveIndex)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return fieldList; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generates statements which clear all fields so that the change detector is dehydrated. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   genDehydrateFields(): string { | 
					
						
							|  |  |  |     var fields = this.getAllFieldNames(); | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |     ListWrapper.removeAt(fields, CONTEXT_INDEX); | 
					
						
							| 
									
										
										
										
											2015-07-29 10:43:07 -07:00
										 |  |  |     if (ListWrapper.isEmpty(fields)) return ''; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // At least one assignment.
 | 
					
						
							|  |  |  |     fields.push(`${this.utilName}.uninitialized;`); | 
					
						
							|  |  |  |     return ListWrapper.join(fields, ' = '); | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generates statements destroying all pipe variables. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   genPipeOnDestroy(): string { | 
					
						
							| 
									
										
										
										
											2015-08-12 10:46:06 -07:00
										 |  |  |     return ListWrapper.join( | 
					
						
							|  |  |  |         ListWrapper.map( | 
					
						
							|  |  |  |             ListWrapper.filter(this.records, (r) => { return r.isPipeRecord(); }), | 
					
						
							|  |  |  |             (r) => { | 
					
						
							|  |  |  |               return `${this.utilName}.callPipeOnDestroy(${this.getPipeName(r.selfIndex)});`; | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |         '\n'); | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   getPipeName(idx: int): string { | 
					
						
							|  |  |  |     return this._addFieldPrefix(`${this._sanitizedNames[idx]}_pipe`); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getAllDirectiveNames(): List<string> { | 
					
						
							|  |  |  |     return ListWrapper.map(this.directiveRecords, d => this.getDirectiveName(d.directiveIndex)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   getDirectiveName(d: DirectiveIndex): string { | 
					
						
							|  |  |  |     return this._addFieldPrefix(`directive_${d.name}`); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   getAllDetectorNames(): List<string> { | 
					
						
							|  |  |  |     return ListWrapper.map( | 
					
						
							|  |  |  |         ListWrapper.filter(this.directiveRecords, r => r.isOnPushChangeDetection()), | 
					
						
							|  |  |  |         (d) => this.getDetectorName(d.directiveIndex)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 12:43:41 -07:00
										 |  |  |   getDetectorName(d: DirectiveIndex): string { return this._addFieldPrefix(`detector_${d.name}`); } | 
					
						
							| 
									
										
										
										
											2015-07-23 16:08:55 -07:00
										 |  |  | } |