
* feat(animations): support animation trigger template callbacks * refactor(animations): deport TCB away from animation-land forever
212 lines
9.0 KiB
TypeScript
212 lines
9.0 KiB
TypeScript
/**
|
|
* @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 {AnimationOutput} from '../../core_private';
|
|
import {CompileDirectiveMetadata} from '../compile_metadata';
|
|
import {ListWrapper, StringMapWrapper} from '../facade/collection';
|
|
import {StringWrapper, isBlank, isPresent} from '../facade/lang';
|
|
import {Identifiers, identifierToken} from '../identifiers';
|
|
import * as o from '../output/output_ast';
|
|
import {BoundEventAst, DirectiveAst} from '../template_parser/template_ast';
|
|
|
|
import {CompileBinding} from './compile_binding';
|
|
import {CompileElement} from './compile_element';
|
|
import {CompileMethod} from './compile_method';
|
|
import {EventHandlerVars, ViewProperties} from './constants';
|
|
import {convertCdStatementToIr} from './expression_converter';
|
|
|
|
export class CompileElementAnimationOutput {
|
|
constructor(public listener: CompileEventListener, public output: AnimationOutput) {}
|
|
}
|
|
|
|
export class CompileEventListener {
|
|
private _method: CompileMethod;
|
|
private _hasComponentHostListener: boolean = false;
|
|
private _methodName: string;
|
|
private _eventParam: o.FnParam;
|
|
private _actionResultExprs: o.Expression[] = [];
|
|
|
|
static getOrCreate(
|
|
compileElement: CompileElement, eventTarget: string, eventName: string,
|
|
targetEventListeners: CompileEventListener[]): CompileEventListener {
|
|
var listener = targetEventListeners.find(
|
|
listener => listener.eventTarget == eventTarget && listener.eventName == eventName);
|
|
if (isBlank(listener)) {
|
|
listener = new CompileEventListener(
|
|
compileElement, eventTarget, eventName, targetEventListeners.length);
|
|
targetEventListeners.push(listener);
|
|
}
|
|
return listener;
|
|
}
|
|
|
|
get methodName() { return this._methodName; }
|
|
|
|
constructor(
|
|
public compileElement: CompileElement, public eventTarget: string, public eventName: string,
|
|
listenerIndex: number) {
|
|
this._method = new CompileMethod(compileElement.view);
|
|
this._methodName =
|
|
`_handle_${santitizeEventName(eventName)}_${compileElement.nodeIndex}_${listenerIndex}`;
|
|
this._eventParam = new o.FnParam(
|
|
EventHandlerVars.event.name,
|
|
o.importType(this.compileElement.view.genConfig.renderTypes.renderEvent));
|
|
}
|
|
|
|
addAction(
|
|
hostEvent: BoundEventAst, directive: CompileDirectiveMetadata,
|
|
directiveInstance: o.Expression) {
|
|
if (isPresent(directive) && directive.isComponent) {
|
|
this._hasComponentHostListener = true;
|
|
}
|
|
this._method.resetDebugInfo(this.compileElement.nodeIndex, hostEvent);
|
|
var context = isPresent(directiveInstance) ? directiveInstance :
|
|
this.compileElement.view.componentContext;
|
|
var actionStmts = convertCdStatementToIr(
|
|
this.compileElement.view, context, hostEvent.handler, this.compileElement.nodeIndex);
|
|
var lastIndex = actionStmts.length - 1;
|
|
if (lastIndex >= 0) {
|
|
var lastStatement = actionStmts[lastIndex];
|
|
var returnExpr = convertStmtIntoExpression(lastStatement);
|
|
var preventDefaultVar = o.variable(`pd_${this._actionResultExprs.length}`);
|
|
this._actionResultExprs.push(preventDefaultVar);
|
|
if (isPresent(returnExpr)) {
|
|
// Note: We need to cast the result of the method call to dynamic,
|
|
// as it might be a void method!
|
|
actionStmts[lastIndex] =
|
|
preventDefaultVar.set(returnExpr.cast(o.DYNAMIC_TYPE).notIdentical(o.literal(false)))
|
|
.toDeclStmt(null, [o.StmtModifier.Final]);
|
|
}
|
|
}
|
|
this._method.addStmts(actionStmts);
|
|
}
|
|
|
|
finishMethod() {
|
|
var markPathToRootStart = this._hasComponentHostListener ?
|
|
this.compileElement.appElement.prop('componentView') :
|
|
o.THIS_EXPR;
|
|
var resultExpr: o.Expression = o.literal(true);
|
|
this._actionResultExprs.forEach((expr) => { resultExpr = resultExpr.and(expr); });
|
|
var stmts =
|
|
(<o.Statement[]>[markPathToRootStart.callMethod('markPathToRootAsCheckOnce', []).toStmt()])
|
|
.concat(this._method.finish())
|
|
.concat([new o.ReturnStatement(resultExpr)]);
|
|
// private is fine here as no child view will reference the event handler...
|
|
this.compileElement.view.eventHandlerMethods.push(new o.ClassMethod(
|
|
this._methodName, [this._eventParam], stmts, o.BOOL_TYPE, [o.StmtModifier.Private]));
|
|
}
|
|
|
|
listenToRenderer() {
|
|
var listenExpr: any /** TODO #9100 */;
|
|
var eventListener = o.THIS_EXPR.callMethod(
|
|
'eventHandler',
|
|
[o.THIS_EXPR.prop(this._methodName).callMethod(o.BuiltinMethod.Bind, [o.THIS_EXPR])]);
|
|
if (isPresent(this.eventTarget)) {
|
|
listenExpr = ViewProperties.renderer.callMethod(
|
|
'listenGlobal', [o.literal(this.eventTarget), o.literal(this.eventName), eventListener]);
|
|
} else {
|
|
listenExpr = ViewProperties.renderer.callMethod(
|
|
'listen', [this.compileElement.renderNode, o.literal(this.eventName), eventListener]);
|
|
}
|
|
var disposable = o.variable(`disposable_${this.compileElement.view.disposables.length}`);
|
|
this.compileElement.view.disposables.push(disposable);
|
|
// private is fine here as no child view will reference the event handler...
|
|
this.compileElement.view.createMethod.addStmt(
|
|
disposable.set(listenExpr).toDeclStmt(o.FUNCTION_TYPE, [o.StmtModifier.Private]));
|
|
}
|
|
|
|
listenToAnimation(output: AnimationOutput) {
|
|
var outputListener = o.THIS_EXPR.callMethod(
|
|
'eventHandler',
|
|
[o.THIS_EXPR.prop(this._methodName).callMethod(o.BuiltinMethod.Bind, [o.THIS_EXPR])]);
|
|
|
|
// tie the property callback method to the view animations map
|
|
var stmt = o.THIS_EXPR
|
|
.callMethod(
|
|
'registerAnimationOutput',
|
|
[
|
|
this.compileElement.renderNode,
|
|
o.importExpr(Identifiers.AnimationOutput).instantiate([
|
|
o.literal(output.name), o.literal(output.phase)
|
|
]),
|
|
outputListener
|
|
])
|
|
.toStmt();
|
|
this.compileElement.view.createMethod.addStmt(stmt);
|
|
}
|
|
|
|
listenToDirective(directiveInstance: o.Expression, observablePropName: string) {
|
|
var subscription = o.variable(`subscription_${this.compileElement.view.subscriptions.length}`);
|
|
this.compileElement.view.subscriptions.push(subscription);
|
|
var eventListener = o.THIS_EXPR.callMethod(
|
|
'eventHandler',
|
|
[o.THIS_EXPR.prop(this._methodName).callMethod(o.BuiltinMethod.Bind, [o.THIS_EXPR])]);
|
|
this.compileElement.view.createMethod.addStmt(
|
|
subscription
|
|
.set(directiveInstance.prop(observablePropName)
|
|
.callMethod(o.BuiltinMethod.SubscribeObservable, [eventListener]))
|
|
.toDeclStmt(null, [o.StmtModifier.Final]));
|
|
}
|
|
}
|
|
|
|
export function collectEventListeners(
|
|
hostEvents: BoundEventAst[], dirs: DirectiveAst[],
|
|
compileElement: CompileElement): CompileEventListener[] {
|
|
var eventListeners: CompileEventListener[] = [];
|
|
hostEvents.forEach((hostEvent) => {
|
|
compileElement.view.bindings.push(new CompileBinding(compileElement, hostEvent));
|
|
var listener = CompileEventListener.getOrCreate(
|
|
compileElement, hostEvent.target, hostEvent.name, eventListeners);
|
|
listener.addAction(hostEvent, null, null);
|
|
});
|
|
dirs.forEach((directiveAst) => {
|
|
var directiveInstance =
|
|
compileElement.instances.get(identifierToken(directiveAst.directive.type));
|
|
directiveAst.hostEvents.forEach((hostEvent) => {
|
|
compileElement.view.bindings.push(new CompileBinding(compileElement, hostEvent));
|
|
var listener = CompileEventListener.getOrCreate(
|
|
compileElement, hostEvent.target, hostEvent.name, eventListeners);
|
|
listener.addAction(hostEvent, directiveAst.directive, directiveInstance);
|
|
});
|
|
});
|
|
eventListeners.forEach((listener) => listener.finishMethod());
|
|
return eventListeners;
|
|
}
|
|
|
|
export function bindDirectiveOutputs(
|
|
directiveAst: DirectiveAst, directiveInstance: o.Expression,
|
|
eventListeners: CompileEventListener[]) {
|
|
StringMapWrapper.forEach(
|
|
directiveAst.directive.outputs,
|
|
(eventName: any /** TODO #9100 */, observablePropName: any /** TODO #9100 */) => {
|
|
eventListeners.filter(listener => listener.eventName == eventName).forEach((listener) => {
|
|
listener.listenToDirective(directiveInstance, observablePropName);
|
|
});
|
|
});
|
|
}
|
|
|
|
export function bindRenderOutputs(eventListeners: CompileEventListener[]) {
|
|
eventListeners.forEach(listener => listener.listenToRenderer());
|
|
}
|
|
|
|
export function bindAnimationOutputs(eventListeners: CompileElementAnimationOutput[]) {
|
|
eventListeners.forEach(entry => { entry.listener.listenToAnimation(entry.output); });
|
|
}
|
|
|
|
function convertStmtIntoExpression(stmt: o.Statement): o.Expression {
|
|
if (stmt instanceof o.ExpressionStatement) {
|
|
return stmt.expr;
|
|
} else if (stmt instanceof o.ReturnStatement) {
|
|
return stmt.value;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function santitizeEventName(name: string): string {
|
|
return StringWrapper.replaceAll(name, /[^a-zA-Z_]/g, '_');
|
|
}
|