385 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
		
		
			
		
	
	
			385 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 
								 | 
							
								import {ddescribe, describe, it, iit, xit, expect, beforeEach, afterEach} from 'angular2/test_lib';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import {Parser, Lexer} from 'angular2/src/core/change_detection/change_detection';
							 | 
						||
| 
								 | 
							
								import {TemplateParser, splitClasses} from 'angular2/src/compiler/template_parser';
							 | 
						||
| 
								 | 
							
								import {HtmlParser} from 'angular2/src/compiler/html_parser';
							 | 
						||
| 
								 | 
							
								import {DirectiveMetadata, ComponentMetadata, TypeMeta} from 'angular2/src/compiler/api';
							 | 
						||
| 
								 | 
							
								import {
							 | 
						||
| 
								 | 
							
								  templateVisitAll,
							 | 
						||
| 
								 | 
							
								  TemplateAstVisitor,
							 | 
						||
| 
								 | 
							
								  TemplateAst,
							 | 
						||
| 
								 | 
							
								  NgContentAst,
							 | 
						||
| 
								 | 
							
								  EmbeddedTemplateAst,
							 | 
						||
| 
								 | 
							
								  ElementAst,
							 | 
						||
| 
								 | 
							
								  VariableAst,
							 | 
						||
| 
								 | 
							
								  BoundEventAst,
							 | 
						||
| 
								 | 
							
								  BoundPropertyAst,
							 | 
						||
| 
								 | 
							
								  AttrAst,
							 | 
						||
| 
								 | 
							
								  BoundTextAst,
							 | 
						||
| 
								 | 
							
								  TextAst
							 | 
						||
| 
								 | 
							
								} from 'angular2/src/compiler/template_ast';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import {Unparser} from '../core/change_detection/parser/unparser';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var expressionUnparser = new Unparser();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								export function main() {
							 | 
						||
| 
								 | 
							
								  describe('TemplateParser', () => {
							 | 
						||
| 
								 | 
							
								    var domParser: HtmlParser;
							 | 
						||
| 
								 | 
							
								    var parser: TemplateParser;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    beforeEach(() => {
							 | 
						||
| 
								 | 
							
								      domParser = new HtmlParser();
							 | 
						||
| 
								 | 
							
								      parser = new TemplateParser(new Parser(new Lexer()));
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    function parse(template: string, directives: DirectiveMetadata[]): TemplateAst[] {
							 | 
						||
| 
								 | 
							
								      return parser.parse(domParser.parse(template, 'TestComp'), directives);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    describe('parse', () => {
							 | 
						||
| 
								 | 
							
								      describe('nodes without bindings', () => {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse text nodes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('a', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([[TextAst, 'a', 'TestComp > #text(a):nth-child(0)']]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse elements with attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div a=b>', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [AttrAst, 'a', 'b', 'TestComp > div:nth-child(0)[a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should parse ngContent', () => {
							 | 
						||
| 
								 | 
							
								        var parsed = parse('<ng-content select="a">', []);
							 | 
						||
| 
								 | 
							
								        expect(humanizeTemplateAsts(parsed))
							 | 
						||
| 
								 | 
							
								            .toEqual([[NgContentAst, 'a', 'TestComp > ng-content:nth-child(0)']]);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should parse bound text nodes', () => {
							 | 
						||
| 
								 | 
							
								        expect(humanizeTemplateAsts(parse('{{a}}', [])))
							 | 
						||
| 
								 | 
							
								            .toEqual([[BoundTextAst, '{{ a }}', 'TestComp > #text({{a}}):nth-child(0)']]);
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      describe('property, event and variable bindings', () => {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound properties via [...] and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div [prop]="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'prop', 'v', 'TestComp > div:nth-child(0)[[prop]=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should camel case bound properties', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div [some-prop]="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'someProp', 'v', 'TestComp > div:nth-child(0)[[some-prop]=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound properties via bind- and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div bind-prop="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'prop', 'v', 'TestComp > div:nth-child(0)[bind-prop=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound properties via {{...}} and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div prop="{{v}}">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'prop', '{{ v }}', 'TestComp > div:nth-child(0)[prop={{v}}]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound events via (...) and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div (event)="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundEventAst, 'event', 'v', 'TestComp > div:nth-child(0)[(event)=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should camel case event names', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div (some-event)="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundEventAst, 'someEvent', 'v', 'TestComp > div:nth-child(0)[(some-event)=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound events via on- and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div on-event="v">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundEventAst, 'event', 'v', 'TestComp > div:nth-child(0)[on-event=v]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound events and properties via [(...)] and not report them as attributes',
							 | 
						||
| 
								 | 
							
								           () => {
							 | 
						||
| 
								 | 
							
								             expect(humanizeTemplateAsts(parse('<div [(prop)]="v">', [])))
							 | 
						||
| 
								 | 
							
								                 .toEqual([
							 | 
						||
| 
								 | 
							
								                   [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                   [BoundPropertyAst, 'prop', 'v', 'TestComp > div:nth-child(0)[[(prop)]=v]'],
							 | 
						||
| 
								 | 
							
								                   [
							 | 
						||
| 
								 | 
							
								                     BoundEventAst,
							 | 
						||
| 
								 | 
							
								                     'prop',
							 | 
						||
| 
								 | 
							
								                     'v = $event',
							 | 
						||
| 
								 | 
							
								                     'TestComp > div:nth-child(0)[[(prop)]=v]'
							 | 
						||
| 
								 | 
							
								                   ]
							 | 
						||
| 
								 | 
							
								                 ]);
							 | 
						||
| 
								 | 
							
								           });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound events and properties via bindon- and not report them as attributes',
							 | 
						||
| 
								 | 
							
								           () => {
							 | 
						||
| 
								 | 
							
								             expect(humanizeTemplateAsts(parse('<div bindon-prop="v">', [])))
							 | 
						||
| 
								 | 
							
								                 .toEqual([
							 | 
						||
| 
								 | 
							
								                   [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                   [BoundPropertyAst, 'prop', 'v', 'TestComp > div:nth-child(0)[bindon-prop=v]'],
							 | 
						||
| 
								 | 
							
								                   [
							 | 
						||
| 
								 | 
							
								                     BoundEventAst,
							 | 
						||
| 
								 | 
							
								                     'prop',
							 | 
						||
| 
								 | 
							
								                     'v = $event',
							 | 
						||
| 
								 | 
							
								                     'TestComp > div:nth-child(0)[bindon-prop=v]'
							 | 
						||
| 
								 | 
							
								                   ]
							 | 
						||
| 
								 | 
							
								                 ]);
							 | 
						||
| 
								 | 
							
								           });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse variables via #... and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div #a="b">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[#a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse variables via var-... and not report them as attributes', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div var-a="b">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[var-a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should camel case variables', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div var-some-a="b">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'someA', 'b', 'TestComp > div:nth-child(0)[var-some-a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should use $implicit as variable name if none was specified', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div var-a>', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', '$implicit', 'TestComp > div:nth-child(0)[var-a=]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      describe('directives', () => {
							 | 
						||
| 
								 | 
							
								        it('should locate directives ordered by typeName and components first', () => {
							 | 
						||
| 
								 | 
							
								          var dirA =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[a=b]', type: new TypeMeta({typeName: 'DirA'})});
							 | 
						||
| 
								 | 
							
								          var dirB =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[a]', type: new TypeMeta({typeName: 'DirB'})});
							 | 
						||
| 
								 | 
							
								          var comp =
							 | 
						||
| 
								 | 
							
								              new ComponentMetadata({selector: 'div', type: new TypeMeta({typeName: 'ZComp'})});
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div a="b">', [dirB, dirA, comp])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [comp, dirA, dirB], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [AttrAst, 'a', 'b', 'TestComp > div:nth-child(0)[a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should locate directives in property bindings', () => {
							 | 
						||
| 
								 | 
							
								          var dirA =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[a=b]', type: new TypeMeta({typeName: 'DirA'})});
							 | 
						||
| 
								 | 
							
								          var dirB =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[b]', type: new TypeMeta({typeName: 'DirB'})});
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div [a]="b">', [dirA, dirB])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [dirA], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'a', 'b', 'TestComp > div:nth-child(0)[[a]=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should locate directives in variable bindings', () => {
							 | 
						||
| 
								 | 
							
								          var dirA =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[a=b]', type: new TypeMeta({typeName: 'DirA'})});
							 | 
						||
| 
								 | 
							
								          var dirB =
							 | 
						||
| 
								 | 
							
								              new DirectiveMetadata({selector: '[b]', type: new TypeMeta({typeName: 'DirB'})});
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div #a="b">', [dirA, dirB])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [ElementAst, [dirA], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[#a=b]']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      describe('explicit templates', () => {
							 | 
						||
| 
								 | 
							
								        it('should create embedded templates for <template> elements', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<template></template>', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([[EmbeddedTemplateAst, [], 'TestComp > template:nth-child(0)']]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      describe('inline templates', () => {
							 | 
						||
| 
								 | 
							
								        it('should wrap the element into an EmbeddedTemplateAST', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div template>', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [EmbeddedTemplateAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse bound properties', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div template="ngIf test">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [EmbeddedTemplateAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [
							 | 
						||
| 
								 | 
							
								                  BoundPropertyAst,
							 | 
						||
| 
								 | 
							
								                  'ngIf',
							 | 
						||
| 
								 | 
							
								                  'test',
							 | 
						||
| 
								 | 
							
								                  'TestComp > div:nth-child(0)[template=ngIf test]'
							 | 
						||
| 
								 | 
							
								                ],
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse variables via #...', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div template="ngIf #a=b">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [EmbeddedTemplateAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[template=ngIf #a=b]'],
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should parse variables via var ...', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div template="ngIf var a=b">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [EmbeddedTemplateAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[template=ngIf var a=b]'],
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        describe('directives', () => {
							 | 
						||
| 
								 | 
							
								          it('should locate directives in property bindings', () => {
							 | 
						||
| 
								 | 
							
								            var dirA =
							 | 
						||
| 
								 | 
							
								                new DirectiveMetadata({selector: '[a=b]', type: new TypeMeta({typeName: 'DirA'})});
							 | 
						||
| 
								 | 
							
								            var dirB =
							 | 
						||
| 
								 | 
							
								                new DirectiveMetadata({selector: '[b]', type: new TypeMeta({typeName: 'DirB'})});
							 | 
						||
| 
								 | 
							
								            expect(humanizeTemplateAsts(parse('<div template="a b" b>', [dirA, dirB])))
							 | 
						||
| 
								 | 
							
								                .toEqual([
							 | 
						||
| 
								 | 
							
								                  [EmbeddedTemplateAst, [dirA], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                  [BoundPropertyAst, 'a', 'b', 'TestComp > div:nth-child(0)[template=a b]'],
							 | 
						||
| 
								 | 
							
								                  [ElementAst, [dirB], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                  [AttrAst, 'b', '', 'TestComp > div:nth-child(0)[b=]']
							 | 
						||
| 
								 | 
							
								                ]);
							 | 
						||
| 
								 | 
							
								          });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								          it('should locate directives in variable bindings', () => {
							 | 
						||
| 
								 | 
							
								            var dirA =
							 | 
						||
| 
								 | 
							
								                new DirectiveMetadata({selector: '[a=b]', type: new TypeMeta({typeName: 'DirA'})});
							 | 
						||
| 
								 | 
							
								            var dirB =
							 | 
						||
| 
								 | 
							
								                new DirectiveMetadata({selector: '[b]', type: new TypeMeta({typeName: 'DirB'})});
							 | 
						||
| 
								 | 
							
								            expect(humanizeTemplateAsts(parse('<div template="#a=b" b>', [dirA, dirB])))
							 | 
						||
| 
								 | 
							
								                .toEqual([
							 | 
						||
| 
								 | 
							
								                  [EmbeddedTemplateAst, [dirA], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                  [VariableAst, 'a', 'b', 'TestComp > div:nth-child(0)[template=#a=b]'],
							 | 
						||
| 
								 | 
							
								                  [ElementAst, [dirB], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                  [AttrAst, 'b', '', 'TestComp > div:nth-child(0)[b=]']
							 | 
						||
| 
								 | 
							
								                ]);
							 | 
						||
| 
								 | 
							
								          });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        it('should work with *... and use the attribute name as property binding name', () => {
							 | 
						||
| 
								 | 
							
								          expect(humanizeTemplateAsts(parse('<div *ng-if="test">', [])))
							 | 
						||
| 
								 | 
							
								              .toEqual([
							 | 
						||
| 
								 | 
							
								                [EmbeddedTemplateAst, [], 'TestComp > div:nth-child(0)'],
							 | 
						||
| 
								 | 
							
								                [BoundPropertyAst, 'ngIf', 'test', 'TestComp > div:nth-child(0)[*ng-if=test]'],
							 | 
						||
| 
								 | 
							
								                [ElementAst, [], 'TestComp > div:nth-child(0)']
							 | 
						||
| 
								 | 
							
								              ]);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								      });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    describe('splitClasses', () => {
							 | 
						||
| 
								 | 
							
								      it('should keep an empty class', () => { expect(splitClasses('a')).toEqual(['a']); });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should split 2 classes', () => { expect(splitClasses('a b')).toEqual(['a', 'b']); });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								      it('should trim classes', () => { expect(splitClasses(' a  b ')).toEqual(['a', 'b']); });
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								  });
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								export function humanizeTemplateAsts(templateAsts: TemplateAst[]): any[] {
							 | 
						||
| 
								 | 
							
								  var humanizer = new TemplateHumanizer();
							 | 
						||
| 
								 | 
							
								  templateVisitAll(humanizer, templateAsts);
							 | 
						||
| 
								 | 
							
								  return humanizer.result;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class TemplateHumanizer implements TemplateAstVisitor {
							 | 
						||
| 
								 | 
							
								  result: any[] = [];
							 | 
						||
| 
								 | 
							
								  visitNgContent(ast: NgContentAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([NgContentAst, ast.select, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitEmbeddedTemplate(ast: EmbeddedTemplateAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([EmbeddedTemplateAst, ast.directives, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.attrs);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.properties);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.vars);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.children);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitElement(ast: ElementAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([ElementAst, ast.directives, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.attrs);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.properties);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.events);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.vars);
							 | 
						||
| 
								 | 
							
								    templateVisitAll(this, ast.children);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitVariable(ast: VariableAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([VariableAst, ast.name, ast.value, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitEvent(ast: BoundEventAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push(
							 | 
						||
| 
								 | 
							
								        [BoundEventAst, ast.name, expressionUnparser.unparse(ast.handler), ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitProperty(ast: BoundPropertyAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push(
							 | 
						||
| 
								 | 
							
								        [BoundPropertyAst, ast.name, expressionUnparser.unparse(ast.value), ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitAttr(ast: AttrAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([AttrAst, ast.name, ast.value, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitBoundText(ast: BoundTextAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([BoundTextAst, expressionUnparser.unparse(ast.value), ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								  visitText(ast: TextAst): any {
							 | 
						||
| 
								 | 
							
								    this.result.push([TextAst, ast.value, ast.sourceInfo]);
							 | 
						||
| 
								 | 
							
								    return null;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 |