2016-10-13 16:34:37 -07: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
|
|
|
|
*/
|
|
|
|
|
|
|
|
import {Injectable} from '@angular/core';
|
|
|
|
|
|
|
|
import {CompileDirectiveMetadata, CompileIdentifierMetadata} from './compile_metadata';
|
2016-10-24 08:43:06 -07:00
|
|
|
import {createCheckBindingField, createCheckBindingStmt} from './compiler_util/binding_util';
|
2016-10-24 09:58:52 -07:00
|
|
|
import {convertPropertyBinding} from './compiler_util/expression_converter';
|
|
|
|
import {writeToRenderer} from './compiler_util/render_util';
|
2016-10-13 16:34:37 -07:00
|
|
|
import {CompilerConfig} from './config';
|
2016-10-24 09:58:52 -07:00
|
|
|
import {Parser} from './expression_parser/parser';
|
2016-10-13 16:34:37 -07:00
|
|
|
import {Identifiers, resolveIdentifier} from './identifiers';
|
2016-10-24 09:58:52 -07:00
|
|
|
import {DEFAULT_INTERPOLATION_CONFIG} from './ml_parser/interpolation_config';
|
2016-10-21 13:37:51 -07:00
|
|
|
import {ClassBuilder, createClassStmt} from './output/class_builder';
|
2016-10-13 16:34:37 -07:00
|
|
|
import * as o from './output/output_ast';
|
2016-10-24 09:58:52 -07:00
|
|
|
import {ParseError, ParseErrorLevel, ParseLocation, ParseSourceFile, ParseSourceSpan} from './parse_util';
|
|
|
|
import {Console, LifecycleHooks, isDefaultChangeDetectionStrategy} from './private_import_core';
|
|
|
|
import {ElementSchemaRegistry} from './schema/element_schema_registry';
|
|
|
|
import {BindingParser} from './template_parser/binding_parser';
|
|
|
|
import {BoundElementPropertyAst, BoundEventAst} from './template_parser/template_ast';
|
2016-10-13 16:34:37 -07:00
|
|
|
|
|
|
|
export class DirectiveWrapperCompileResult {
|
|
|
|
constructor(public statements: o.Statement[], public dirWrapperClassVar: string) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
const CONTEXT_FIELD_NAME = 'context';
|
|
|
|
const CHANGES_FIELD_NAME = 'changes';
|
|
|
|
const CHANGED_FIELD_NAME = 'changed';
|
|
|
|
|
|
|
|
const CURR_VALUE_VAR = o.variable('currValue');
|
|
|
|
const THROW_ON_CHANGE_VAR = o.variable('throwOnChange');
|
|
|
|
const FORCE_UPDATE_VAR = o.variable('forceUpdate');
|
|
|
|
const VIEW_VAR = o.variable('view');
|
|
|
|
const RENDER_EL_VAR = o.variable('el');
|
|
|
|
|
|
|
|
const RESET_CHANGES_STMT = o.THIS_EXPR.prop(CHANGES_FIELD_NAME).set(o.literalMap([])).toStmt();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* We generate directive wrappers to prevent code bloat when a directive is used.
|
|
|
|
* A directive wrapper encapsulates
|
|
|
|
* the dirty checking for `@Input`, the handling of `@HostListener` / `@HostBinding`
|
|
|
|
* and calling the lifecyclehooks `ngOnInit`, `ngOnChanges`, `ngDoCheck`.
|
|
|
|
*
|
|
|
|
* So far, only `@Input` and the lifecycle hooks have been implemented.
|
|
|
|
*/
|
|
|
|
@Injectable()
|
|
|
|
export class DirectiveWrapperCompiler {
|
|
|
|
static dirWrapperClassName(id: CompileIdentifierMetadata) { return `Wrapper_${id.name}`; }
|
|
|
|
|
2016-10-24 09:58:52 -07:00
|
|
|
constructor(
|
|
|
|
private compilerConfig: CompilerConfig, private _exprParser: Parser,
|
|
|
|
private _schemaRegistry: ElementSchemaRegistry, private _console: Console) {}
|
2016-10-13 16:34:37 -07:00
|
|
|
|
|
|
|
compile(dirMeta: CompileDirectiveMetadata): DirectiveWrapperCompileResult {
|
2016-10-21 13:37:51 -07:00
|
|
|
const builder = new DirectiveWrapperBuilder(this.compilerConfig, dirMeta);
|
|
|
|
Object.keys(dirMeta.inputs).forEach((inputFieldName) => {
|
2016-10-24 11:11:31 -07:00
|
|
|
addCheckInputMethod(inputFieldName, builder);
|
2016-10-21 13:37:51 -07:00
|
|
|
});
|
2016-10-24 09:58:52 -07:00
|
|
|
addDetectChangesInInputPropsMethod(builder);
|
|
|
|
|
|
|
|
const hostParseResult = parseHostBindings(dirMeta, this._exprParser, this._schemaRegistry);
|
|
|
|
reportParseErrors(hostParseResult.errors, this._console);
|
|
|
|
// host properties are change detected by the DirectiveWrappers,
|
|
|
|
// except for the animation properties as they need close integration with animation events
|
|
|
|
// and DirectiveWrappers don't support
|
|
|
|
// event listeners right now.
|
|
|
|
addDetectChangesInHostPropsMethod(
|
|
|
|
hostParseResult.hostProps.filter(hostProp => !hostProp.isAnimation), builder);
|
|
|
|
|
|
|
|
// TODO(tbosch): implement hostListeners via DirectiveWrapper as well!
|
|
|
|
|
2016-10-21 13:37:51 -07:00
|
|
|
const classStmt = builder.build();
|
|
|
|
return new DirectiveWrapperCompileResult([classStmt], classStmt.name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DirectiveWrapperBuilder implements ClassBuilder {
|
|
|
|
fields: o.ClassField[] = [];
|
|
|
|
getters: o.ClassGetter[] = [];
|
|
|
|
methods: o.ClassMethod[] = [];
|
|
|
|
ctorStmts: o.Statement[] = [];
|
|
|
|
|
|
|
|
genChanges: boolean;
|
|
|
|
ngOnChanges: boolean;
|
|
|
|
ngOnInit: boolean;
|
|
|
|
ngDoCheck: boolean;
|
|
|
|
|
|
|
|
constructor(public compilerConfig: CompilerConfig, public dirMeta: CompileDirectiveMetadata) {
|
|
|
|
const dirLifecycleHooks = dirMeta.type.lifecycleHooks;
|
|
|
|
this.genChanges = dirLifecycleHooks.indexOf(LifecycleHooks.OnChanges) !== -1 ||
|
|
|
|
this.compilerConfig.logBindingUpdate;
|
|
|
|
this.ngOnChanges = dirLifecycleHooks.indexOf(LifecycleHooks.OnChanges) !== -1;
|
|
|
|
this.ngOnInit = dirLifecycleHooks.indexOf(LifecycleHooks.OnInit) !== -1;
|
|
|
|
this.ngDoCheck = dirLifecycleHooks.indexOf(LifecycleHooks.DoCheck) !== -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
build(): o.ClassStmt {
|
2016-10-13 16:34:37 -07:00
|
|
|
const dirDepParamNames: string[] = [];
|
2016-10-21 13:37:51 -07:00
|
|
|
for (let i = 0; i < this.dirMeta.type.diDeps.length; i++) {
|
2016-10-13 16:34:37 -07:00
|
|
|
dirDepParamNames.push(`p${i}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
const fields: o.ClassField[] = [
|
2016-10-21 13:37:51 -07:00
|
|
|
new o.ClassField(CONTEXT_FIELD_NAME, o.importType(this.dirMeta.type)),
|
2016-10-13 16:34:37 -07:00
|
|
|
new o.ClassField(CHANGED_FIELD_NAME, o.BOOL_TYPE),
|
|
|
|
];
|
|
|
|
const ctorStmts: o.Statement[] =
|
|
|
|
[o.THIS_EXPR.prop(CHANGED_FIELD_NAME).set(o.literal(false)).toStmt()];
|
2016-10-21 13:37:51 -07:00
|
|
|
if (this.genChanges) {
|
2016-10-13 16:34:37 -07:00
|
|
|
fields.push(new o.ClassField(CHANGES_FIELD_NAME, new o.MapType(o.DYNAMIC_TYPE)));
|
|
|
|
ctorStmts.push(RESET_CHANGES_STMT);
|
|
|
|
}
|
|
|
|
|
|
|
|
ctorStmts.push(
|
|
|
|
o.THIS_EXPR.prop(CONTEXT_FIELD_NAME)
|
2016-10-21 13:37:51 -07:00
|
|
|
.set(o.importExpr(this.dirMeta.type)
|
2016-10-13 16:34:37 -07:00
|
|
|
.instantiate(dirDepParamNames.map((paramName) => o.variable(paramName))))
|
|
|
|
.toStmt());
|
|
|
|
|
2016-10-21 13:37:51 -07:00
|
|
|
return createClassStmt({
|
|
|
|
name: DirectiveWrapperCompiler.dirWrapperClassName(this.dirMeta.type),
|
|
|
|
ctorParams: dirDepParamNames.map((paramName) => new o.FnParam(paramName, o.DYNAMIC_TYPE)),
|
2016-10-24 11:11:31 -07:00
|
|
|
builders: [{fields, ctorStmts}, this]
|
|
|
|
});
|
2016-10-13 16:34:37 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-24 09:58:52 -07:00
|
|
|
function addDetectChangesInInputPropsMethod(builder: DirectiveWrapperBuilder) {
|
2016-10-13 16:34:37 -07:00
|
|
|
const changedVar = o.variable('changed');
|
|
|
|
const stmts: o.Statement[] = [
|
|
|
|
changedVar.set(o.THIS_EXPR.prop(CHANGED_FIELD_NAME)).toDeclStmt(),
|
|
|
|
o.THIS_EXPR.prop(CHANGED_FIELD_NAME).set(o.literal(false)).toStmt(),
|
|
|
|
];
|
|
|
|
const lifecycleStmts: o.Statement[] = [];
|
|
|
|
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.genChanges) {
|
2016-10-13 16:34:37 -07:00
|
|
|
const onChangesStmts: o.Statement[] = [];
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.ngOnChanges) {
|
2016-10-13 16:34:37 -07:00
|
|
|
onChangesStmts.push(o.THIS_EXPR.prop(CONTEXT_FIELD_NAME)
|
|
|
|
.callMethod('ngOnChanges', [o.THIS_EXPR.prop(CHANGES_FIELD_NAME)])
|
|
|
|
.toStmt());
|
|
|
|
}
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.compilerConfig.logBindingUpdate) {
|
2016-10-13 16:34:37 -07:00
|
|
|
onChangesStmts.push(
|
|
|
|
o.importExpr(resolveIdentifier(Identifiers.setBindingDebugInfoForChanges))
|
|
|
|
.callFn(
|
|
|
|
[VIEW_VAR.prop('renderer'), RENDER_EL_VAR, o.THIS_EXPR.prop(CHANGES_FIELD_NAME)])
|
|
|
|
.toStmt());
|
|
|
|
}
|
|
|
|
onChangesStmts.push(RESET_CHANGES_STMT);
|
|
|
|
lifecycleStmts.push(new o.IfStmt(changedVar, onChangesStmts));
|
|
|
|
}
|
|
|
|
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.ngOnInit) {
|
2016-10-13 16:34:37 -07:00
|
|
|
lifecycleStmts.push(new o.IfStmt(
|
|
|
|
VIEW_VAR.prop('numberOfChecks').identical(new o.LiteralExpr(0)),
|
|
|
|
[o.THIS_EXPR.prop(CONTEXT_FIELD_NAME).callMethod('ngOnInit', []).toStmt()]));
|
|
|
|
}
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.ngDoCheck) {
|
2016-10-13 16:34:37 -07:00
|
|
|
lifecycleStmts.push(o.THIS_EXPR.prop(CONTEXT_FIELD_NAME).callMethod('ngDoCheck', []).toStmt());
|
|
|
|
}
|
|
|
|
if (lifecycleStmts.length > 0) {
|
|
|
|
stmts.push(new o.IfStmt(o.not(THROW_ON_CHANGE_VAR), lifecycleStmts));
|
|
|
|
}
|
|
|
|
stmts.push(new o.ReturnStatement(changedVar));
|
|
|
|
|
2016-10-21 13:37:51 -07:00
|
|
|
builder.methods.push(new o.ClassMethod(
|
2016-10-24 09:58:52 -07:00
|
|
|
'detectChangesInInputProps',
|
2016-10-13 16:34:37 -07:00
|
|
|
[
|
|
|
|
new o.FnParam(
|
|
|
|
VIEW_VAR.name, o.importType(resolveIdentifier(Identifiers.AppView), [o.DYNAMIC_TYPE])),
|
|
|
|
new o.FnParam(RENDER_EL_VAR.name, o.DYNAMIC_TYPE),
|
|
|
|
new o.FnParam(THROW_ON_CHANGE_VAR.name, o.BOOL_TYPE),
|
|
|
|
],
|
2016-10-21 13:37:51 -07:00
|
|
|
stmts, o.BOOL_TYPE));
|
2016-10-13 16:34:37 -07:00
|
|
|
}
|
|
|
|
|
2016-10-24 11:11:31 -07:00
|
|
|
function addCheckInputMethod(input: string, builder: DirectiveWrapperBuilder) {
|
2016-10-24 08:43:06 -07:00
|
|
|
const field = createCheckBindingField(builder);
|
2016-10-13 16:34:37 -07:00
|
|
|
var onChangeStatements: o.Statement[] = [
|
|
|
|
o.THIS_EXPR.prop(CHANGED_FIELD_NAME).set(o.literal(true)).toStmt(),
|
|
|
|
o.THIS_EXPR.prop(CONTEXT_FIELD_NAME).prop(input).set(CURR_VALUE_VAR).toStmt(),
|
|
|
|
];
|
2016-10-21 13:37:51 -07:00
|
|
|
if (builder.genChanges) {
|
2016-10-13 16:34:37 -07:00
|
|
|
onChangeStatements.push(o.THIS_EXPR.prop(CHANGES_FIELD_NAME)
|
|
|
|
.key(o.literal(input))
|
|
|
|
.set(o.importExpr(resolveIdentifier(Identifiers.SimpleChange))
|
2016-10-24 08:43:06 -07:00
|
|
|
.instantiate([field.expression, CURR_VALUE_VAR]))
|
2016-10-13 16:34:37 -07:00
|
|
|
.toStmt());
|
|
|
|
}
|
|
|
|
|
2016-10-24 08:43:06 -07:00
|
|
|
var methodBody: o.Statement[] = createCheckBindingStmt(
|
|
|
|
{currValExpr: CURR_VALUE_VAR, forceUpdate: FORCE_UPDATE_VAR, stmts: []}, field.expression,
|
|
|
|
THROW_ON_CHANGE_VAR, onChangeStatements);
|
2016-10-21 13:37:51 -07:00
|
|
|
builder.methods.push(new o.ClassMethod(
|
2016-10-13 16:34:37 -07:00
|
|
|
`check_${input}`,
|
|
|
|
[
|
|
|
|
new o.FnParam(CURR_VALUE_VAR.name, o.DYNAMIC_TYPE),
|
|
|
|
new o.FnParam(THROW_ON_CHANGE_VAR.name, o.BOOL_TYPE),
|
|
|
|
new o.FnParam(FORCE_UPDATE_VAR.name, o.BOOL_TYPE),
|
|
|
|
],
|
2016-10-21 13:37:51 -07:00
|
|
|
methodBody));
|
2016-10-13 16:34:37 -07:00
|
|
|
}
|
2016-10-24 09:58:52 -07:00
|
|
|
|
|
|
|
function addDetectChangesInHostPropsMethod(
|
|
|
|
hostProps: BoundElementPropertyAst[], builder: DirectiveWrapperBuilder) {
|
|
|
|
const stmts: o.Statement[] = [];
|
|
|
|
const methodParams: o.FnParam[] = [
|
|
|
|
new o.FnParam(
|
|
|
|
VIEW_VAR.name, o.importType(resolveIdentifier(Identifiers.AppView), [o.DYNAMIC_TYPE])),
|
|
|
|
new o.FnParam(RENDER_EL_VAR.name, o.DYNAMIC_TYPE),
|
|
|
|
new o.FnParam(THROW_ON_CHANGE_VAR.name, o.BOOL_TYPE),
|
|
|
|
];
|
|
|
|
hostProps.forEach((hostProp) => {
|
|
|
|
const field = createCheckBindingField(builder);
|
|
|
|
const evalResult = convertPropertyBinding(
|
|
|
|
builder, null, o.THIS_EXPR.prop(CONTEXT_FIELD_NAME), hostProp.value, field.bindingId);
|
|
|
|
if (!evalResult) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let securityContextExpr: o.ReadVarExpr;
|
|
|
|
if (hostProp.needsRuntimeSecurityContext) {
|
|
|
|
securityContextExpr = o.variable(`secCtx_${methodParams.length}`);
|
|
|
|
methodParams.push(new o.FnParam(
|
|
|
|
securityContextExpr.name, o.importType(resolveIdentifier(Identifiers.SecurityContext))));
|
|
|
|
}
|
|
|
|
stmts.push(...createCheckBindingStmt(
|
|
|
|
evalResult, field.expression, THROW_ON_CHANGE_VAR,
|
|
|
|
writeToRenderer(
|
|
|
|
VIEW_VAR, hostProp, RENDER_EL_VAR, evalResult.currValExpr,
|
|
|
|
builder.compilerConfig.logBindingUpdate, securityContextExpr)));
|
|
|
|
});
|
|
|
|
builder.methods.push(new o.ClassMethod('detectChangesInHostProps', methodParams, stmts));
|
|
|
|
}
|
|
|
|
|
|
|
|
class ParseResult {
|
|
|
|
constructor(
|
|
|
|
public hostProps: BoundElementPropertyAst[], public hostListeners: BoundEventAst[],
|
|
|
|
public errors: ParseError[]) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseHostBindings(
|
|
|
|
dirMeta: CompileDirectiveMetadata, exprParser: Parser,
|
|
|
|
schemaRegistry: ElementSchemaRegistry): ParseResult {
|
|
|
|
const errors: ParseError[] = [];
|
|
|
|
const parser =
|
|
|
|
new BindingParser(exprParser, DEFAULT_INTERPOLATION_CONFIG, schemaRegistry, [], errors);
|
|
|
|
const sourceFileName = dirMeta.type.moduleUrl ?
|
|
|
|
`in Directive ${dirMeta.type.name} in ${dirMeta.type.moduleUrl}` :
|
|
|
|
`in Directive ${dirMeta.type.name}`;
|
|
|
|
const sourceFile = new ParseSourceFile('', sourceFileName);
|
|
|
|
const sourceSpan = new ParseSourceSpan(
|
|
|
|
new ParseLocation(sourceFile, null, null, null),
|
|
|
|
new ParseLocation(sourceFile, null, null, null));
|
|
|
|
const parsedHostProps = parser.createDirectiveHostPropertyAsts(dirMeta, sourceSpan);
|
|
|
|
const parsedHostListeners = parser.createDirectiveHostEventAsts(dirMeta, sourceSpan);
|
|
|
|
|
|
|
|
return new ParseResult(parsedHostProps, parsedHostListeners, errors);
|
|
|
|
}
|
|
|
|
|
|
|
|
function reportParseErrors(parseErrors: ParseError[], console: Console) {
|
|
|
|
const warnings = parseErrors.filter(error => error.level === ParseErrorLevel.WARNING);
|
|
|
|
const errors = parseErrors.filter(error => error.level === ParseErrorLevel.FATAL);
|
|
|
|
|
|
|
|
if (warnings.length > 0) {
|
|
|
|
this._console.warn(`Directive parse warnings:\n${warnings.join('\n')}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors.length > 0) {
|
|
|
|
throw new Error(`Directive parse errors:\n${errors.join('\n')}`);
|
|
|
|
}
|
|
|
|
}
|