BREACKING CHANGE: Before: @Directive({properties: ['one'], events: ['two']}) After: @Directive({inputs: ['one'], outputs: ['two']}) Before: @Component({properties: ['one'], events: ['two']}) After: @Componet({inputs: ['one'], outputs: ['two']}) Before: class A {@Property() one; @Event() two;} After: class A {@Input() one; @Output() two;}
216 lines
8.7 KiB
TypeScript
216 lines
8.7 KiB
TypeScript
import {ListWrapper} from 'angular2/src/core/facade/collection';
|
|
import {isPresent, isBlank} from 'angular2/src/core/facade/lang';
|
|
import {reflector} from 'angular2/src/core/reflection/reflection';
|
|
|
|
import {
|
|
ChangeDetection,
|
|
DirectiveIndex,
|
|
BindingRecord,
|
|
DirectiveRecord,
|
|
ProtoChangeDetector,
|
|
ChangeDetectionStrategy,
|
|
ChangeDetectorDefinition,
|
|
ChangeDetectorGenConfig,
|
|
ASTWithSource
|
|
} from 'angular2/src/core/change_detection/change_detection';
|
|
|
|
import {CompileDirectiveMetadata, CompileTypeMetadata} from './directive_metadata';
|
|
import {
|
|
TemplateAst,
|
|
ElementAst,
|
|
BoundTextAst,
|
|
PropertyBindingType,
|
|
DirectiveAst,
|
|
TemplateAstVisitor,
|
|
templateVisitAll,
|
|
NgContentAst,
|
|
EmbeddedTemplateAst,
|
|
VariableAst,
|
|
BoundElementPropertyAst,
|
|
BoundEventAst,
|
|
BoundDirectivePropertyAst,
|
|
AttrAst,
|
|
TextAst
|
|
} from './template_ast';
|
|
import {LifecycleHooks} from 'angular2/src/core/compiler/interfaces';
|
|
|
|
export function createChangeDetectorDefinitions(
|
|
componentType: CompileTypeMetadata, componentStrategy: ChangeDetectionStrategy,
|
|
genConfig: ChangeDetectorGenConfig, parsedTemplate: TemplateAst[]): ChangeDetectorDefinition[] {
|
|
var pvVisitors = [];
|
|
var visitor = new ProtoViewVisitor(null, pvVisitors, componentStrategy);
|
|
templateVisitAll(visitor, parsedTemplate);
|
|
return createChangeDefinitions(pvVisitors, componentType, genConfig);
|
|
}
|
|
|
|
class ProtoViewVisitor implements TemplateAstVisitor {
|
|
viewIndex: number;
|
|
boundTextCount: number = 0;
|
|
boundElementCount: number = 0;
|
|
variableNames: string[] = [];
|
|
bindingRecords: BindingRecord[] = [];
|
|
eventRecords: BindingRecord[] = [];
|
|
directiveRecords: DirectiveRecord[] = [];
|
|
|
|
constructor(public parent: ProtoViewVisitor, public allVisitors: ProtoViewVisitor[],
|
|
public strategy: ChangeDetectionStrategy) {
|
|
this.viewIndex = allVisitors.length;
|
|
allVisitors.push(this);
|
|
}
|
|
|
|
visitEmbeddedTemplate(ast: EmbeddedTemplateAst, context: any): any {
|
|
this.boundElementCount++;
|
|
for (var i = 0; i < ast.directives.length; i++) {
|
|
ast.directives[i].visit(this, i);
|
|
}
|
|
|
|
var childVisitor =
|
|
new ProtoViewVisitor(this, this.allVisitors, ChangeDetectionStrategy.Default);
|
|
// Attention: variables present on an embedded template count towards
|
|
// the embedded template and not the template anchor!
|
|
templateVisitAll(childVisitor, ast.vars);
|
|
templateVisitAll(childVisitor, ast.children);
|
|
return null;
|
|
}
|
|
|
|
visitElement(ast: ElementAst, context: any): any {
|
|
if (ast.isBound()) {
|
|
this.boundElementCount++;
|
|
}
|
|
templateVisitAll(this, ast.inputs, null);
|
|
templateVisitAll(this, ast.outputs);
|
|
templateVisitAll(this, ast.exportAsVars);
|
|
for (var i = 0; i < ast.directives.length; i++) {
|
|
ast.directives[i].visit(this, i);
|
|
}
|
|
templateVisitAll(this, ast.children);
|
|
return null;
|
|
}
|
|
|
|
visitNgContent(ast: NgContentAst, context: any): any { return null; }
|
|
|
|
visitVariable(ast: VariableAst, context: any): any {
|
|
this.variableNames.push(ast.name);
|
|
return null;
|
|
}
|
|
|
|
visitEvent(ast: BoundEventAst, directiveRecord: DirectiveRecord): any {
|
|
var bindingRecord =
|
|
isPresent(directiveRecord) ?
|
|
BindingRecord.createForHostEvent(ast.handler, ast.fullName, directiveRecord) :
|
|
BindingRecord.createForEvent(ast.handler, ast.fullName, this.boundElementCount - 1);
|
|
this.eventRecords.push(bindingRecord);
|
|
return null;
|
|
}
|
|
|
|
visitElementProperty(ast: BoundElementPropertyAst, directiveRecord: DirectiveRecord): any {
|
|
var boundElementIndex = this.boundElementCount - 1;
|
|
var dirIndex = isPresent(directiveRecord) ? directiveRecord.directiveIndex : null;
|
|
var bindingRecord;
|
|
if (ast.type === PropertyBindingType.Property) {
|
|
bindingRecord =
|
|
isPresent(dirIndex) ?
|
|
BindingRecord.createForHostProperty(dirIndex, ast.value, ast.name) :
|
|
BindingRecord.createForElementProperty(ast.value, boundElementIndex, ast.name);
|
|
} else if (ast.type === PropertyBindingType.Attribute) {
|
|
bindingRecord =
|
|
isPresent(dirIndex) ?
|
|
BindingRecord.createForHostAttribute(dirIndex, ast.value, ast.name) :
|
|
BindingRecord.createForElementAttribute(ast.value, boundElementIndex, ast.name);
|
|
} else if (ast.type === PropertyBindingType.Class) {
|
|
bindingRecord =
|
|
isPresent(dirIndex) ?
|
|
BindingRecord.createForHostClass(dirIndex, ast.value, ast.name) :
|
|
BindingRecord.createForElementClass(ast.value, boundElementIndex, ast.name);
|
|
} else if (ast.type === PropertyBindingType.Style) {
|
|
bindingRecord =
|
|
isPresent(dirIndex) ?
|
|
BindingRecord.createForHostStyle(dirIndex, ast.value, ast.name, ast.unit) :
|
|
BindingRecord.createForElementStyle(ast.value, boundElementIndex, ast.name, ast.unit);
|
|
}
|
|
this.bindingRecords.push(bindingRecord);
|
|
return null;
|
|
}
|
|
visitAttr(ast: AttrAst, context: any): any { return null; }
|
|
visitBoundText(ast: BoundTextAst, context: any): any {
|
|
var boundTextIndex = this.boundTextCount++;
|
|
this.bindingRecords.push(BindingRecord.createForTextNode(ast.value, boundTextIndex));
|
|
return null;
|
|
}
|
|
visitText(ast: TextAst, context: any): any { return null; }
|
|
visitDirective(ast: DirectiveAst, directiveIndexAsNumber: number): any {
|
|
var directiveIndex = new DirectiveIndex(this.boundElementCount - 1, directiveIndexAsNumber);
|
|
var directiveMetadata = ast.directive;
|
|
var directiveRecord = new DirectiveRecord({
|
|
directiveIndex: directiveIndex,
|
|
callAfterContentInit:
|
|
directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.AfterContentInit) !== -1,
|
|
callAfterContentChecked:
|
|
directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.AfterContentChecked) !== -1,
|
|
callAfterViewInit:
|
|
directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.AfterViewInit) !== -1,
|
|
callAfterViewChecked:
|
|
directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.AfterViewChecked) !== -1,
|
|
callOnChanges: directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.OnChanges) !== -1,
|
|
callDoCheck: directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.DoCheck) !== -1,
|
|
callOnInit: directiveMetadata.lifecycleHooks.indexOf(LifecycleHooks.OnInit) !== -1,
|
|
changeDetection: directiveMetadata.changeDetection
|
|
});
|
|
this.directiveRecords.push(directiveRecord);
|
|
|
|
templateVisitAll(this, ast.inputs, directiveRecord);
|
|
var bindingRecords = this.bindingRecords;
|
|
if (directiveRecord.callOnChanges) {
|
|
bindingRecords.push(BindingRecord.createDirectiveOnChanges(directiveRecord));
|
|
}
|
|
if (directiveRecord.callOnInit) {
|
|
bindingRecords.push(BindingRecord.createDirectiveOnInit(directiveRecord));
|
|
}
|
|
if (directiveRecord.callDoCheck) {
|
|
bindingRecords.push(BindingRecord.createDirectiveDoCheck(directiveRecord));
|
|
}
|
|
templateVisitAll(this, ast.hostProperties, directiveRecord);
|
|
templateVisitAll(this, ast.hostEvents, directiveRecord);
|
|
templateVisitAll(this, ast.exportAsVars);
|
|
return null;
|
|
}
|
|
visitDirectiveProperty(ast: BoundDirectivePropertyAst, directiveRecord: DirectiveRecord): any {
|
|
// TODO: these setters should eventually be created by change detection, to make
|
|
// it monomorphic!
|
|
var setter = reflector.setter(ast.directiveName);
|
|
this.bindingRecords.push(
|
|
BindingRecord.createForDirective(ast.value, ast.directiveName, setter, directiveRecord));
|
|
return null;
|
|
}
|
|
}
|
|
|
|
|
|
function createChangeDefinitions(pvVisitors: ProtoViewVisitor[], componentType: CompileTypeMetadata,
|
|
genConfig: ChangeDetectorGenConfig): ChangeDetectorDefinition[] {
|
|
var pvVariableNames = _collectNestedProtoViewsVariableNames(pvVisitors);
|
|
return pvVisitors.map(pvVisitor => {
|
|
var viewType = pvVisitor.viewIndex === 0 ? 'component' : 'embedded';
|
|
var id = _protoViewId(componentType, pvVisitor.viewIndex, viewType);
|
|
return new ChangeDetectorDefinition(
|
|
id, pvVisitor.strategy, pvVariableNames[pvVisitor.viewIndex], pvVisitor.bindingRecords,
|
|
pvVisitor.eventRecords, pvVisitor.directiveRecords, genConfig);
|
|
|
|
});
|
|
}
|
|
|
|
function _collectNestedProtoViewsVariableNames(pvVisitors: ProtoViewVisitor[]): string[][] {
|
|
var nestedPvVariableNames: string[][] = ListWrapper.createFixedSize(pvVisitors.length);
|
|
pvVisitors.forEach((pv) => {
|
|
var parentVariableNames: string[] =
|
|
isPresent(pv.parent) ? nestedPvVariableNames[pv.parent.viewIndex] : [];
|
|
nestedPvVariableNames[pv.viewIndex] = parentVariableNames.concat(pv.variableNames);
|
|
});
|
|
return nestedPvVariableNames;
|
|
}
|
|
|
|
|
|
function _protoViewId(hostComponentType: CompileTypeMetadata, pvIndex: number, viewType: string):
|
|
string {
|
|
return `${hostComponentType.name}_${viewType}_${pvIndex}`;
|
|
}
|