490 lines
17 KiB
TypeScript
490 lines
17 KiB
TypeScript
|
import {
|
||
|
ddescribe,
|
||
|
describe,
|
||
|
xdescribe,
|
||
|
it,
|
||
|
iit,
|
||
|
xit,
|
||
|
expect,
|
||
|
beforeEach,
|
||
|
afterEach,
|
||
|
AsyncTestCompleter,
|
||
|
inject
|
||
|
} from 'angular2/test_lib';
|
||
|
|
||
|
import {IS_DART} from '../platform';
|
||
|
import {CONST_EXPR, stringify, isType, Type, isBlank} from 'angular2/src/core/facade/lang';
|
||
|
import {PromiseWrapper, Promise} from 'angular2/src/core/facade/async';
|
||
|
import {HtmlParser} from 'angular2/src/compiler/html_parser';
|
||
|
import {TemplateParser} from 'angular2/src/compiler/template_parser';
|
||
|
import {MockSchemaRegistry} from './template_parser_spec';
|
||
|
import {Parser, Lexer} from 'angular2/src/core/change_detection/change_detection';
|
||
|
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 {
|
||
|
DirectiveMetadata,
|
||
|
TypeMetadata,
|
||
|
TemplateMetadata,
|
||
|
SourceModule
|
||
|
} from 'angular2/src/compiler/api';
|
||
|
import {ViewEncapsulation} from 'angular2/src/core/render/api';
|
||
|
import {evalModule} from './eval_module';
|
||
|
import {escapeSingleQuoteString} from 'angular2/src/compiler/util';
|
||
|
|
||
|
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 MODULE_NAME = 'angular2/test/compiler/command_compiler_spec';
|
||
|
const TEMPLATE_COMMANDS_MODULE_NAME = 'angular2/src/core/compiler/template_commands';
|
||
|
|
||
|
// Attention: read by eval!
|
||
|
export class RootComp {}
|
||
|
export class SomeDir {}
|
||
|
export class AComp {}
|
||
|
|
||
|
var RootCompTypeMeta =
|
||
|
new TypeMetadata({typeName: 'RootComp', id: 1, type: RootComp, typeUrl: MODULE_NAME});
|
||
|
var SomeDirTypeMeta =
|
||
|
new TypeMetadata({typeName: 'SomeDir', id: 2, type: SomeDir, typeUrl: MODULE_NAME});
|
||
|
var ACompTypeMeta = new TypeMetadata({typeName: 'AComp', id: 3, type: AComp, typeUrl: MODULE_NAME});
|
||
|
|
||
|
var NESTED_COMPONENT = new CompiledTemplate('someNestedComponentId', []);
|
||
|
|
||
|
export function main() {
|
||
|
describe('CommandCompiler', () => {
|
||
|
var domParser: HtmlParser;
|
||
|
var parser: TemplateParser;
|
||
|
var commandCompiler: CommandCompiler;
|
||
|
var componentTemplateFactory: Function;
|
||
|
|
||
|
beforeEach(() => {
|
||
|
domParser = new HtmlParser();
|
||
|
parser = new TemplateParser(
|
||
|
new Parser(new Lexer()),
|
||
|
new MockSchemaRegistry({'invalidProp': false}, {'mappedAttr': 'mappedProp'}));
|
||
|
commandCompiler = new CommandCompiler();
|
||
|
});
|
||
|
|
||
|
function createComp({type, selector, template, encapsulation, ngContentSelectors}: {
|
||
|
type?: TypeMetadata,
|
||
|
selector?: string,
|
||
|
template?: string,
|
||
|
encapsulation?: ViewEncapsulation,
|
||
|
ngContentSelectors?: string[]
|
||
|
}): DirectiveMetadata {
|
||
|
if (isBlank(encapsulation)) {
|
||
|
encapsulation = ViewEncapsulation.None;
|
||
|
}
|
||
|
if (isBlank(selector)) {
|
||
|
selector = 'root';
|
||
|
}
|
||
|
if (isBlank(ngContentSelectors)) {
|
||
|
ngContentSelectors = [];
|
||
|
}
|
||
|
if (isBlank(template)) {
|
||
|
template = '';
|
||
|
}
|
||
|
return new DirectiveMetadata({
|
||
|
selector: selector,
|
||
|
isComponent: true,
|
||
|
type: type,
|
||
|
template: new TemplateMetadata({
|
||
|
template: template,
|
||
|
ngContentSelectors: ngContentSelectors,
|
||
|
encapsulation: encapsulation
|
||
|
})
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function createDirective(type: TypeMetadata, selector: string): DirectiveMetadata {
|
||
|
return new DirectiveMetadata({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,
|
||
|
'AComp'
|
||
|
],
|
||
|
[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,
|
||
|
'AComp'
|
||
|
],
|
||
|
[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, 'AComp'],
|
||
|
[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, 'AComp'],
|
||
|
[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 = (directiveType: TypeMetadata) => {
|
||
|
return new CompiledTemplate(directiveType.typeName, []);
|
||
|
};
|
||
|
});
|
||
|
|
||
|
function run(component: DirectiveMetadata, directives: DirectiveMetadata[]):
|
||
|
Promise<any[][]> {
|
||
|
var parsedTemplate = parser.parse(
|
||
|
domParser.parse(component.template.template, component.type.typeName), directives);
|
||
|
var commands = commandCompiler.compileComponentRuntime(component, parsedTemplate,
|
||
|
componentTemplateFactory);
|
||
|
return PromiseWrapper.resolve(humanize(commands));
|
||
|
}
|
||
|
|
||
|
createTests(run);
|
||
|
});
|
||
|
|
||
|
|
||
|
describe('compileComponentCodeGen', () => {
|
||
|
beforeEach(() => {
|
||
|
componentTemplateFactory = (directiveType: TypeMetadata, imports: string[][]) => {
|
||
|
imports.push([TEMPLATE_COMMANDS_MODULE_NAME, 'tcm']);
|
||
|
return `new tcm.CompiledTemplate(${escapeSingleQuoteString(directiveType.typeName)}, [])`;
|
||
|
};
|
||
|
});
|
||
|
|
||
|
function run(component: DirectiveMetadata, directives: DirectiveMetadata[]):
|
||
|
Promise<any[][]> {
|
||
|
var parsedTemplate = parser.parse(
|
||
|
domParser.parse(component.template.template, component.type.typeName), directives);
|
||
|
var sourceModule = commandCompiler.compileComponentCodeGen(component, parsedTemplate,
|
||
|
componentTemplateFactory);
|
||
|
var testableModule = createTestableModule(sourceModule);
|
||
|
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,
|
||
|
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(sourceModule: SourceModule): SourceModule {
|
||
|
var testableSource;
|
||
|
var testableImports = [[MODULE_NAME, 'mocks']].concat(sourceModule.imports);
|
||
|
if (IS_DART) {
|
||
|
testableSource = `${sourceModule.source}
|
||
|
run(_) { return mocks.humanize(COMMANDS); }`;
|
||
|
} else {
|
||
|
testableSource = `${sourceModule.source}
|
||
|
exports.run = function(_) { return mocks.humanize(COMMANDS); }`;
|
||
|
}
|
||
|
return new SourceModule(null, testableSource, testableImports);
|
||
|
}
|