2017-11-20 10:21:17 -08: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
|
|
|
|
|
*/
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
import {CompileDirectiveMetadata, CompilePipeSummary, CompileTokenMetadata, CompileTypeMetadata, flatten, identifierName, rendererTypeName, tokenReference, viewClassName} from '../compile_metadata';
|
2017-11-20 10:21:17 -08:00
|
|
|
|
import {CompileReflector} from '../compile_reflector';
|
|
|
|
|
import {BindingForm, BuiltinConverter, EventHandlerVars, LocalResolver, convertActionBinding, convertPropertyBinding, convertPropertyBindingBuiltins} from '../compiler_util/expression_converter';
|
|
|
|
|
import {ConstantPool, DefinitionKind} from '../constant_pool';
|
|
|
|
|
import {AST} from '../expression_parser/ast';
|
|
|
|
|
import {Identifiers} from '../identifiers';
|
|
|
|
|
import * as o from '../output/output_ast';
|
|
|
|
|
import {ParseSourceSpan} from '../parse_util';
|
|
|
|
|
import {CssSelector} from '../selector';
|
2018-01-26 17:12:39 -08:00
|
|
|
|
import {AttrAst, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, DirectiveAst, ElementAst, EmbeddedTemplateAst, NgContentAst, PropertyBindingType, ProviderAst, QueryMatch, RecursiveTemplateAstVisitor, ReferenceAst, TemplateAst, TemplateAstVisitor, TextAst, VariableAst, templateVisitAll} from '../template_parser/template_ast';
|
2017-11-20 10:21:17 -08:00
|
|
|
|
import {OutputContext, error} from '../util';
|
|
|
|
|
|
|
|
|
|
import {Identifiers as R3} from './r3_identifiers';
|
|
|
|
|
|
2018-01-26 17:12:39 -08:00
|
|
|
|
|
|
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
|
/** Name of the context parameter passed into a template function */
|
|
|
|
|
const CONTEXT_NAME = 'ctx';
|
|
|
|
|
|
|
|
|
|
/** Name of the creation mode flag passed into a template function */
|
|
|
|
|
const CREATION_MODE_FLAG = 'cm';
|
|
|
|
|
|
|
|
|
|
/** Name of the temporary to use during data binding */
|
|
|
|
|
const TEMPORARY_NAME = '_t';
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
/** The prefix reference variables */
|
|
|
|
|
const REFERENCE_PREFIX = '_r';
|
|
|
|
|
|
2018-01-24 11:19:39 -08:00
|
|
|
|
/** The name of the implicit context reference */
|
|
|
|
|
const IMPLICIT_REFERENCE = '$implicit';
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
export function compileDirective(
|
|
|
|
|
outputCtx: OutputContext, directive: CompileDirectiveMetadata, reflector: CompileReflector) {
|
|
|
|
|
const definitionMapValues: {key: string, quoted: boolean, value: o.Expression}[] = [];
|
|
|
|
|
|
2018-01-25 13:20:42 -08:00
|
|
|
|
// e.g. 'type: MyDirective`
|
|
|
|
|
definitionMapValues.push(
|
|
|
|
|
{key: 'type', value: outputCtx.importExpr(directive.type.reference), quoted: false});
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// e.g. `factory: () => new MyApp(injectElementRef())`
|
|
|
|
|
const templateFactory = createFactory(directive.type, outputCtx, reflector);
|
|
|
|
|
definitionMapValues.push({key: 'factory', value: templateFactory, quoted: false});
|
|
|
|
|
|
|
|
|
|
const className = identifierName(directive.type) !;
|
|
|
|
|
className || error(`Cannot resolver the name of ${directive.type}`);
|
|
|
|
|
|
|
|
|
|
// Create the partial class to be merged with the actual class.
|
|
|
|
|
outputCtx.statements.push(new o.ClassStmt(
|
|
|
|
|
/* name */ className,
|
|
|
|
|
/* parent */ null,
|
|
|
|
|
/* fields */[new o.ClassField(
|
|
|
|
|
/* name */ 'ngDirectiveDef',
|
|
|
|
|
/* type */ o.INFERRED_TYPE,
|
|
|
|
|
/* modifiers */[o.StmtModifier.Static],
|
|
|
|
|
/* initializer */ o.importExpr(R3.defineDirective).callFn([o.literalMap(
|
|
|
|
|
definitionMapValues)]))],
|
|
|
|
|
/* getters */[],
|
|
|
|
|
/* constructorMethod */ new o.ClassMethod(null, [], []),
|
|
|
|
|
/* methods */[]));
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
|
export function compileComponent(
|
|
|
|
|
outputCtx: OutputContext, component: CompileDirectiveMetadata, template: TemplateAst[],
|
|
|
|
|
reflector: CompileReflector) {
|
|
|
|
|
const definitionMapValues: {key: string, quoted: boolean, value: o.Expression}[] = [];
|
|
|
|
|
|
2018-01-25 13:20:42 -08:00
|
|
|
|
// e.g. `type: MyApp`
|
|
|
|
|
definitionMapValues.push(
|
|
|
|
|
{key: 'type', value: outputCtx.importExpr(component.type.reference), quoted: false});
|
|
|
|
|
|
|
|
|
|
// e.g. `tag: 'my-app'`
|
2017-11-20 10:21:17 -08:00
|
|
|
|
// This is optional and only included if the first selector of a component has element.
|
|
|
|
|
const selector = component.selector && CssSelector.parse(component.selector);
|
|
|
|
|
const firstSelector = selector && selector[0];
|
|
|
|
|
if (firstSelector && firstSelector.hasElementSelector()) {
|
|
|
|
|
definitionMapValues.push({key: 'tag', value: o.literal(firstSelector.element), quoted: false});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// e.g. `attr: ["class", ".my.app"]
|
|
|
|
|
// This is optional an only included if the first selector of a component specifies attributes.
|
|
|
|
|
if (firstSelector) {
|
|
|
|
|
const selectorAttributes = firstSelector.getAttrs();
|
|
|
|
|
if (selectorAttributes.length) {
|
|
|
|
|
definitionMapValues.push({
|
|
|
|
|
key: 'attrs',
|
2018-01-11 15:37:56 -08:00
|
|
|
|
value: outputCtx.constantPool.getConstLiteral(
|
|
|
|
|
o.literalArr(selectorAttributes.map(
|
|
|
|
|
value => value != null ? o.literal(value) : o.literal(undefined))),
|
|
|
|
|
/* forceShared */ true),
|
2017-11-20 10:21:17 -08:00
|
|
|
|
quoted: false
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-22 15:35:18 -08:00
|
|
|
|
// e.g. `factory: function MyApp_Factory() { return new MyApp(injectElementRef()); }`
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const templateFactory = createFactory(component.type, outputCtx, reflector);
|
|
|
|
|
definitionMapValues.push({key: 'factory', value: templateFactory, quoted: false});
|
|
|
|
|
|
2018-01-22 15:35:18 -08:00
|
|
|
|
// e.g. `template: function MyComponent_Template(_ctx, _cm) {...}`
|
|
|
|
|
const templateTypeName = component.type.reference.name;
|
|
|
|
|
const templateName = templateTypeName ? `${templateTypeName}_Template` : null;
|
2017-11-20 10:21:17 -08:00
|
|
|
|
const templateFunctionExpression =
|
2018-01-11 15:37:56 -08:00
|
|
|
|
new TemplateDefinitionBuilder(
|
2018-01-22 15:35:18 -08:00
|
|
|
|
outputCtx, outputCtx.constantPool, reflector, CONTEXT_NAME, ROOT_SCOPE.nestedScope(), 0,
|
2018-01-26 17:12:39 -08:00
|
|
|
|
component.template !.ngContentSelectors, templateTypeName, templateName)
|
2018-01-24 11:19:39 -08:00
|
|
|
|
.buildTemplateFunction(template, []);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
definitionMapValues.push({key: 'template', value: templateFunctionExpression, quoted: false});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const className = identifierName(component.type) !;
|
|
|
|
|
className || error(`Cannot resolver the name of ${component.type}`);
|
|
|
|
|
|
|
|
|
|
// Create the partial class to be merged with the actual class.
|
|
|
|
|
outputCtx.statements.push(new o.ClassStmt(
|
|
|
|
|
/* name */ className,
|
|
|
|
|
/* parent */ null,
|
|
|
|
|
/* fields */[new o.ClassField(
|
|
|
|
|
/* name */ 'ngComponentDef',
|
|
|
|
|
/* type */ o.INFERRED_TYPE,
|
|
|
|
|
/* modifiers */[o.StmtModifier.Static],
|
|
|
|
|
/* initializer */ o.importExpr(R3.defineComponent).callFn([o.literalMap(
|
|
|
|
|
definitionMapValues)]))],
|
|
|
|
|
/* getters */[],
|
|
|
|
|
/* constructorMethod */ new o.ClassMethod(null, [], []),
|
|
|
|
|
/* methods */[]));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// TODO: Remove these when the things are fully supported
|
|
|
|
|
function unknown<T>(arg: o.Expression | o.Statement | TemplateAst): never {
|
2018-01-26 17:12:39 -08:00
|
|
|
|
throw new Error(
|
|
|
|
|
`Builder ${this.constructor.name} is unable to handle ${arg.constructor.name} yet`);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
2018-01-26 17:12:39 -08:00
|
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
|
function unsupported(feature: string): never {
|
|
|
|
|
if (this) {
|
|
|
|
|
throw new Error(`Builder ${this.constructor.name} doesn't support ${feature} yet`);
|
|
|
|
|
}
|
2018-01-25 15:38:39 -08:00
|
|
|
|
throw new Error(`Feature ${feature} is not supported yet`);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const BINDING_INSTRUCTION_MAP: {[index: number]: o.ExternalReference | undefined} = {
|
|
|
|
|
[PropertyBindingType.Property]: R3.elementProperty,
|
|
|
|
|
[PropertyBindingType.Attribute]: R3.elementAttribute,
|
|
|
|
|
[PropertyBindingType.Class]: R3.elementClass,
|
|
|
|
|
[PropertyBindingType.Style]: R3.elementStyle
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function interpolate(args: o.Expression[]): o.Expression {
|
|
|
|
|
args = args.slice(1); // Ignore the length prefix added for render2
|
|
|
|
|
switch (args.length) {
|
|
|
|
|
case 3:
|
|
|
|
|
return o.importExpr(R3.bind1).callFn(args);
|
|
|
|
|
case 5:
|
|
|
|
|
return o.importExpr(R3.bind2).callFn(args);
|
|
|
|
|
case 7:
|
|
|
|
|
return o.importExpr(R3.bind3).callFn(args);
|
|
|
|
|
case 9:
|
|
|
|
|
return o.importExpr(R3.bind4).callFn(args);
|
|
|
|
|
case 11:
|
|
|
|
|
return o.importExpr(R3.bind5).callFn(args);
|
|
|
|
|
case 13:
|
|
|
|
|
return o.importExpr(R3.bind6).callFn(args);
|
|
|
|
|
case 15:
|
|
|
|
|
return o.importExpr(R3.bind7).callFn(args);
|
|
|
|
|
case 17:
|
|
|
|
|
return o.importExpr(R3.bind8).callFn(args);
|
|
|
|
|
}
|
|
|
|
|
(args.length > 19 && args.length % 2 == 1) ||
|
|
|
|
|
error(`Invalid interpolation argument length ${args.length}`);
|
|
|
|
|
return o.importExpr(R3.bindV).callFn(args);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
class BindingScope {
|
|
|
|
|
private map = new Map<string, o.Expression>();
|
|
|
|
|
private referenceNameIndex = 0;
|
|
|
|
|
|
|
|
|
|
constructor(private parent: BindingScope|null) {}
|
|
|
|
|
|
|
|
|
|
get(name: string): o.Expression|null {
|
|
|
|
|
let current: BindingScope|null = this;
|
|
|
|
|
while (current) {
|
|
|
|
|
const value = current.map.get(name);
|
|
|
|
|
if (value != null) {
|
|
|
|
|
// Cache the value locally.
|
|
|
|
|
this.map.set(name, value);
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
current = current.parent;
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set(name: string, variableName: string): BindingScope {
|
|
|
|
|
!this.map.has(name) ||
|
|
|
|
|
error(`The name ${name} is already defined in scope to be ${this.map.get(name)}`);
|
|
|
|
|
this.map.set(name, o.variable(variableName));
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nestedScope(): BindingScope { return new BindingScope(this); }
|
|
|
|
|
|
|
|
|
|
freshReferenceName(): string {
|
|
|
|
|
let current: BindingScope|null = this;
|
|
|
|
|
// Find the top scope as it maintains the global reference count
|
|
|
|
|
while (current.parent) current = current.parent;
|
|
|
|
|
return `${REFERENCE_PREFIX}${current.referenceNameIndex++}`;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const ROOT_SCOPE = new BindingScope(null).set('$event', '$event');
|
|
|
|
|
|
|
|
|
|
class TemplateDefinitionBuilder implements TemplateAstVisitor, LocalResolver {
|
2017-11-20 10:21:17 -08:00
|
|
|
|
private _dataIndex = 0;
|
|
|
|
|
private _bindingContext = 0;
|
2018-01-11 15:37:56 -08:00
|
|
|
|
private _referenceIndex = 0;
|
2017-11-20 10:21:17 -08:00
|
|
|
|
private _temporaryAllocated = false;
|
|
|
|
|
private _prefix: o.Statement[] = [];
|
|
|
|
|
private _creationMode: o.Statement[] = [];
|
|
|
|
|
private _bindingMode: o.Statement[] = [];
|
|
|
|
|
private _hostMode: o.Statement[] = [];
|
|
|
|
|
private _refreshMode: o.Statement[] = [];
|
|
|
|
|
private _postfix: o.Statement[] = [];
|
2018-01-26 17:12:39 -08:00
|
|
|
|
private _contentProjections: Map<NgContentAst, NgContentInfo>;
|
|
|
|
|
private _projectionDefinitionIndex = 0;
|
2017-11-20 10:21:17 -08:00
|
|
|
|
private unsupported = unsupported;
|
|
|
|
|
private invalid = invalid;
|
|
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
|
private outputCtx: OutputContext, private constantPool: ConstantPool,
|
2018-01-22 15:35:18 -08:00
|
|
|
|
private reflector: CompileReflector, private contextParameter: string,
|
2018-01-26 17:12:39 -08:00
|
|
|
|
private bindingScope: BindingScope, private level = 0, private ngContentSelectors: string[],
|
|
|
|
|
private contextName: string|null, private templateName: string|null) {}
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
2018-01-24 11:19:39 -08:00
|
|
|
|
buildTemplateFunction(asts: TemplateAst[], variables: VariableAst[]): o.FunctionExpr {
|
|
|
|
|
// Create variable bindings
|
|
|
|
|
for (const variable of variables) {
|
|
|
|
|
const variableName = variable.name;
|
|
|
|
|
const expression =
|
|
|
|
|
o.variable(this.contextParameter).prop(variable.value || IMPLICIT_REFERENCE);
|
|
|
|
|
const scopedName = this.bindingScope.freshReferenceName();
|
|
|
|
|
const declaration = o.variable(scopedName).set(expression).toDeclStmt(o.INFERRED_TYPE, [
|
|
|
|
|
o.StmtModifier.Final
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
// Add the reference to the local scope.
|
|
|
|
|
this.bindingScope.set(variableName, scopedName);
|
|
|
|
|
|
|
|
|
|
// Declare the local variable in binding mode
|
|
|
|
|
this._bindingMode.push(declaration);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-26 17:12:39 -08:00
|
|
|
|
// Collect content projections
|
|
|
|
|
if (this.ngContentSelectors && this.ngContentSelectors.length > 0) {
|
|
|
|
|
const contentProjections = getContentProjection(asts, this.ngContentSelectors);
|
|
|
|
|
this._contentProjections = contentProjections;
|
|
|
|
|
if (contentProjections.size > 0) {
|
|
|
|
|
const infos: R3CssSelector[] = [];
|
|
|
|
|
Array.from(contentProjections.values()).forEach(info => {
|
|
|
|
|
if (info.selector) {
|
|
|
|
|
infos[info.index - 1] = info.selector;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
const projectionIndex = this._projectionDefinitionIndex = this.allocateDataSlot();
|
|
|
|
|
const parameters: o.Expression[] = [o.literal(projectionIndex)];
|
|
|
|
|
!infos.some(value => !value) || error(`content project information skipped an index`);
|
|
|
|
|
if (infos.length > 1) {
|
|
|
|
|
parameters.push(this.outputCtx.constantPool.getConstLiteral(
|
|
|
|
|
asLiteral(infos), /* forceShared */ true));
|
|
|
|
|
}
|
|
|
|
|
this.instruction(this._creationMode, null, R3.projectionDef, ...parameters);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
|
templateVisitAll(this, asts);
|
|
|
|
|
|
|
|
|
|
return o.fn(
|
|
|
|
|
[
|
|
|
|
|
new o.FnParam(this.contextParameter, null), new o.FnParam(CREATION_MODE_FLAG, o.BOOL_TYPE)
|
|
|
|
|
],
|
|
|
|
|
[
|
|
|
|
|
// Temporary variable declarations (i.e. let _t: any;)
|
|
|
|
|
...this._prefix,
|
|
|
|
|
|
|
|
|
|
// Creating mode (i.e. if (cm) { ... })
|
|
|
|
|
o.ifStmt(o.variable(CREATION_MODE_FLAG), this._creationMode),
|
|
|
|
|
|
|
|
|
|
// Binding mode (i.e. ɵp(...))
|
|
|
|
|
...this._bindingMode,
|
|
|
|
|
|
|
|
|
|
// Host mode (i.e. Comp.h(...))
|
|
|
|
|
...this._hostMode,
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// Refresh mode (i.e. Comp.r(...))
|
2017-11-20 10:21:17 -08:00
|
|
|
|
...this._refreshMode,
|
|
|
|
|
|
|
|
|
|
// Nested templates (i.e. function CompTemplate() {})
|
|
|
|
|
...this._postfix
|
|
|
|
|
],
|
2018-01-22 15:35:18 -08:00
|
|
|
|
o.INFERRED_TYPE, null, this.templateName);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
getLocal(name: string): o.Expression|null { return this.bindingScope.get(name); }
|
|
|
|
|
|
2018-01-26 17:12:39 -08:00
|
|
|
|
visitNgContent(ast: NgContentAst) {
|
|
|
|
|
const info = this._contentProjections.get(ast) !;
|
|
|
|
|
info || error(`Expected ${ast.sourceSpan} to be included in content projection collection`);
|
|
|
|
|
const slot = this.allocateDataSlot();
|
|
|
|
|
const parameters = [o.literal(slot), o.literal(this._projectionDefinitionIndex)];
|
|
|
|
|
if (info.index !== 0) {
|
|
|
|
|
parameters.push(o.literal(info.index));
|
|
|
|
|
}
|
|
|
|
|
this.instruction(this._creationMode, ast.sourceSpan, R3.projection, ...parameters);
|
|
|
|
|
}
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
private _computeDirectivesArray(directives: DirectiveAst[]) {
|
|
|
|
|
const directiveIndexMap = new Map<any, number>();
|
|
|
|
|
const directiveExpressions: o.Expression[] =
|
|
|
|
|
directives.filter(directive => !directive.directive.isComponent).map(directive => {
|
|
|
|
|
directiveIndexMap.set(directive.directive.type.reference, this.allocateDataSlot());
|
|
|
|
|
return this.typeReference(directive.directive.type.reference);
|
|
|
|
|
});
|
|
|
|
|
return {
|
|
|
|
|
directivesArray: directiveExpressions.length ?
|
|
|
|
|
this.constantPool.getConstLiteral(
|
|
|
|
|
o.literalArr(directiveExpressions), /* forceShared */ true) :
|
|
|
|
|
o.literal(null),
|
|
|
|
|
directiveIndexMap
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
|
visitElement(ast: ElementAst) {
|
|
|
|
|
let bindingCount = 0;
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const elementIndex = this.allocateDataSlot();
|
|
|
|
|
let componentIndex: number|undefined = undefined;
|
|
|
|
|
const referenceDataSlots = new Map<string, number>();
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// Element creation mode
|
|
|
|
|
const component = findComponent(ast.directives);
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const nullNode = o.literal(null, o.INFERRED_TYPE);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
const parameters: o.Expression[] = [o.literal(elementIndex)];
|
2018-01-11 15:37:56 -08:00
|
|
|
|
|
|
|
|
|
// Add component type or element tag
|
2017-11-20 10:21:17 -08:00
|
|
|
|
if (component) {
|
|
|
|
|
parameters.push(this.typeReference(component.directive.type.reference));
|
2018-01-11 15:37:56 -08:00
|
|
|
|
componentIndex = this.allocateDataSlot();
|
2017-11-20 10:21:17 -08:00
|
|
|
|
} else {
|
|
|
|
|
parameters.push(o.literal(ast.name));
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// Add attributes array
|
2017-11-20 10:21:17 -08:00
|
|
|
|
const attributes: o.Expression[] = [];
|
|
|
|
|
for (let attr of ast.attrs) {
|
|
|
|
|
attributes.push(o.literal(attr.name), o.literal(attr.value));
|
|
|
|
|
}
|
2018-01-11 15:37:56 -08:00
|
|
|
|
parameters.push(
|
|
|
|
|
attributes.length > 0 ?
|
|
|
|
|
this.constantPool.getConstLiteral(o.literalArr(attributes), /* forceShared */ true) :
|
|
|
|
|
nullNode);
|
|
|
|
|
|
|
|
|
|
// Add directives array
|
|
|
|
|
const {directivesArray, directiveIndexMap} = this._computeDirectivesArray(ast.directives);
|
|
|
|
|
parameters.push(directiveIndexMap.size > 0 ? directivesArray : nullNode);
|
|
|
|
|
|
|
|
|
|
if (component && componentIndex != null) {
|
|
|
|
|
// Record the data slot for the component
|
|
|
|
|
directiveIndexMap.set(component.directive.type.reference, componentIndex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add references array
|
|
|
|
|
if (ast.references && ast.references.length > 0) {
|
|
|
|
|
const references =
|
|
|
|
|
flatten(ast.references.map(reference => {
|
|
|
|
|
const slot = this.allocateDataSlot();
|
|
|
|
|
referenceDataSlots.set(reference.name, slot);
|
|
|
|
|
// Generate the update temporary.
|
|
|
|
|
const variableName = this.bindingScope.freshReferenceName();
|
|
|
|
|
this._bindingMode.push(o.variable(variableName, o.INFERRED_TYPE)
|
|
|
|
|
.set(o.importExpr(R3.memory).callFn([o.literal(slot)]))
|
|
|
|
|
.toDeclStmt(o.INFERRED_TYPE, [o.StmtModifier.Final]));
|
|
|
|
|
this.bindingScope.set(reference.name, variableName);
|
|
|
|
|
return [reference.name, reference.originalValue];
|
|
|
|
|
})).map(value => o.literal(value));
|
|
|
|
|
parameters.push(
|
|
|
|
|
this.constantPool.getConstLiteral(o.literalArr(references), /* forceShared */ true));
|
|
|
|
|
} else {
|
|
|
|
|
parameters.push(nullNode);
|
|
|
|
|
}
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// Remove trailing null nodes as they are implied.
|
|
|
|
|
while (parameters[parameters.length - 1] === nullNode) {
|
|
|
|
|
parameters.pop();
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// Generate the instruction create element instruction
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this.instruction(this._creationMode, ast.sourceSpan, R3.createElement, ...parameters);
|
|
|
|
|
|
|
|
|
|
const implicit = o.variable(this.contextParameter);
|
|
|
|
|
|
|
|
|
|
// Generate element input bindings
|
|
|
|
|
for (let input of ast.inputs) {
|
|
|
|
|
if (input.isAnimation) {
|
|
|
|
|
this.unsupported('animations');
|
|
|
|
|
}
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// TODO(chuckj): Built-in transform?
|
2017-11-20 10:21:17 -08:00
|
|
|
|
const convertedBinding = convertPropertyBinding(
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this, implicit, input.value, this.bindingContext(), BindingForm.TrySimple, interpolate);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this._bindingMode.push(...convertedBinding.stmts);
|
|
|
|
|
const parameters =
|
|
|
|
|
[o.literal(elementIndex), o.literal(input.name), convertedBinding.currValExpr];
|
|
|
|
|
const instruction = BINDING_INSTRUCTION_MAP[input.type];
|
|
|
|
|
if (instruction) {
|
|
|
|
|
// TODO(chuckj): runtime: security context?
|
|
|
|
|
this.instruction(
|
|
|
|
|
this._bindingMode, input.sourceSpan, instruction, o.literal(elementIndex),
|
|
|
|
|
o.literal(input.name), convertedBinding.currValExpr);
|
|
|
|
|
} else {
|
|
|
|
|
this.unsupported(`binding ${PropertyBindingType[input.type]}`);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Generate directives input bindings
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this._visitDirectives(ast.directives, implicit, elementIndex, directiveIndexMap);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// Traverse element child nodes
|
|
|
|
|
templateVisitAll(this, ast.children);
|
|
|
|
|
|
|
|
|
|
// Finish element construction mode.
|
|
|
|
|
this.instruction(this._creationMode, ast.endSourceSpan || ast.sourceSpan, R3.elementEnd);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
private _visitDirectives(
|
|
|
|
|
directives: DirectiveAst[], implicit: o.Expression, nodeIndex: number,
|
|
|
|
|
directiveIndexMap: Map<any, number>) {
|
2017-11-20 10:21:17 -08:00
|
|
|
|
for (let directive of directives) {
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const directiveIndex = directiveIndexMap.get(directive.directive.type.reference);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// Creation mode
|
|
|
|
|
// e.g. D(0, TodoComponentDef.n(), TodoComponentDef);
|
|
|
|
|
const directiveType = directive.directive.type.reference;
|
|
|
|
|
const kind =
|
|
|
|
|
directive.directive.isComponent ? DefinitionKind.Component : DefinitionKind.Directive;
|
|
|
|
|
|
|
|
|
|
// Note: *do not cache* calls to this.directiveOf() as the constant pool needs to know if the
|
|
|
|
|
// node is referenced multiple times to know that it must generate the reference into a
|
|
|
|
|
// temporary.
|
|
|
|
|
|
|
|
|
|
// Bindings
|
|
|
|
|
for (const input of directive.inputs) {
|
|
|
|
|
const convertedBinding = convertPropertyBinding(
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this, implicit, input.value, this.bindingContext(), BindingForm.TrySimple, interpolate);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this._bindingMode.push(...convertedBinding.stmts);
|
|
|
|
|
this.instruction(
|
2018-01-24 11:19:39 -08:00
|
|
|
|
this._bindingMode, directive.sourceSpan, R3.elementProperty, o.literal(nodeIndex),
|
|
|
|
|
o.literal(input.templateName),
|
|
|
|
|
o.importExpr(R3.bind).callFn([convertedBinding.currValExpr]));
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// e.g. TodoComponentDef.r(0, 0);
|
|
|
|
|
this._refreshMode.push(
|
|
|
|
|
this.definitionOf(directiveType, kind)
|
|
|
|
|
.callMethod(R3.REFRESH_METHOD, [o.literal(directiveIndex), o.literal(nodeIndex)])
|
|
|
|
|
.toStmt());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
visitEmbeddedTemplate(ast: EmbeddedTemplateAst) {
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const templateIndex = this.allocateDataSlot();
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
2018-01-22 15:35:18 -08:00
|
|
|
|
const templateRef = this.reflector.resolveExternalReference(Identifiers.TemplateRef);
|
|
|
|
|
const templateDirective = ast.directives.find(
|
|
|
|
|
directive => directive.directive.type.diDeps.some(
|
|
|
|
|
dependency =>
|
|
|
|
|
dependency.token != null && (tokenReference(dependency.token) == templateRef)));
|
|
|
|
|
const contextName =
|
|
|
|
|
this.contextName && templateDirective && templateDirective.directive.type.reference.name ?
|
|
|
|
|
`${this.contextName}_${templateDirective.directive.type.reference.name}` :
|
|
|
|
|
null;
|
|
|
|
|
const templateName =
|
|
|
|
|
contextName ? `${contextName}_Template_${templateIndex}` : `Template_${templateIndex}`;
|
2017-11-20 10:21:17 -08:00
|
|
|
|
const templateContext = `ctx${this.level}`;
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const {directivesArray, directiveIndexMap} = this._computeDirectivesArray(ast.directives);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// e.g. C(1, C1Template)
|
|
|
|
|
this.instruction(
|
|
|
|
|
this._creationMode, ast.sourceSpan, R3.containerCreate, o.literal(templateIndex),
|
2018-01-11 15:37:56 -08:00
|
|
|
|
directivesArray, o.variable(templateName));
|
|
|
|
|
|
|
|
|
|
// e.g. Cr(1)
|
|
|
|
|
this.instruction(
|
|
|
|
|
this._refreshMode, ast.sourceSpan, R3.containerRefreshStart, o.literal(templateIndex));
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
// Generate directives
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this._visitDirectives(
|
2018-01-11 15:37:56 -08:00
|
|
|
|
ast.directives, o.variable(this.contextParameter), templateIndex, directiveIndexMap);
|
|
|
|
|
|
|
|
|
|
// e.g. cr();
|
|
|
|
|
this.instruction(this._refreshMode, ast.sourceSpan, R3.containerRefreshEnd);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// Create the template function
|
|
|
|
|
const templateVisitor = new TemplateDefinitionBuilder(
|
2018-01-22 15:35:18 -08:00
|
|
|
|
this.outputCtx, this.constantPool, this.reflector, templateContext,
|
2018-01-26 17:12:39 -08:00
|
|
|
|
this.bindingScope.nestedScope(), this.level + 1, this.ngContentSelectors, contextName,
|
|
|
|
|
templateName);
|
2018-01-24 11:19:39 -08:00
|
|
|
|
const templateFunctionExpr = templateVisitor.buildTemplateFunction(ast.children, ast.variables);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this._postfix.push(templateFunctionExpr.toDeclStmt(templateName, null));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// These should be handled in the template or element directly.
|
|
|
|
|
readonly visitReference = invalid;
|
|
|
|
|
readonly visitVariable = invalid;
|
|
|
|
|
readonly visitEvent = invalid;
|
|
|
|
|
readonly visitElementProperty = invalid;
|
|
|
|
|
readonly visitAttr = invalid;
|
|
|
|
|
|
|
|
|
|
visitBoundText(ast: BoundTextAst) {
|
2018-01-11 15:37:56 -08:00
|
|
|
|
const nodeIndex = this.allocateDataSlot();
|
2017-11-20 10:21:17 -08:00
|
|
|
|
|
|
|
|
|
// Creation mode
|
|
|
|
|
this.instruction(this._creationMode, ast.sourceSpan, R3.text, o.literal(nodeIndex));
|
|
|
|
|
|
|
|
|
|
// Refresh mode
|
|
|
|
|
this.instruction(
|
|
|
|
|
this._refreshMode, ast.sourceSpan, R3.textCreateBound, o.literal(nodeIndex),
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this.bind(o.variable(CONTEXT_NAME), ast.value, ast.sourceSpan));
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
visitText(ast: TextAst) {
|
|
|
|
|
// Text is defined in creation mode only.
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this.instruction(
|
|
|
|
|
this._creationMode, ast.sourceSpan, R3.text, o.literal(this.allocateDataSlot()),
|
|
|
|
|
o.literal(ast.value));
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// These should be handled in the template or element directly
|
|
|
|
|
readonly visitDirective = invalid;
|
|
|
|
|
readonly visitDirectiveProperty = invalid;
|
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
|
private allocateDataSlot() { return this._dataIndex++; }
|
2017-11-20 10:21:17 -08:00
|
|
|
|
private bindingContext() { return `${this._bindingContext++}`; }
|
|
|
|
|
|
|
|
|
|
private instruction(
|
2018-01-26 17:12:39 -08:00
|
|
|
|
statements: o.Statement[], span: ParseSourceSpan|null, reference: o.ExternalReference,
|
2017-11-20 10:21:17 -08:00
|
|
|
|
...params: o.Expression[]) {
|
|
|
|
|
statements.push(o.importExpr(reference, null, span).callFn(params, span).toStmt());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private typeReference(type: any): o.Expression { return this.outputCtx.importExpr(type); }
|
|
|
|
|
|
|
|
|
|
private definitionOf(type: any, kind: DefinitionKind): o.Expression {
|
|
|
|
|
return this.constantPool.getDefinition(type, kind, this.outputCtx);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private temp(): o.ReadVarExpr {
|
|
|
|
|
if (!this._temporaryAllocated) {
|
|
|
|
|
this._prefix.push(o.variable(TEMPORARY_NAME, o.DYNAMIC_TYPE, null)
|
|
|
|
|
.set(o.literal(undefined))
|
|
|
|
|
.toDeclStmt(o.DYNAMIC_TYPE));
|
|
|
|
|
this._temporaryAllocated = true;
|
|
|
|
|
}
|
|
|
|
|
return o.variable(TEMPORARY_NAME);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private convertPropertyBinding(implicit: o.Expression, value: AST): o.Expression {
|
|
|
|
|
const convertedPropertyBinding = convertPropertyBinding(
|
2018-01-11 15:37:56 -08:00
|
|
|
|
this, implicit, value, this.bindingContext(), BindingForm.TrySimple, interpolate);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
this._refreshMode.push(...convertedPropertyBinding.stmts);
|
|
|
|
|
return convertedPropertyBinding.currValExpr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bind(implicit: o.Expression, value: AST, sourceSpan: ParseSourceSpan): o.Expression {
|
2018-01-11 15:37:56 -08:00
|
|
|
|
return this.convertPropertyBinding(implicit, value);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function createFactory(
|
|
|
|
|
type: CompileTypeMetadata, outputCtx: OutputContext,
|
|
|
|
|
reflector: CompileReflector): o.FunctionExpr {
|
|
|
|
|
let args: o.Expression[] = [];
|
|
|
|
|
|
|
|
|
|
const elementRef = reflector.resolveExternalReference(Identifiers.ElementRef);
|
|
|
|
|
const templateRef = reflector.resolveExternalReference(Identifiers.TemplateRef);
|
|
|
|
|
const viewContainerRef = reflector.resolveExternalReference(Identifiers.ViewContainerRef);
|
|
|
|
|
|
|
|
|
|
for (let dependency of type.diDeps) {
|
|
|
|
|
if (dependency.isValue) {
|
|
|
|
|
unsupported('value dependencies');
|
|
|
|
|
}
|
|
|
|
|
if (dependency.isHost) {
|
|
|
|
|
unsupported('host dependencies');
|
|
|
|
|
}
|
|
|
|
|
const token = dependency.token;
|
|
|
|
|
if (token) {
|
|
|
|
|
const tokenRef = tokenReference(token);
|
|
|
|
|
if (tokenRef === elementRef) {
|
|
|
|
|
args.push(o.importExpr(R3.injectElementRef).callFn([]));
|
|
|
|
|
} else if (tokenRef === templateRef) {
|
|
|
|
|
args.push(o.importExpr(R3.injectTemplateRef).callFn([]));
|
|
|
|
|
} else if (tokenRef === viewContainerRef) {
|
|
|
|
|
args.push(o.importExpr(R3.injectViewContainerRef).callFn([]));
|
|
|
|
|
} else {
|
2018-01-25 15:38:39 -08:00
|
|
|
|
const value =
|
|
|
|
|
token.identifier != null ? outputCtx.importExpr(tokenRef) : o.literal(tokenRef);
|
|
|
|
|
args.push(o.importExpr(R3.inject).callFn([value]));
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
unsupported('dependency without a token');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return o.fn(
|
|
|
|
|
[],
|
|
|
|
|
[new o.ReturnStatement(new o.InstantiateExpr(outputCtx.importExpr(type.reference), args))],
|
2018-01-22 15:35:18 -08:00
|
|
|
|
o.INFERRED_TYPE, null, type.reference.name ? `${type.reference.name}_Factory` : null);
|
2017-11-20 10:21:17 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function invalid<T>(arg: o.Expression | o.Statement | TemplateAst): never {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Invalid state: Visitor ${this.constructor.name} doesn't handle ${o.constructor.name}`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function findComponent(directives: DirectiveAst[]): DirectiveAst|undefined {
|
|
|
|
|
return directives.filter(directive => directive.directive.isComponent)[0];
|
|
|
|
|
}
|
2018-01-26 17:12:39 -08:00
|
|
|
|
|
|
|
|
|
interface NgContentInfo {
|
|
|
|
|
index: number;
|
|
|
|
|
selector?: R3CssSelector;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class ContentProjectionVisitor extends RecursiveTemplateAstVisitor {
|
|
|
|
|
private index = 1;
|
|
|
|
|
constructor(
|
|
|
|
|
private projectionMap: Map<NgContentAst, NgContentInfo>,
|
|
|
|
|
private ngContentSelectors: string[]) {
|
|
|
|
|
super();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
visitNgContent(ast: NgContentAst) {
|
|
|
|
|
const selectorText = this.ngContentSelectors[ast.index];
|
|
|
|
|
selectorText != null || error(`could not find selector for index ${ast.index} in ${ast}`);
|
|
|
|
|
if (!selectorText || selectorText === '*') {
|
|
|
|
|
this.projectionMap.set(ast, {index: 0});
|
|
|
|
|
} else {
|
|
|
|
|
const cssSelectors = CssSelector.parse(selectorText);
|
|
|
|
|
this.projectionMap.set(
|
|
|
|
|
ast, {index: this.index++, selector: parseSelectorsToR3Selector(cssSelectors)});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getContentProjection(asts: TemplateAst[], ngContentSelectors: string[]) {
|
|
|
|
|
const projectIndexMap = new Map<NgContentAst, NgContentInfo>();
|
|
|
|
|
const visitor = new ContentProjectionVisitor(projectIndexMap, ngContentSelectors);
|
|
|
|
|
templateVisitAll(visitor, asts);
|
|
|
|
|
return projectIndexMap;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// These are a copy the CSS types from core/src/render3/interfaces/projection.ts
|
|
|
|
|
// They are duplicated here as they cannot be directly referenced from core.
|
|
|
|
|
type R3SimpleCssSelector = (string | null)[];
|
|
|
|
|
type R3CssSelectorWithNegations =
|
|
|
|
|
[R3SimpleCssSelector, null] | [R3SimpleCssSelector, R3SimpleCssSelector];
|
|
|
|
|
type R3CssSelector = R3CssSelectorWithNegations[];
|
|
|
|
|
|
|
|
|
|
function parserSelectorToSimpleSelector(selector: CssSelector): R3SimpleCssSelector {
|
|
|
|
|
const classes =
|
|
|
|
|
selector.classNames && selector.classNames.length ? ['class', ...selector.classNames] : [];
|
|
|
|
|
return [selector.element, ...selector.attrs, ...classes];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function parserSelectorToR3Selector(selector: CssSelector): R3CssSelectorWithNegations {
|
|
|
|
|
const positive = parserSelectorToSimpleSelector(selector);
|
|
|
|
|
const negative = selector.notSelectors && selector.notSelectors.length &&
|
|
|
|
|
parserSelectorToSimpleSelector(selector.notSelectors[0]);
|
|
|
|
|
|
|
|
|
|
return negative ? [positive, negative] : [positive, null];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function parseSelectorsToR3Selector(selectors: CssSelector[]): R3CssSelector {
|
|
|
|
|
return selectors.map(parserSelectorToR3Selector);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function asLiteral(value: any): o.Expression {
|
|
|
|
|
if (Array.isArray(value)) {
|
|
|
|
|
return o.literalArr(value.map(asLiteral));
|
|
|
|
|
}
|
|
|
|
|
return o.literal(value, o.INFERRED_TYPE);
|
|
|
|
|
}
|