import {isBlank, isPresent, FunctionWrapper, BaseException} from "angular2/src/facade/lang"; import {List, Map, ListWrapper, StringMapWrapper} from "angular2/src/facade/collection"; import {Locals} from "./locals"; export class AST { eval(context, locals: Locals) { throw new BaseException("Not supported"); } get isAssignable(): boolean { return false; } assign(context, locals: Locals, value) { throw new BaseException("Not supported"); } visit(visitor: AstVisitor): any { return null; } toString(): string { return "AST"; } } export class EmptyExpr extends AST { eval(context, locals: Locals): any { return null; } visit(visitor: AstVisitor) { // do nothing } } export class ImplicitReceiver extends AST { eval(context, locals: Locals): any { return context; } visit(visitor: AstVisitor): any { return visitor.visitImplicitReceiver(this); } } /** * Multiple expressions separated by a semicolon. */ export class Chain extends AST { constructor(public expressions: List) { super(); } eval(context, locals: Locals): any { var result; for (var i = 0; i < this.expressions.length; i++) { var last = this.expressions[i].eval(context, locals); if (isPresent(last)) result = last; } return result; } visit(visitor: AstVisitor): any { return visitor.visitChain(this); } } export class Conditional extends AST { constructor(public condition: AST, public trueExp: AST, public falseExp: AST) { super(); } eval(context, locals: Locals): any { if (this.condition.eval(context, locals)) { return this.trueExp.eval(context, locals); } else { return this.falseExp.eval(context, locals); } } visit(visitor: AstVisitor): any { return visitor.visitConditional(this); } } export class If extends AST { constructor(public condition: AST, public trueExp: AST, public falseExp?: AST) { super(); } eval(context, locals) { if (this.condition.eval(context, locals)) { this.trueExp.eval(context, locals); } else if (isPresent(this.falseExp)) { this.falseExp.eval(context, locals); } } visit(visitor: AstVisitor): any { return visitor.visitIf(this); } } export class AccessMember extends AST { constructor(public receiver: AST, public name: string, public getter: Function, public setter: Function) { super(); } eval(context, locals: Locals): any { if (this.receiver instanceof ImplicitReceiver && isPresent(locals) && locals.contains(this.name)) { return locals.get(this.name); } else { var evaluatedReceiver = this.receiver.eval(context, locals); return this.getter(evaluatedReceiver); } } get isAssignable(): boolean { return true; } assign(context, locals: Locals, value): any { var evaluatedContext = this.receiver.eval(context, locals); if (this.receiver instanceof ImplicitReceiver && isPresent(locals) && locals.contains(this.name)) { throw new BaseException(`Cannot reassign a variable binding ${this.name}`); } else { return this.setter(evaluatedContext, value); } } visit(visitor: AstVisitor): any { return visitor.visitAccessMember(this); } } export class SafeAccessMember extends AST { constructor(public receiver: AST, public name: string, public getter: Function, public setter: Function) { super(); } eval(context, locals: Locals): any { var evaluatedReceiver = this.receiver.eval(context, locals); return isBlank(evaluatedReceiver) ? null : this.getter(evaluatedReceiver); } visit(visitor: AstVisitor): any { return visitor.visitSafeAccessMember(this); } } export class KeyedAccess extends AST { constructor(public obj: AST, public key: AST) { super(); } eval(context, locals: Locals): any { var obj: any = this.obj.eval(context, locals); var key: any = this.key.eval(context, locals); return obj[key]; } get isAssignable(): boolean { return true; } assign(context, locals: Locals, value): any { var obj: any = this.obj.eval(context, locals); var key: any = this.key.eval(context, locals); obj[key] = value; return value; } visit(visitor: AstVisitor): any { return visitor.visitKeyedAccess(this); } } export class BindingPipe extends AST { constructor(public exp: AST, public name: string, public args: List) { super(); } visit(visitor: AstVisitor): any { return visitor.visitPipe(this); } } export class LiteralPrimitive extends AST { constructor(public value) { super(); } eval(context, locals: Locals): any { return this.value; } visit(visitor: AstVisitor): any { return visitor.visitLiteralPrimitive(this); } } export class LiteralArray extends AST { constructor(public expressions: List) { super(); } eval(context, locals: Locals): any { return ListWrapper.map(this.expressions, (e) => e.eval(context, locals)); } visit(visitor: AstVisitor): any { return visitor.visitLiteralArray(this); } } export class LiteralMap extends AST { constructor(public keys: List, public values: List) { super(); } eval(context, locals: Locals): any { var res = StringMapWrapper.create(); for (var i = 0; i < this.keys.length; ++i) { StringMapWrapper.set(res, this.keys[i], this.values[i].eval(context, locals)); } return res; } visit(visitor: AstVisitor): any { return visitor.visitLiteralMap(this); } } export class Interpolation extends AST { constructor(public strings: List, public expressions: List) { super(); } eval(context, locals): any { throw new BaseException("evaluating an Interpolation is not supported"); } visit(visitor: AstVisitor) { visitor.visitInterpolation(this); } } export class Binary extends AST { constructor(public operation: string, public left: AST, public right: AST) { super(); } eval(context, locals: Locals): any { var left: any = this.left.eval(context, locals); switch (this.operation) { case '&&': return left && this.right.eval(context, locals); case '||': return left || this.right.eval(context, locals); } var right: any = this.right.eval(context, locals); switch (this.operation) { case '+': return left + right; case '-': return left - right; case '*': return left * right; case '/': return left / right; case '%': return left % right; case '==': return left == right; case '!=': return left != right; case '===': return left === right; case '!==': return left !== right; case '<': return left < right; case '>': return left > right; case '<=': return left <= right; case '>=': return left >= right; case '^': return left ^ right; case '&': return left & right; } throw 'Internal error [$operation] not handled'; } visit(visitor: AstVisitor): any { return visitor.visitBinary(this); } } export class PrefixNot extends AST { constructor(public expression: AST) { super(); } eval(context, locals: Locals): any { return !this.expression.eval(context, locals); } visit(visitor: AstVisitor): any { return visitor.visitPrefixNot(this); } } export class Assignment extends AST { constructor(public target: AST, public value: AST) { super(); } eval(context, locals: Locals): any { return this.target.assign(context, locals, this.value.eval(context, locals)); } visit(visitor: AstVisitor): any { return visitor.visitAssignment(this); } } export class MethodCall extends AST { constructor(public receiver: AST, public name: string, public fn: Function, public args: List) { super(); } eval(context, locals: Locals): any { var evaluatedArgs = evalList(context, locals, this.args); if (this.receiver instanceof ImplicitReceiver && isPresent(locals) && locals.contains(this.name)) { var fn = locals.get(this.name); return FunctionWrapper.apply(fn, evaluatedArgs); } else { var evaluatedReceiver = this.receiver.eval(context, locals); return this.fn(evaluatedReceiver, evaluatedArgs); } } visit(visitor: AstVisitor): any { return visitor.visitMethodCall(this); } } export class SafeMethodCall extends AST { constructor(public receiver: AST, public name: string, public fn: Function, public args: List) { super(); } eval(context, locals: Locals): any { var evaluatedReceiver = this.receiver.eval(context, locals); if (isBlank(evaluatedReceiver)) return null; var evaluatedArgs = evalList(context, locals, this.args); return this.fn(evaluatedReceiver, evaluatedArgs); } visit(visitor: AstVisitor): any { return visitor.visitSafeMethodCall(this); } } export class FunctionCall extends AST { constructor(public target: AST, public args: List) { super(); } eval(context, locals: Locals): any { var obj: any = this.target.eval(context, locals); if (!(obj instanceof Function)) { throw new BaseException(`${obj} is not a function`); } return FunctionWrapper.apply(obj, evalList(context, locals, this.args)); } visit(visitor: AstVisitor): any { return visitor.visitFunctionCall(this); } } export class ASTWithSource extends AST { constructor(public ast: AST, public source: string, public location: string) { super(); } eval(context, locals: Locals): any { return this.ast.eval(context, locals); } get isAssignable(): boolean { return this.ast.isAssignable; } assign(context, locals: Locals, value): any { return this.ast.assign(context, locals, value); } visit(visitor: AstVisitor): any { return this.ast.visit(visitor); } toString(): string { return `${this.source} in ${this.location}`; } } export class TemplateBinding { constructor(public key: string, public keyIsVar: boolean, public name: string, public expression: ASTWithSource) {} } export interface AstVisitor { visitAccessMember(ast: AccessMember): any; visitAssignment(ast: Assignment): any; visitBinary(ast: Binary): any; visitChain(ast: Chain): any; visitConditional(ast: Conditional): any; visitIf(ast: If): any; visitPipe(ast: BindingPipe): any; visitFunctionCall(ast: FunctionCall): any; visitImplicitReceiver(ast: ImplicitReceiver): any; visitInterpolation(ast: Interpolation): any; visitKeyedAccess(ast: KeyedAccess): any; visitLiteralArray(ast: LiteralArray): any; visitLiteralMap(ast: LiteralMap): any; visitLiteralPrimitive(ast: LiteralPrimitive): any; visitMethodCall(ast: MethodCall): any; visitPrefixNot(ast: PrefixNot): any; visitSafeAccessMember(ast: SafeAccessMember): any; visitSafeMethodCall(ast: SafeMethodCall): any; } export class AstTransformer implements AstVisitor { visitImplicitReceiver(ast: ImplicitReceiver): ImplicitReceiver { return ast; } visitInterpolation(ast: Interpolation): Interpolation { return new Interpolation(ast.strings, this.visitAll(ast.expressions)); } visitLiteralPrimitive(ast: LiteralPrimitive): LiteralPrimitive { return new LiteralPrimitive(ast.value); } visitAccessMember(ast: AccessMember): AccessMember { return new AccessMember(ast.receiver.visit(this), ast.name, ast.getter, ast.setter); } visitSafeAccessMember(ast: SafeAccessMember): SafeAccessMember { return new SafeAccessMember(ast.receiver.visit(this), ast.name, ast.getter, ast.setter); } visitMethodCall(ast: MethodCall): MethodCall { return new MethodCall(ast.receiver.visit(this), ast.name, ast.fn, this.visitAll(ast.args)); } visitSafeMethodCall(ast: SafeMethodCall): SafeMethodCall { return new SafeMethodCall(ast.receiver.visit(this), ast.name, ast.fn, this.visitAll(ast.args)); } visitFunctionCall(ast: FunctionCall): FunctionCall { return new FunctionCall(ast.target.visit(this), this.visitAll(ast.args)); } visitLiteralArray(ast: LiteralArray): LiteralArray { return new LiteralArray(this.visitAll(ast.expressions)); } visitLiteralMap(ast: LiteralMap): LiteralMap { return new LiteralMap(ast.keys, this.visitAll(ast.values)); } visitBinary(ast: Binary): Binary { return new Binary(ast.operation, ast.left.visit(this), ast.right.visit(this)); } visitPrefixNot(ast: PrefixNot): PrefixNot { return new PrefixNot(ast.expression.visit(this)); } visitConditional(ast: Conditional): Conditional { return new Conditional(ast.condition.visit(this), ast.trueExp.visit(this), ast.falseExp.visit(this)); } visitPipe(ast: BindingPipe): BindingPipe { return new BindingPipe(ast.exp.visit(this), ast.name, this.visitAll(ast.args)); } visitKeyedAccess(ast: KeyedAccess): KeyedAccess { return new KeyedAccess(ast.obj.visit(this), ast.key.visit(this)); } visitAll(asts: List): List { var res = ListWrapper.createFixedSize(asts.length); for (var i = 0; i < asts.length; ++i) { res[i] = asts[i].visit(this); } return res; } visitChain(ast: Chain): Chain { return new Chain(this.visitAll(ast.expressions)); } visitAssignment(ast: Assignment): Assignment { return new Assignment(ast.target.visit(this), ast.value.visit(this)); } visitIf(ast: If): If { let falseExp = isPresent(ast.falseExp) ? ast.falseExp.visit(this) : null; return new If(ast.condition.visit(this), ast.trueExp.visit(this), falseExp); } } var _evalListCache = [ [], [0], [0, 0], [0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0] ]; function evalList(context, locals: Locals, exps: List): any[] { var length = exps.length; if (length > 10) { throw new BaseException("Cannot have more than 10 argument"); } var result = _evalListCache[length]; for (var i = 0; i < length; i++) { result[i] = exps[i].eval(context, locals); } return result; }