486 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			486 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import {
 | 
						|
  ddescribe,
 | 
						|
  describe,
 | 
						|
  xdescribe,
 | 
						|
  it,
 | 
						|
  iit,
 | 
						|
  xit,
 | 
						|
  expect,
 | 
						|
  beforeEach,
 | 
						|
  afterEach,
 | 
						|
  AsyncTestCompleter,
 | 
						|
  inject,
 | 
						|
  beforeEachBindings
 | 
						|
} from 'angular2/test_lib';
 | 
						|
 | 
						|
import {CONST_EXPR, stringify, isType, Type, isBlank} from 'angular2/src/core/facade/lang';
 | 
						|
import {PromiseWrapper, Promise} from 'angular2/src/core/facade/async';
 | 
						|
import {TemplateParser} from 'angular2/src/compiler/template_parser';
 | 
						|
import {
 | 
						|
  CommandVisitor,
 | 
						|
  TextCmd,
 | 
						|
  NgContentCmd,
 | 
						|
  BeginElementCmd,
 | 
						|
  BeginComponentCmd,
 | 
						|
  EmbeddedTemplateCmd,
 | 
						|
  TemplateCmd,
 | 
						|
  visitAllCommands,
 | 
						|
  CompiledTemplate
 | 
						|
} from 'angular2/src/core/compiler/template_commands';
 | 
						|
import {CommandCompiler} from 'angular2/src/compiler/command_compiler';
 | 
						|
import {
 | 
						|
  NormalizedDirectiveMetadata,
 | 
						|
  TypeMetadata,
 | 
						|
  NormalizedTemplateMetadata
 | 
						|
} from 'angular2/src/compiler/directive_metadata';
 | 
						|
import {SourceModule, SourceExpression, moduleRef} from 'angular2/src/compiler/source_module';
 | 
						|
import {ViewEncapsulation} from 'angular2/src/core/render/api';
 | 
						|
import {evalModule} from './eval_module';
 | 
						|
import {
 | 
						|
  escapeSingleQuoteString,
 | 
						|
  codeGenValueFn,
 | 
						|
  codeGenExportVariable
 | 
						|
} from 'angular2/src/compiler/util';
 | 
						|
import {TEST_BINDINGS} from './test_bindings';
 | 
						|
 | 
						|
const BEGIN_ELEMENT = 'BEGIN_ELEMENT';
 | 
						|
const END_ELEMENT = 'END_ELEMENT';
 | 
						|
const BEGIN_COMPONENT = 'BEGIN_COMPONENT';
 | 
						|
const END_COMPONENT = 'END_COMPONENT';
 | 
						|
const TEXT = 'TEXT';
 | 
						|
const NG_CONTENT = 'NG_CONTENT';
 | 
						|
const EMBEDDED_TEMPLATE = 'EMBEDDED_TEMPLATE';
 | 
						|
 | 
						|
// Attention: These module names have to correspond to real modules!
 | 
						|
const THIS_MODULE_NAME = 'angular2/test/compiler/command_compiler_spec';
 | 
						|
var THIS_MODULE_REF = moduleRef(THIS_MODULE_NAME);
 | 
						|
var TEMPLATE_COMMANDS_MODULE_REF = moduleRef('angular2/src/core/compiler/template_commands');
 | 
						|
 | 
						|
// Attention: read by eval!
 | 
						|
export class RootComp {}
 | 
						|
export class SomeDir {}
 | 
						|
export class AComp {}
 | 
						|
 | 
						|
var RootCompTypeMeta =
 | 
						|
    new TypeMetadata({id: 1, name: 'RootComp', runtime: RootComp, moduleId: THIS_MODULE_NAME});
 | 
						|
var SomeDirTypeMeta =
 | 
						|
    new TypeMetadata({id: 2, name: 'SomeDir', runtime: SomeDir, moduleId: THIS_MODULE_NAME});
 | 
						|
var ACompTypeMeta =
 | 
						|
    new TypeMetadata({id: 3, name: 'AComp', runtime: AComp, moduleId: THIS_MODULE_NAME});
 | 
						|
 | 
						|
var NESTED_COMPONENT = new CompiledTemplate(45, () => []);
 | 
						|
 | 
						|
