| 
									
										
										
										
											2016-06-23 09:47:54 -07: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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-21 11:41:25 -07:00
										 |  |  | import {ParseError, ParseSourceSpan} from '../parse_util'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-30 14:59:23 -07:00
										 |  |  | import {HtmlAst, HtmlAstVisitor, HtmlAttrAst, HtmlCommentAst, HtmlElementAst, HtmlExpansionAst, HtmlExpansionCaseAst, HtmlTextAst, htmlVisitAll} from './html_ast'; | 
					
						
							| 
									
										
										
										
											2016-07-21 11:41:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  | // http://cldr.unicode.org/index/cldr-spec/plural-rules
 | 
					
						
							|  |  |  | const PLURAL_CASES: string[] = ['zero', 'one', 'two', 'few', 'many', 'other']; | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Expands special forms into elements. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For example, | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * { messages.length, plural, | 
					
						
							|  |  |  |  *   =0 {zero} | 
					
						
							|  |  |  |  *   =1 {one} | 
					
						
							| 
									
										
										
										
											2016-06-09 13:48:53 -07:00
										 |  |  |  *   other {more than one} | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  |  * } | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * will be expanded into | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |  * <ng-container [ngPlural]="messages.length"> | 
					
						
							|  |  |  |  *   <template ngPluralCase="=0">zero</ng-container> | 
					
						
							|  |  |  |  *   <template ngPluralCase="=1">one</ng-container> | 
					
						
							|  |  |  |  *   <template ngPluralCase="other">more than one</ng-container> | 
					
						
							|  |  |  |  * </ng-container> | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | export function expandNodes(nodes: HtmlAst[]): ExpansionResult { | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |   const expander = new _Expander(); | 
					
						
							|  |  |  |   return new ExpansionResult(htmlVisitAll(expander, nodes), expander.isExpanded, expander.errors); | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class ExpansionResult { | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  |   constructor(public nodes: HtmlAst[], public expanded: boolean, public errors: ParseError[]) {} | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-30 14:59:23 -07:00
										 |  |  | export class ExpansionError extends ParseError { | 
					
						
							|  |  |  |   constructor(span: ParseSourceSpan, errorMsg: string) { super(span, errorMsg); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-14 17:50:23 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Expand expansion forms (plural, select) to directives | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @internal | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | class _Expander implements HtmlAstVisitor { | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |   isExpanded: boolean = false; | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  |   errors: ParseError[] = []; | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   visitElement(ast: HtmlElementAst, context: any): any { | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |     return new HtmlElementAst( | 
					
						
							|  |  |  |         ast.name, ast.attrs, htmlVisitAll(this, ast.children), ast.sourceSpan, ast.startSourceSpan, | 
					
						
							|  |  |  |         ast.endSourceSpan); | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   visitAttr(ast: HtmlAttrAst, context: any): any { return ast; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   visitText(ast: HtmlTextAst, context: any): any { return ast; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   visitComment(ast: HtmlCommentAst, context: any): any { return ast; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   visitExpansion(ast: HtmlExpansionAst, context: any): any { | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |     this.isExpanded = true; | 
					
						
							|  |  |  |     return ast.type == 'plural' ? _expandPluralForm(ast, this.errors) : | 
					
						
							|  |  |  |                                   _expandDefaultForm(ast, this.errors); | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   visitExpansionCase(ast: HtmlExpansionCaseAst, context: any): any { | 
					
						
							| 
									
										
										
										
											2016-06-30 14:59:23 -07:00
										 |  |  |     throw new Error('Should not be reached'); | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  | function _expandPluralForm(ast: HtmlExpansionAst, errors: ParseError[]): HtmlElementAst { | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |   const children = ast.cases.map(c => { | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  |     if (PLURAL_CASES.indexOf(c.value) == -1 && !c.value.match(/^=\d+$/)) { | 
					
						
							| 
									
										
										
										
											2016-06-30 14:59:23 -07:00
										 |  |  |       errors.push(new ExpansionError( | 
					
						
							| 
									
										
										
										
											2016-06-09 14:53:03 -07:00
										 |  |  |           c.valueSourceSpan, | 
					
						
							|  |  |  |           `Plural cases should be "=<number>" or one of ${PLURAL_CASES.join(", ")}`)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const expansionResult = expandNodes(c.expression); | 
					
						
							| 
									
										
										
										
											2016-06-14 17:50:23 -07:00
										 |  |  |     errors.push(...expansionResult.errors); | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |     return new HtmlElementAst( | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |         `template`, [new HtmlAttrAst('ngPluralCase', `${c.value}`, c.valueSourceSpan)], | 
					
						
							|  |  |  |         expansionResult.nodes, c.sourceSpan, c.sourceSpan, c.sourceSpan); | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |   const switchAttr = new HtmlAttrAst('[ngPlural]', ast.switchValue, ast.switchValueSourceSpan); | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |   return new HtmlElementAst( | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |       'ng-container', [switchAttr], children, ast.sourceSpan, ast.sourceSpan, ast.sourceSpan); | 
					
						
							| 
									
										
										
										
											2016-04-12 11:46:49 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  | function _expandDefaultForm(ast: HtmlExpansionAst, errors: ParseError[]): HtmlElementAst { | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  |   let children = ast.cases.map(c => { | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |     const expansionResult = expandNodes(c.expression); | 
					
						
							|  |  |  |     errors.push(...expansionResult.errors); | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |     return new HtmlElementAst( | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |         `template`, [new HtmlAttrAst('ngSwitchCase', `${c.value}`, c.valueSourceSpan)], | 
					
						
							|  |  |  |         expansionResult.nodes, c.sourceSpan, c.sourceSpan, c.sourceSpan); | 
					
						
							| 
									
										
										
										
											2016-04-13 16:01:25 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |   const switchAttr = new HtmlAttrAst('[ngSwitch]', ast.switchValue, ast.switchValueSourceSpan); | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |   return new HtmlElementAst( | 
					
						
							| 
									
										
										
										
											2016-06-17 11:38:24 -07:00
										 |  |  |       'ng-container', [switchAttr], children, ast.sourceSpan, ast.sourceSpan, ast.sourceSpan); | 
					
						
							| 
									
										
										
										
											2016-04-28 17:50:03 -07:00
										 |  |  | } |