export function main() {
 | 
						|
  describe('CommandCompiler', () => {
 | 
						|
    beforeEachBindings(() => TEST_BINDINGS);
 | 
						|
 | 
						|
    var parser: TemplateParser;
 | 
						|
    var commandCompiler: CommandCompiler;
 | 
						|
    var componentTemplateFactory: Function;
 | 
						|
 | 
						|
    beforeEach(inject([TemplateParser, CommandCompiler], (_templateParser, _commandCompiler) => {
 | 
						|
      parser = _templateParser;
 | 
						|
      commandCompiler = _commandCompiler;
 | 
						|
    }));
 | 
						|
 | 
						|
    function createComp({type, selector, template, encapsulation, ngContentSelectors}: {
 | 
						|
      type?: TypeMetadata,
 | 
						|
      selector?: string,
 | 
						|
      template?: string,
 | 
						|
      encapsulation?: ViewEncapsulation,
 | 
						|
      ngContentSelectors?: string[]
 | 
						|
    }): NormalizedDirectiveMetadata {
 | 
						|
      if (isBlank(encapsulation)) {
 | 
						|
        encapsulation = ViewEncapsulation.None;
 | 
						|
      }
 | 
						|
      if (isBlank(selector)) {
 | 
						|
        selector = 'root';
 | 
						|
      }
 | 
						|
      if (isBlank(ngContentSelectors)) {
 | 
						|
        ngContentSelectors = [];
 | 
						|
      }
 | 
						|
      if (isBlank(template)) {
 | 
						|
        template = '';
 | 
						|
      }
 | 
						|
      return new NormalizedDirectiveMetadata({
 | 
						|
        selector: selector,
 | 
						|
        isComponent: true,
 | 
						|
        type: type,
 | 
						|
        template: new NormalizedTemplateMetadata({
 | 
						|
          template: template,
 | 
						|
          ngContentSelectors: ngContentSelectors,
 | 
						|
          encapsulation: encapsulation
 | 
						|
        })
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    function createDirective(type: TypeMetadata, selector: string): NormalizedDirectiveMetadata {
 | 
						|
      return new NormalizedDirectiveMetadata({selector: selector, isComponent: false, type: type});
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    function createTests(run: Function) {
 | 
						|
      describe('text', () => {
 | 
						|
 | 
						|
        it('should create unbound text commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: 'a'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([[TEXT, 'a', false, null]]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should create bound text commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: '{{a}}'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([[TEXT, null, true, null]]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
      });
 | 
						|
 | 
						|
      describe('elements', () => {
 | 
						|
 | 
						|
        it('should create unbound element commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: '<div a="b">'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [BEGIN_ELEMENT, 'div', ['a', 'b'], [], [], [], false, null],
 | 
						|
                     [END_ELEMENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should create bound element commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<div a="b" #some-var="someValue" (click)="someHandler">'
 | 
						|
             });
 | 
						|
             var dir = createDirective(SomeDirTypeMeta, '[a]');
 | 
						|
             run(rootComp, [dir])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [
 | 
						|
                       BEGIN_ELEMENT,
 | 
						|
                       'div',
 | 
						|
                       ['a', 'b'],
 | 
						|
                       ['click'],
 | 
						|
                       ['someVar', 'someValue'],
 | 
						|
                       ['SomeDirType'],
 | 
						|
                       true,
 | 
						|
                       null
 | 
						|
                     ],
 | 
						|
                     [END_ELEMENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should emulate style encapsulation', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<div>',
 | 
						|
               encapsulation: ViewEncapsulation.Emulated
 | 
						|
             });
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [BEGIN_ELEMENT, 'div', ['_ngcontent-1', ''], [], [], [], false, null],
 | 
						|
                     [END_ELEMENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should create nested nodes', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: '<div>a</div>'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [BEGIN_ELEMENT, 'div', [], [], [], [], false, null],
 | 
						|
                     [TEXT, 'a', false, null],
 | 
						|
                     [END_ELEMENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
      });
 | 
						|
 | 
						|
      describe('components', () => {
 | 
						|
 | 
						|
        it('should create component commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<a a="b" #some-var="someValue" (click)="someHandler">'
 | 
						|
             });
 | 
						|
             var comp = createComp({type: ACompTypeMeta, selector: 'a'});
 | 
						|
             run(rootComp, [comp])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [
 | 
						|
                       BEGIN_COMPONENT,
 | 
						|
                       'a',
 | 
						|
                       ['a', 'b'],
 | 
						|
                       ['click'],
 | 
						|
                       ['someVar', 'someValue'],
 | 
						|
                       ['ACompType'],
 | 
						|
                       false,
 | 
						|
                       null,
 | 
						|
                       3
 | 
						|
                     ],
 | 
						|
                     [END_COMPONENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should emulate style encapsulation on host elements',
 | 
						|
           inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<a></a>',
 | 
						|
               encapsulation: ViewEncapsulation.Emulated
 | 
						|
             });
 | 
						|
             var comp = createComp(
 | 
						|
                 {type: ACompTypeMeta, selector: 'a', encapsulation: ViewEncapsulation.Emulated});
 | 
						|
             run(rootComp, [comp])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [
 | 
						|
                       BEGIN_COMPONENT,
 | 
						|
                       'a',
 | 
						|
                       ['_nghost-3', '', '_ngcontent-1', ''],
 | 
						|
                       [],
 | 
						|
                       [],
 | 
						|
                       ['ACompType'],
 | 
						|
                       false,
 | 
						|
                       null,
 | 
						|
                       3
 | 
						|
                     ],
 | 
						|
                     [END_COMPONENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should set nativeShadow flag', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: '<a></a>'});
 | 
						|
             var comp = createComp(
 | 
						|
                 {type: ACompTypeMeta, selector: 'a', encapsulation: ViewEncapsulation.Native});
 | 
						|
             run(rootComp, [comp])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [BEGIN_COMPONENT, 'a', [], [], [], ['ACompType'], true, null, 3],
 | 
						|
                     [END_COMPONENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should create nested nodes and set ngContentIndex',
 | 
						|
           inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({type: RootCompTypeMeta, template: '<a>t</a>'});
 | 
						|
             var comp = createComp({type: ACompTypeMeta, selector: 'a', ngContentSelectors: ['*']});
 | 
						|
             run(rootComp, [comp])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [BEGIN_COMPONENT, 'a', [], [], [], ['ACompType'], false, null, 3],
 | 
						|
                     [TEXT, 't', false, 0],
 | 
						|
                     [END_COMPONENT]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
      });
 | 
						|
 | 
						|
      describe('embedded templates', () => {
 | 
						|
        it('should create embedded template commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<template a="b" #some-var="someValue"></template>'
 | 
						|
             });
 | 
						|
             var dir = createDirective(SomeDirTypeMeta, '[a]');
 | 
						|
             run(rootComp, [dir])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([
 | 
						|
                     [
 | 
						|
                       EMBEDDED_TEMPLATE,
 | 
						|
                       ['a', 'b'],
 | 
						|
                       ['someVar', 'someValue'],
 | 
						|
                       ['SomeDirType'],
 | 
						|
                       false,
 | 
						|
                       null,
 | 
						|
                       []
 | 
						|
                     ]
 | 
						|
                   ]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should created nested nodes', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp =
 | 
						|
                 createComp({type: RootCompTypeMeta, template: '<template>t</template>'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual(
 | 
						|
                       [[EMBEDDED_TEMPLATE, [], [], [], false, null, [[TEXT, 't', false, null]]]]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
        it('should calculate wether the template is merged based on nested ng-content elements',
 | 
						|
           inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp = createComp({
 | 
						|
               type: RootCompTypeMeta,
 | 
						|
               template: '<template><ng-content></ng-content></template>'
 | 
						|
             });
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual(
 | 
						|
                       [[EMBEDDED_TEMPLATE, [], [], [], true, null, [[NG_CONTENT, null]]]]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
 | 
						|
      });
 | 
						|
 | 
						|
      describe('ngContent', () => {
 | 
						|
        it('should create ng-content commands', inject([AsyncTestCompleter], (async) => {
 | 
						|
             var rootComp =
 | 
						|
                 createComp({type: RootCompTypeMeta, template: '<ng-content></ng-content>'});
 | 
						|
             run(rootComp, [])
 | 
						|
                 .then((data) => {
 | 
						|
                   expect(data).toEqual([[NG_CONTENT, null]]);
 | 
						|
                   async.done();
 | 
						|
                 });
 | 
						|
           }));
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    describe('compileComponentRuntime', () => {
 | 
						|
      beforeEach(() => {
 | 
						|
        componentTemplateFactory = (directive: NormalizedDirectiveMetadata) => {
 | 
						|
          return new CompiledTemplate(directive.type.id, () => []);
 | 
						|
        };
 | 
						|
      });
 | 
						|
 | 
						|
      function run(component: NormalizedDirectiveMetadata,
 | 
						|
                   directives: NormalizedDirectiveMetadata[]): Promise<any[][]> {
 | 
						|
        var parsedTemplate =
 | 
						|
            parser.parse(component.template.template, directives, component.type.name);
 | 
						|
        var commands = commandCompiler.compileComponentRuntime(component, parsedTemplate,
 | 
						|
                                                               componentTemplateFactory);
 | 
						|
        return PromiseWrapper.resolve(humanize(commands));
 | 
						|
      }
 | 
						|
 | 
						|
      createTests(run);
 | 
						|
    });
 | 
						|
 | 
						|
 | 
						|
    describe('compileComponentCodeGen', () => {
 | 
						|
      beforeEach(() => {
 | 
						|
        componentTemplateFactory = (directive: NormalizedDirectiveMetadata) => {
 | 
						|
          return `new ${TEMPLATE_COMMANDS_MODULE_REF}CompiledTemplate(${directive.type.id}, ${codeGenValueFn([], '{}')})`;
 | 
						|
        };
 | 
						|
      });
 | 
						|
 | 
						|
      function run(component: NormalizedDirectiveMetadata,
 | 
						|
                   directives: NormalizedDirectiveMetadata[]): Promise<any[][]> {
 | 
						|
        var parsedTemplate =
 | 
						|
            parser.parse(component.template.template, directives, component.type.name);
 | 
						|
        var sourceModule = commandCompiler.compileComponentCodeGen(component, parsedTemplate,
 | 
						|
                                                                   componentTemplateFactory);
 | 
						|
        var testableModule = createTestableModule(sourceModule).getSourceWithImports();
 | 
						|
        return evalModule(testableModule.source, testableModule.imports, null);
 | 
						|
      }
 | 
						|
 | 
						|
      createTests(run);
 | 
						|
    });
 | 
						|
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
// Attention: read by eval!
 | 
						|
export function humanize(cmds: TemplateCmd[]): any[][] {
 | 
						|
  var visitor = new CommandHumanizer();
 | 
						|
  visitAllCommands(visitor, cmds);
 | 
						|
  return visitor.result;
 | 
						|
}
 | 
						|
 | 
						|
function checkAndStringifyType(type: Type): string {
 | 
						|
  expect(isType(type)).toBe(true);
 | 
						|
  return `${stringify(type)}Type`;
 | 
						|
}
 | 
						|
 | 
						|
class CommandHumanizer implements CommandVisitor {
 | 
						|
  result: any[][] = [];
 | 
						|
  visitText(cmd: TextCmd, context: any): any {
 | 
						|
    this.result.push([TEXT, cmd.value, cmd.isBound, cmd.ngContentIndex]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitNgContent(cmd: NgContentCmd, context: any): any {
 | 
						|
    this.result.push([NG_CONTENT, cmd.ngContentIndex]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitBeginElement(cmd: BeginElementCmd, context: any): any {
 | 
						|
    this.result.push([
 | 
						|
      BEGIN_ELEMENT,
 | 
						|
      cmd.name,
 | 
						|
      cmd.attrNameAndValues,
 | 
						|
      cmd.eventNames,
 | 
						|
      cmd.variableNameAndValues,
 | 
						|
      cmd.directives.map(checkAndStringifyType),
 | 
						|
      cmd.isBound,
 | 
						|
      cmd.ngContentIndex
 | 
						|
    ]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitEndElement(context: any): any {
 | 
						|
    this.result.push([END_ELEMENT]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitBeginComponent(cmd: BeginComponentCmd, context: any): any {
 | 
						|
    this.result.push([
 | 
						|
      BEGIN_COMPONENT,
 | 
						|
      cmd.name,
 | 
						|
      cmd.attrNameAndValues,
 | 
						|
      cmd.eventNames,
 | 
						|
      cmd.variableNameAndValues,
 | 
						|
      cmd.directives.map(checkAndStringifyType),
 | 
						|
      cmd.nativeShadow,
 | 
						|
      cmd.ngContentIndex,
 | 
						|
      // TODO humanizeTemplate(cmd.template)
 | 
						|
      cmd.template.id
 | 
						|
    ]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitEndComponent(context: any): any {
 | 
						|
    this.result.push([END_COMPONENT]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
  visitEmbeddedTemplate(cmd: EmbeddedTemplateCmd, context: any): any {
 | 
						|
    this.result.push([
 | 
						|
      EMBEDDED_TEMPLATE,
 | 
						|
      cmd.attrNameAndValues,
 | 
						|
      cmd.variableNameAndValues,
 | 
						|
      cmd.directives.map(checkAndStringifyType),
 | 
						|
      cmd.isMerged,
 | 
						|
      cmd.ngContentIndex,
 | 
						|
      humanize(cmd.children)
 | 
						|
    ]);
 | 
						|
    return null;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function createTestableModule(source: SourceExpression): SourceModule {
 | 
						|
  var resultExpression = `${THIS_MODULE_REF}humanize(${source.expression})`;
 | 
						|
  var testableSource = `${source.declarations.join('\n')}
 | 
						|
  ${codeGenExportVariable('run')}${codeGenValueFn(['_'], resultExpression)};`;
 | 
						|
  return new SourceModule(null, testableSource);
 | 
						|
}
 |