[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | import * as cdAst from '../expression_parser/ast';
|
---|
| 9 | import { Identifiers } from '../identifiers';
|
---|
| 10 | import * as o from '../output/output_ast';
|
---|
| 11 | import { ParseSourceSpan } from '../parse_util';
|
---|
| 12 | export class EventHandlerVars {
|
---|
| 13 | }
|
---|
| 14 | EventHandlerVars.event = o.variable('$event');
|
---|
| 15 | export class ConvertActionBindingResult {
|
---|
| 16 | constructor(
|
---|
| 17 | /**
|
---|
| 18 | * Render2 compatible statements,
|
---|
| 19 | */
|
---|
| 20 | stmts,
|
---|
| 21 | /**
|
---|
| 22 | * Variable name used with render2 compatible statements.
|
---|
| 23 | */
|
---|
| 24 | allowDefault) {
|
---|
| 25 | this.stmts = stmts;
|
---|
| 26 | this.allowDefault = allowDefault;
|
---|
| 27 | /**
|
---|
| 28 | * This is bit of a hack. It converts statements which render2 expects to statements which are
|
---|
| 29 | * expected by render3.
|
---|
| 30 | *
|
---|
| 31 | * Example: `<div click="doSomething($event)">` will generate:
|
---|
| 32 | *
|
---|
| 33 | * Render3:
|
---|
| 34 | * ```
|
---|
| 35 | * const pd_b:any = ((<any>ctx.doSomething($event)) !== false);
|
---|
| 36 | * return pd_b;
|
---|
| 37 | * ```
|
---|
| 38 | *
|
---|
| 39 | * but render2 expects:
|
---|
| 40 | * ```
|
---|
| 41 | * return ctx.doSomething($event);
|
---|
| 42 | * ```
|
---|
| 43 | */
|
---|
| 44 | // TODO(misko): remove this hack once we no longer support ViewEngine.
|
---|
| 45 | this.render3Stmts = stmts.map((statement) => {
|
---|
| 46 | if (statement instanceof o.DeclareVarStmt && statement.name == allowDefault.name &&
|
---|
| 47 | statement.value instanceof o.BinaryOperatorExpr) {
|
---|
| 48 | const lhs = statement.value.lhs;
|
---|
| 49 | return new o.ReturnStatement(lhs.value);
|
---|
| 50 | }
|
---|
| 51 | return statement;
|
---|
| 52 | });
|
---|
| 53 | }
|
---|
| 54 | }
|
---|
| 55 | /**
|
---|
| 56 | * Converts the given expression AST into an executable output AST, assuming the expression is
|
---|
| 57 | * used in an action binding (e.g. an event handler).
|
---|
| 58 | */
|
---|
| 59 | export function convertActionBinding(localResolver, implicitReceiver, action, bindingId, interpolationFunction, baseSourceSpan, implicitReceiverAccesses, globals) {
|
---|
| 60 | if (!localResolver) {
|
---|
| 61 | localResolver = new DefaultLocalResolver(globals);
|
---|
| 62 | }
|
---|
| 63 | const actionWithoutBuiltins = convertPropertyBindingBuiltins({
|
---|
| 64 | createLiteralArrayConverter: (argCount) => {
|
---|
| 65 | // Note: no caching for literal arrays in actions.
|
---|
| 66 | return (args) => o.literalArr(args);
|
---|
| 67 | },
|
---|
| 68 | createLiteralMapConverter: (keys) => {
|
---|
| 69 | // Note: no caching for literal maps in actions.
|
---|
| 70 | return (values) => {
|
---|
| 71 | const entries = keys.map((k, i) => ({
|
---|
| 72 | key: k.key,
|
---|
| 73 | value: values[i],
|
---|
| 74 | quoted: k.quoted,
|
---|
| 75 | }));
|
---|
| 76 | return o.literalMap(entries);
|
---|
| 77 | };
|
---|
| 78 | },
|
---|
| 79 | createPipeConverter: (name) => {
|
---|
| 80 | throw new Error(`Illegal State: Actions are not allowed to contain pipes. Pipe: ${name}`);
|
---|
| 81 | }
|
---|
| 82 | }, action);
|
---|
| 83 | const visitor = new _AstToIrVisitor(localResolver, implicitReceiver, bindingId, interpolationFunction, baseSourceSpan, implicitReceiverAccesses);
|
---|
| 84 | const actionStmts = [];
|
---|
| 85 | flattenStatements(actionWithoutBuiltins.visit(visitor, _Mode.Statement), actionStmts);
|
---|
| 86 | prependTemporaryDecls(visitor.temporaryCount, bindingId, actionStmts);
|
---|
| 87 | if (visitor.usesImplicitReceiver) {
|
---|
| 88 | localResolver.notifyImplicitReceiverUse();
|
---|
| 89 | }
|
---|
| 90 | const lastIndex = actionStmts.length - 1;
|
---|
| 91 | let preventDefaultVar = null;
|
---|
| 92 | if (lastIndex >= 0) {
|
---|
| 93 | const lastStatement = actionStmts[lastIndex];
|
---|
| 94 | const returnExpr = convertStmtIntoExpression(lastStatement);
|
---|
| 95 | if (returnExpr) {
|
---|
| 96 | // Note: We need to cast the result of the method call to dynamic,
|
---|
| 97 | // as it might be a void method!
|
---|
| 98 | preventDefaultVar = createPreventDefaultVar(bindingId);
|
---|
| 99 | actionStmts[lastIndex] =
|
---|
| 100 | preventDefaultVar.set(returnExpr.cast(o.DYNAMIC_TYPE).notIdentical(o.literal(false)))
|
---|
| 101 | .toDeclStmt(null, [o.StmtModifier.Final]);
|
---|
| 102 | }
|
---|
| 103 | }
|
---|
| 104 | return new ConvertActionBindingResult(actionStmts, preventDefaultVar);
|
---|
| 105 | }
|
---|
| 106 | export function convertPropertyBindingBuiltins(converterFactory, ast) {
|
---|
| 107 | return convertBuiltins(converterFactory, ast);
|
---|
| 108 | }
|
---|
| 109 | export class ConvertPropertyBindingResult {
|
---|
| 110 | constructor(stmts, currValExpr) {
|
---|
| 111 | this.stmts = stmts;
|
---|
| 112 | this.currValExpr = currValExpr;
|
---|
| 113 | }
|
---|
| 114 | }
|
---|
| 115 | export var BindingForm;
|
---|
| 116 | (function (BindingForm) {
|
---|
| 117 | // The general form of binding expression, supports all expressions.
|
---|
| 118 | BindingForm[BindingForm["General"] = 0] = "General";
|
---|
| 119 | // Try to generate a simple binding (no temporaries or statements)
|
---|
| 120 | // otherwise generate a general binding
|
---|
| 121 | BindingForm[BindingForm["TrySimple"] = 1] = "TrySimple";
|
---|
| 122 | // Inlines assignment of temporaries into the generated expression. The result may still
|
---|
| 123 | // have statements attached for declarations of temporary variables.
|
---|
| 124 | // This is the only relevant form for Ivy, the other forms are only used in ViewEngine.
|
---|
| 125 | BindingForm[BindingForm["Expression"] = 2] = "Expression";
|
---|
| 126 | })(BindingForm || (BindingForm = {}));
|
---|
| 127 | /**
|
---|
| 128 | * Converts the given expression AST into an executable output AST, assuming the expression
|
---|
| 129 | * is used in property binding. The expression has to be preprocessed via
|
---|
| 130 | * `convertPropertyBindingBuiltins`.
|
---|
| 131 | */
|
---|
| 132 | export function convertPropertyBinding(localResolver, implicitReceiver, expressionWithoutBuiltins, bindingId, form, interpolationFunction) {
|
---|
| 133 | if (!localResolver) {
|
---|
| 134 | localResolver = new DefaultLocalResolver();
|
---|
| 135 | }
|
---|
| 136 | const visitor = new _AstToIrVisitor(localResolver, implicitReceiver, bindingId, interpolationFunction);
|
---|
| 137 | const outputExpr = expressionWithoutBuiltins.visit(visitor, _Mode.Expression);
|
---|
| 138 | const stmts = getStatementsFromVisitor(visitor, bindingId);
|
---|
| 139 | if (visitor.usesImplicitReceiver) {
|
---|
| 140 | localResolver.notifyImplicitReceiverUse();
|
---|
| 141 | }
|
---|
| 142 | if (visitor.temporaryCount === 0 && form == BindingForm.TrySimple) {
|
---|
| 143 | return new ConvertPropertyBindingResult([], outputExpr);
|
---|
| 144 | }
|
---|
| 145 | else if (form === BindingForm.Expression) {
|
---|
| 146 | return new ConvertPropertyBindingResult(stmts, outputExpr);
|
---|
| 147 | }
|
---|
| 148 | const currValExpr = createCurrValueExpr(bindingId);
|
---|
| 149 | stmts.push(currValExpr.set(outputExpr).toDeclStmt(o.DYNAMIC_TYPE, [o.StmtModifier.Final]));
|
---|
| 150 | return new ConvertPropertyBindingResult(stmts, currValExpr);
|
---|
| 151 | }
|
---|
| 152 | /**
|
---|
| 153 | * Given some expression, such as a binding or interpolation expression, and a context expression to
|
---|
| 154 | * look values up on, visit each facet of the given expression resolving values from the context
|
---|
| 155 | * expression such that a list of arguments can be derived from the found values that can be used as
|
---|
| 156 | * arguments to an external update instruction.
|
---|
| 157 | *
|
---|
| 158 | * @param localResolver The resolver to use to look up expressions by name appropriately
|
---|
| 159 | * @param contextVariableExpression The expression representing the context variable used to create
|
---|
| 160 | * the final argument expressions
|
---|
| 161 | * @param expressionWithArgumentsToExtract The expression to visit to figure out what values need to
|
---|
| 162 | * be resolved and what arguments list to build.
|
---|
| 163 | * @param bindingId A name prefix used to create temporary variable names if they're needed for the
|
---|
| 164 | * arguments generated
|
---|
| 165 | * @returns An array of expressions that can be passed as arguments to instruction expressions like
|
---|
| 166 | * `o.importExpr(R3.propertyInterpolate).callFn(result)`
|
---|
| 167 | */
|
---|
| 168 | export function convertUpdateArguments(localResolver, contextVariableExpression, expressionWithArgumentsToExtract, bindingId) {
|
---|
| 169 | const visitor = new _AstToIrVisitor(localResolver, contextVariableExpression, bindingId, undefined);
|
---|
| 170 | const outputExpr = expressionWithArgumentsToExtract.visit(visitor, _Mode.Expression);
|
---|
| 171 | if (visitor.usesImplicitReceiver) {
|
---|
| 172 | localResolver.notifyImplicitReceiverUse();
|
---|
| 173 | }
|
---|
| 174 | const stmts = getStatementsFromVisitor(visitor, bindingId);
|
---|
| 175 | // Removing the first argument, because it was a length for ViewEngine, not Ivy.
|
---|
| 176 | let args = outputExpr.args.slice(1);
|
---|
| 177 | if (expressionWithArgumentsToExtract instanceof cdAst.Interpolation) {
|
---|
| 178 | // If we're dealing with an interpolation of 1 value with an empty prefix and suffix, reduce the
|
---|
| 179 | // args returned to just the value, because we're going to pass it to a special instruction.
|
---|
| 180 | const strings = expressionWithArgumentsToExtract.strings;
|
---|
| 181 | if (args.length === 3 && strings[0] === '' && strings[1] === '') {
|
---|
| 182 | // Single argument interpolate instructions.
|
---|
| 183 | args = [args[1]];
|
---|
| 184 | }
|
---|
| 185 | else if (args.length >= 19) {
|
---|
| 186 | // 19 or more arguments must be passed to the `interpolateV`-style instructions, which accept
|
---|
| 187 | // an array of arguments
|
---|
| 188 | args = [o.literalArr(args)];
|
---|
| 189 | }
|
---|
| 190 | }
|
---|
| 191 | return { stmts, args };
|
---|
| 192 | }
|
---|
| 193 | function getStatementsFromVisitor(visitor, bindingId) {
|
---|
| 194 | const stmts = [];
|
---|
| 195 | for (let i = 0; i < visitor.temporaryCount; i++) {
|
---|
| 196 | stmts.push(temporaryDeclaration(bindingId, i));
|
---|
| 197 | }
|
---|
| 198 | return stmts;
|
---|
| 199 | }
|
---|
| 200 | function convertBuiltins(converterFactory, ast) {
|
---|
| 201 | const visitor = new _BuiltinAstConverter(converterFactory);
|
---|
| 202 | return ast.visit(visitor);
|
---|
| 203 | }
|
---|
| 204 | function temporaryName(bindingId, temporaryNumber) {
|
---|
| 205 | return `tmp_${bindingId}_${temporaryNumber}`;
|
---|
| 206 | }
|
---|
| 207 | function temporaryDeclaration(bindingId, temporaryNumber) {
|
---|
| 208 | return new o.DeclareVarStmt(temporaryName(bindingId, temporaryNumber));
|
---|
| 209 | }
|
---|
| 210 | function prependTemporaryDecls(temporaryCount, bindingId, statements) {
|
---|
| 211 | for (let i = temporaryCount - 1; i >= 0; i--) {
|
---|
| 212 | statements.unshift(temporaryDeclaration(bindingId, i));
|
---|
| 213 | }
|
---|
| 214 | }
|
---|
| 215 | var _Mode;
|
---|
| 216 | (function (_Mode) {
|
---|
| 217 | _Mode[_Mode["Statement"] = 0] = "Statement";
|
---|
| 218 | _Mode[_Mode["Expression"] = 1] = "Expression";
|
---|
| 219 | })(_Mode || (_Mode = {}));
|
---|
| 220 | function ensureStatementMode(mode, ast) {
|
---|
| 221 | if (mode !== _Mode.Statement) {
|
---|
| 222 | throw new Error(`Expected a statement, but saw ${ast}`);
|
---|
| 223 | }
|
---|
| 224 | }
|
---|
| 225 | function ensureExpressionMode(mode, ast) {
|
---|
| 226 | if (mode !== _Mode.Expression) {
|
---|
| 227 | throw new Error(`Expected an expression, but saw ${ast}`);
|
---|
| 228 | }
|
---|
| 229 | }
|
---|
| 230 | function convertToStatementIfNeeded(mode, expr) {
|
---|
| 231 | if (mode === _Mode.Statement) {
|
---|
| 232 | return expr.toStmt();
|
---|
| 233 | }
|
---|
| 234 | else {
|
---|
| 235 | return expr;
|
---|
| 236 | }
|
---|
| 237 | }
|
---|
| 238 | class _BuiltinAstConverter extends cdAst.AstTransformer {
|
---|
| 239 | constructor(_converterFactory) {
|
---|
| 240 | super();
|
---|
| 241 | this._converterFactory = _converterFactory;
|
---|
| 242 | }
|
---|
| 243 | visitPipe(ast, context) {
|
---|
| 244 | const args = [ast.exp, ...ast.args].map(ast => ast.visit(this, context));
|
---|
| 245 | return new BuiltinFunctionCall(ast.span, ast.sourceSpan, args, this._converterFactory.createPipeConverter(ast.name, args.length));
|
---|
| 246 | }
|
---|
| 247 | visitLiteralArray(ast, context) {
|
---|
| 248 | const args = ast.expressions.map(ast => ast.visit(this, context));
|
---|
| 249 | return new BuiltinFunctionCall(ast.span, ast.sourceSpan, args, this._converterFactory.createLiteralArrayConverter(ast.expressions.length));
|
---|
| 250 | }
|
---|
| 251 | visitLiteralMap(ast, context) {
|
---|
| 252 | const args = ast.values.map(ast => ast.visit(this, context));
|
---|
| 253 | return new BuiltinFunctionCall(ast.span, ast.sourceSpan, args, this._converterFactory.createLiteralMapConverter(ast.keys));
|
---|
| 254 | }
|
---|
| 255 | }
|
---|
| 256 | class _AstToIrVisitor {
|
---|
| 257 | constructor(_localResolver, _implicitReceiver, bindingId, interpolationFunction, baseSourceSpan, implicitReceiverAccesses) {
|
---|
| 258 | this._localResolver = _localResolver;
|
---|
| 259 | this._implicitReceiver = _implicitReceiver;
|
---|
| 260 | this.bindingId = bindingId;
|
---|
| 261 | this.interpolationFunction = interpolationFunction;
|
---|
| 262 | this.baseSourceSpan = baseSourceSpan;
|
---|
| 263 | this.implicitReceiverAccesses = implicitReceiverAccesses;
|
---|
| 264 | this._nodeMap = new Map();
|
---|
| 265 | this._resultMap = new Map();
|
---|
| 266 | this._currentTemporary = 0;
|
---|
| 267 | this.temporaryCount = 0;
|
---|
| 268 | this.usesImplicitReceiver = false;
|
---|
| 269 | }
|
---|
| 270 | visitUnary(ast, mode) {
|
---|
| 271 | let op;
|
---|
| 272 | switch (ast.operator) {
|
---|
| 273 | case '+':
|
---|
| 274 | op = o.UnaryOperator.Plus;
|
---|
| 275 | break;
|
---|
| 276 | case '-':
|
---|
| 277 | op = o.UnaryOperator.Minus;
|
---|
| 278 | break;
|
---|
| 279 | default:
|
---|
| 280 | throw new Error(`Unsupported operator ${ast.operator}`);
|
---|
| 281 | }
|
---|
| 282 | return convertToStatementIfNeeded(mode, new o.UnaryOperatorExpr(op, this._visit(ast.expr, _Mode.Expression), undefined, this.convertSourceSpan(ast.span)));
|
---|
| 283 | }
|
---|
| 284 | visitBinary(ast, mode) {
|
---|
| 285 | let op;
|
---|
| 286 | switch (ast.operation) {
|
---|
| 287 | case '+':
|
---|
| 288 | op = o.BinaryOperator.Plus;
|
---|
| 289 | break;
|
---|
| 290 | case '-':
|
---|
| 291 | op = o.BinaryOperator.Minus;
|
---|
| 292 | break;
|
---|
| 293 | case '*':
|
---|
| 294 | op = o.BinaryOperator.Multiply;
|
---|
| 295 | break;
|
---|
| 296 | case '/':
|
---|
| 297 | op = o.BinaryOperator.Divide;
|
---|
| 298 | break;
|
---|
| 299 | case '%':
|
---|
| 300 | op = o.BinaryOperator.Modulo;
|
---|
| 301 | break;
|
---|
| 302 | case '&&':
|
---|
| 303 | op = o.BinaryOperator.And;
|
---|
| 304 | break;
|
---|
| 305 | case '||':
|
---|
| 306 | op = o.BinaryOperator.Or;
|
---|
| 307 | break;
|
---|
| 308 | case '==':
|
---|
| 309 | op = o.BinaryOperator.Equals;
|
---|
| 310 | break;
|
---|
| 311 | case '!=':
|
---|
| 312 | op = o.BinaryOperator.NotEquals;
|
---|
| 313 | break;
|
---|
| 314 | case '===':
|
---|
| 315 | op = o.BinaryOperator.Identical;
|
---|
| 316 | break;
|
---|
| 317 | case '!==':
|
---|
| 318 | op = o.BinaryOperator.NotIdentical;
|
---|
| 319 | break;
|
---|
| 320 | case '<':
|
---|
| 321 | op = o.BinaryOperator.Lower;
|
---|
| 322 | break;
|
---|
| 323 | case '>':
|
---|
| 324 | op = o.BinaryOperator.Bigger;
|
---|
| 325 | break;
|
---|
| 326 | case '<=':
|
---|
| 327 | op = o.BinaryOperator.LowerEquals;
|
---|
| 328 | break;
|
---|
| 329 | case '>=':
|
---|
| 330 | op = o.BinaryOperator.BiggerEquals;
|
---|
| 331 | break;
|
---|
| 332 | case '??':
|
---|
| 333 | return this.convertNullishCoalesce(ast, mode);
|
---|
| 334 | default:
|
---|
| 335 | throw new Error(`Unsupported operation ${ast.operation}`);
|
---|
| 336 | }
|
---|
| 337 | return convertToStatementIfNeeded(mode, new o.BinaryOperatorExpr(op, this._visit(ast.left, _Mode.Expression), this._visit(ast.right, _Mode.Expression), undefined, this.convertSourceSpan(ast.span)));
|
---|
| 338 | }
|
---|
| 339 | visitChain(ast, mode) {
|
---|
| 340 | ensureStatementMode(mode, ast);
|
---|
| 341 | return this.visitAll(ast.expressions, mode);
|
---|
| 342 | }
|
---|
| 343 | visitConditional(ast, mode) {
|
---|
| 344 | const value = this._visit(ast.condition, _Mode.Expression);
|
---|
| 345 | return convertToStatementIfNeeded(mode, value.conditional(this._visit(ast.trueExp, _Mode.Expression), this._visit(ast.falseExp, _Mode.Expression), this.convertSourceSpan(ast.span)));
|
---|
| 346 | }
|
---|
| 347 | visitPipe(ast, mode) {
|
---|
| 348 | throw new Error(`Illegal state: Pipes should have been converted into functions. Pipe: ${ast.name}`);
|
---|
| 349 | }
|
---|
| 350 | visitFunctionCall(ast, mode) {
|
---|
| 351 | const convertedArgs = this.visitAll(ast.args, _Mode.Expression);
|
---|
| 352 | let fnResult;
|
---|
| 353 | if (ast instanceof BuiltinFunctionCall) {
|
---|
| 354 | fnResult = ast.converter(convertedArgs);
|
---|
| 355 | }
|
---|
| 356 | else {
|
---|
| 357 | fnResult = this._visit(ast.target, _Mode.Expression)
|
---|
| 358 | .callFn(convertedArgs, this.convertSourceSpan(ast.span));
|
---|
| 359 | }
|
---|
| 360 | return convertToStatementIfNeeded(mode, fnResult);
|
---|
| 361 | }
|
---|
| 362 | visitImplicitReceiver(ast, mode) {
|
---|
| 363 | ensureExpressionMode(mode, ast);
|
---|
| 364 | this.usesImplicitReceiver = true;
|
---|
| 365 | return this._implicitReceiver;
|
---|
| 366 | }
|
---|
| 367 | visitThisReceiver(ast, mode) {
|
---|
| 368 | return this.visitImplicitReceiver(ast, mode);
|
---|
| 369 | }
|
---|
| 370 | visitInterpolation(ast, mode) {
|
---|
| 371 | ensureExpressionMode(mode, ast);
|
---|
| 372 | const args = [o.literal(ast.expressions.length)];
|
---|
| 373 | for (let i = 0; i < ast.strings.length - 1; i++) {
|
---|
| 374 | args.push(o.literal(ast.strings[i]));
|
---|
| 375 | args.push(this._visit(ast.expressions[i], _Mode.Expression));
|
---|
| 376 | }
|
---|
| 377 | args.push(o.literal(ast.strings[ast.strings.length - 1]));
|
---|
| 378 | if (this.interpolationFunction) {
|
---|
| 379 | return this.interpolationFunction(args);
|
---|
| 380 | }
|
---|
| 381 | return ast.expressions.length <= 9 ?
|
---|
| 382 | o.importExpr(Identifiers.inlineInterpolate).callFn(args) :
|
---|
| 383 | o.importExpr(Identifiers.interpolate).callFn([
|
---|
| 384 | args[0], o.literalArr(args.slice(1), undefined, this.convertSourceSpan(ast.span))
|
---|
| 385 | ]);
|
---|
| 386 | }
|
---|
| 387 | visitKeyedRead(ast, mode) {
|
---|
| 388 | const leftMostSafe = this.leftMostSafeNode(ast);
|
---|
| 389 | if (leftMostSafe) {
|
---|
| 390 | return this.convertSafeAccess(ast, leftMostSafe, mode);
|
---|
| 391 | }
|
---|
| 392 | else {
|
---|
| 393 | return convertToStatementIfNeeded(mode, this._visit(ast.receiver, _Mode.Expression).key(this._visit(ast.key, _Mode.Expression)));
|
---|
| 394 | }
|
---|
| 395 | }
|
---|
| 396 | visitKeyedWrite(ast, mode) {
|
---|
| 397 | const obj = this._visit(ast.receiver, _Mode.Expression);
|
---|
| 398 | const key = this._visit(ast.key, _Mode.Expression);
|
---|
| 399 | const value = this._visit(ast.value, _Mode.Expression);
|
---|
| 400 | if (obj === this._implicitReceiver) {
|
---|
| 401 | this._localResolver.maybeRestoreView();
|
---|
| 402 | }
|
---|
| 403 | return convertToStatementIfNeeded(mode, obj.key(key).set(value));
|
---|
| 404 | }
|
---|
| 405 | visitLiteralArray(ast, mode) {
|
---|
| 406 | throw new Error(`Illegal State: literal arrays should have been converted into functions`);
|
---|
| 407 | }
|
---|
| 408 | visitLiteralMap(ast, mode) {
|
---|
| 409 | throw new Error(`Illegal State: literal maps should have been converted into functions`);
|
---|
| 410 | }
|
---|
| 411 | visitLiteralPrimitive(ast, mode) {
|
---|
| 412 | // For literal values of null, undefined, true, or false allow type interference
|
---|
| 413 | // to infer the type.
|
---|
| 414 | const type = ast.value === null || ast.value === undefined || ast.value === true || ast.value === true ?
|
---|
| 415 | o.INFERRED_TYPE :
|
---|
| 416 | undefined;
|
---|
| 417 | return convertToStatementIfNeeded(mode, o.literal(ast.value, type, this.convertSourceSpan(ast.span)));
|
---|
| 418 | }
|
---|
| 419 | _getLocal(name, receiver) {
|
---|
| 420 | var _a;
|
---|
| 421 | if (((_a = this._localResolver.globals) === null || _a === void 0 ? void 0 : _a.has(name)) && receiver instanceof cdAst.ThisReceiver) {
|
---|
| 422 | return null;
|
---|
| 423 | }
|
---|
| 424 | return this._localResolver.getLocal(name);
|
---|
| 425 | }
|
---|
| 426 | visitMethodCall(ast, mode) {
|
---|
| 427 | if (ast.receiver instanceof cdAst.ImplicitReceiver &&
|
---|
| 428 | !(ast.receiver instanceof cdAst.ThisReceiver) && ast.name === '$any') {
|
---|
| 429 | const args = this.visitAll(ast.args, _Mode.Expression);
|
---|
| 430 | if (args.length != 1) {
|
---|
| 431 | throw new Error(`Invalid call to $any, expected 1 argument but received ${args.length || 'none'}`);
|
---|
| 432 | }
|
---|
| 433 | return args[0].cast(o.DYNAMIC_TYPE, this.convertSourceSpan(ast.span));
|
---|
| 434 | }
|
---|
| 435 | const leftMostSafe = this.leftMostSafeNode(ast);
|
---|
| 436 | if (leftMostSafe) {
|
---|
| 437 | return this.convertSafeAccess(ast, leftMostSafe, mode);
|
---|
| 438 | }
|
---|
| 439 | else {
|
---|
| 440 | const args = this.visitAll(ast.args, _Mode.Expression);
|
---|
| 441 | const prevUsesImplicitReceiver = this.usesImplicitReceiver;
|
---|
| 442 | let result = null;
|
---|
| 443 | const receiver = this._visit(ast.receiver, _Mode.Expression);
|
---|
| 444 | if (receiver === this._implicitReceiver) {
|
---|
| 445 | const varExpr = this._getLocal(ast.name, ast.receiver);
|
---|
| 446 | if (varExpr) {
|
---|
| 447 | // Restore the previous "usesImplicitReceiver" state since the implicit
|
---|
| 448 | // receiver has been replaced with a resolved local expression.
|
---|
| 449 | this.usesImplicitReceiver = prevUsesImplicitReceiver;
|
---|
| 450 | result = varExpr.callFn(args);
|
---|
| 451 | this.addImplicitReceiverAccess(ast.name);
|
---|
| 452 | }
|
---|
| 453 | }
|
---|
| 454 | if (result == null) {
|
---|
| 455 | result = receiver.callMethod(ast.name, args, this.convertSourceSpan(ast.span));
|
---|
| 456 | }
|
---|
| 457 | return convertToStatementIfNeeded(mode, result);
|
---|
| 458 | }
|
---|
| 459 | }
|
---|
| 460 | visitPrefixNot(ast, mode) {
|
---|
| 461 | return convertToStatementIfNeeded(mode, o.not(this._visit(ast.expression, _Mode.Expression)));
|
---|
| 462 | }
|
---|
| 463 | visitNonNullAssert(ast, mode) {
|
---|
| 464 | return convertToStatementIfNeeded(mode, o.assertNotNull(this._visit(ast.expression, _Mode.Expression)));
|
---|
| 465 | }
|
---|
| 466 | visitPropertyRead(ast, mode) {
|
---|
| 467 | const leftMostSafe = this.leftMostSafeNode(ast);
|
---|
| 468 | if (leftMostSafe) {
|
---|
| 469 | return this.convertSafeAccess(ast, leftMostSafe, mode);
|
---|
| 470 | }
|
---|
| 471 | else {
|
---|
| 472 | let result = null;
|
---|
| 473 | const prevUsesImplicitReceiver = this.usesImplicitReceiver;
|
---|
| 474 | const receiver = this._visit(ast.receiver, _Mode.Expression);
|
---|
| 475 | if (receiver === this._implicitReceiver) {
|
---|
| 476 | result = this._getLocal(ast.name, ast.receiver);
|
---|
| 477 | if (result) {
|
---|
| 478 | // Restore the previous "usesImplicitReceiver" state since the implicit
|
---|
| 479 | // receiver has been replaced with a resolved local expression.
|
---|
| 480 | this.usesImplicitReceiver = prevUsesImplicitReceiver;
|
---|
| 481 | this.addImplicitReceiverAccess(ast.name);
|
---|
| 482 | }
|
---|
| 483 | }
|
---|
| 484 | if (result == null) {
|
---|
| 485 | result = receiver.prop(ast.name);
|
---|
| 486 | }
|
---|
| 487 | return convertToStatementIfNeeded(mode, result);
|
---|
| 488 | }
|
---|
| 489 | }
|
---|
| 490 | visitPropertyWrite(ast, mode) {
|
---|
| 491 | const receiver = this._visit(ast.receiver, _Mode.Expression);
|
---|
| 492 | const prevUsesImplicitReceiver = this.usesImplicitReceiver;
|
---|
| 493 | let varExpr = null;
|
---|
| 494 | if (receiver === this._implicitReceiver) {
|
---|
| 495 | const localExpr = this._getLocal(ast.name, ast.receiver);
|
---|
| 496 | if (localExpr) {
|
---|
| 497 | if (localExpr instanceof o.ReadPropExpr) {
|
---|
| 498 | // If the local variable is a property read expression, it's a reference
|
---|
| 499 | // to a 'context.property' value and will be used as the target of the
|
---|
| 500 | // write expression.
|
---|
| 501 | varExpr = localExpr;
|
---|
| 502 | // Restore the previous "usesImplicitReceiver" state since the implicit
|
---|
| 503 | // receiver has been replaced with a resolved local expression.
|
---|
| 504 | this.usesImplicitReceiver = prevUsesImplicitReceiver;
|
---|
| 505 | this.addImplicitReceiverAccess(ast.name);
|
---|
| 506 | }
|
---|
| 507 | else {
|
---|
| 508 | // Otherwise it's an error.
|
---|
| 509 | const receiver = ast.name;
|
---|
| 510 | const value = (ast.value instanceof cdAst.PropertyRead) ? ast.value.name : undefined;
|
---|
| 511 | throw new Error(`Cannot assign value "${value}" to template variable "${receiver}". Template variables are read-only.`);
|
---|
| 512 | }
|
---|
| 513 | }
|
---|
| 514 | }
|
---|
| 515 | // If no local expression could be produced, use the original receiver's
|
---|
| 516 | // property as the target.
|
---|
| 517 | if (varExpr === null) {
|
---|
| 518 | varExpr = receiver.prop(ast.name);
|
---|
| 519 | }
|
---|
| 520 | return convertToStatementIfNeeded(mode, varExpr.set(this._visit(ast.value, _Mode.Expression)));
|
---|
| 521 | }
|
---|
| 522 | visitSafePropertyRead(ast, mode) {
|
---|
| 523 | return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);
|
---|
| 524 | }
|
---|
| 525 | visitSafeMethodCall(ast, mode) {
|
---|
| 526 | return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);
|
---|
| 527 | }
|
---|
| 528 | visitSafeKeyedRead(ast, mode) {
|
---|
| 529 | return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);
|
---|
| 530 | }
|
---|
| 531 | visitAll(asts, mode) {
|
---|
| 532 | return asts.map(ast => this._visit(ast, mode));
|
---|
| 533 | }
|
---|
| 534 | visitQuote(ast, mode) {
|
---|
| 535 | throw new Error(`Quotes are not supported for evaluation!
|
---|
| 536 | Statement: ${ast.uninterpretedExpression} located at ${ast.location}`);
|
---|
| 537 | }
|
---|
| 538 | _visit(ast, mode) {
|
---|
| 539 | const result = this._resultMap.get(ast);
|
---|
| 540 | if (result)
|
---|
| 541 | return result;
|
---|
| 542 | return (this._nodeMap.get(ast) || ast).visit(this, mode);
|
---|
| 543 | }
|
---|
| 544 | convertSafeAccess(ast, leftMostSafe, mode) {
|
---|
| 545 | // If the expression contains a safe access node on the left it needs to be converted to
|
---|
| 546 | // an expression that guards the access to the member by checking the receiver for blank. As
|
---|
| 547 | // execution proceeds from left to right, the left most part of the expression must be guarded
|
---|
| 548 | // first but, because member access is left associative, the right side of the expression is at
|
---|
| 549 | // the top of the AST. The desired result requires lifting a copy of the left part of the
|
---|
| 550 | // expression up to test it for blank before generating the unguarded version.
|
---|
| 551 | // Consider, for example the following expression: a?.b.c?.d.e
|
---|
| 552 | // This results in the ast:
|
---|
| 553 | // .
|
---|
| 554 | // / \
|
---|
| 555 | // ?. e
|
---|
| 556 | // / \
|
---|
| 557 | // . d
|
---|
| 558 | // / \
|
---|
| 559 | // ?. c
|
---|
| 560 | // / \
|
---|
| 561 | // a b
|
---|
| 562 | // The following tree should be generated:
|
---|
| 563 | //
|
---|
| 564 | // /---- ? ----\
|
---|
| 565 | // / | \
|
---|
| 566 | // a /--- ? ---\ null
|
---|
| 567 | // / | \
|
---|
| 568 | // . . null
|
---|
| 569 | // / \ / \
|
---|
| 570 | // . c . e
|
---|
| 571 | // / \ / \
|
---|
| 572 | // a b . d
|
---|
| 573 | // / \
|
---|
| 574 | // . c
|
---|
| 575 | // / \
|
---|
| 576 | // a b
|
---|
| 577 | //
|
---|
| 578 | // Notice that the first guard condition is the left hand of the left most safe access node
|
---|
| 579 | // which comes in as leftMostSafe to this routine.
|
---|
| 580 | let guardedExpression = this._visit(leftMostSafe.receiver, _Mode.Expression);
|
---|
| 581 | let temporary = undefined;
|
---|
| 582 | if (this.needsTemporaryInSafeAccess(leftMostSafe.receiver)) {
|
---|
| 583 | // If the expression has method calls or pipes then we need to save the result into a
|
---|
| 584 | // temporary variable to avoid calling stateful or impure code more than once.
|
---|
| 585 | temporary = this.allocateTemporary();
|
---|
| 586 | // Preserve the result in the temporary variable
|
---|
| 587 | guardedExpression = temporary.set(guardedExpression);
|
---|
| 588 | // Ensure all further references to the guarded expression refer to the temporary instead.
|
---|
| 589 | this._resultMap.set(leftMostSafe.receiver, temporary);
|
---|
| 590 | }
|
---|
| 591 | const condition = guardedExpression.isBlank();
|
---|
| 592 | // Convert the ast to an unguarded access to the receiver's member. The map will substitute
|
---|
| 593 | // leftMostNode with its unguarded version in the call to `this.visit()`.
|
---|
| 594 | if (leftMostSafe instanceof cdAst.SafeMethodCall) {
|
---|
| 595 | this._nodeMap.set(leftMostSafe, new cdAst.MethodCall(leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.nameSpan, leftMostSafe.receiver, leftMostSafe.name, leftMostSafe.args, leftMostSafe.argumentSpan));
|
---|
| 596 | }
|
---|
| 597 | else if (leftMostSafe instanceof cdAst.SafeKeyedRead) {
|
---|
| 598 | this._nodeMap.set(leftMostSafe, new cdAst.KeyedRead(leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.receiver, leftMostSafe.key));
|
---|
| 599 | }
|
---|
| 600 | else {
|
---|
| 601 | this._nodeMap.set(leftMostSafe, new cdAst.PropertyRead(leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.nameSpan, leftMostSafe.receiver, leftMostSafe.name));
|
---|
| 602 | }
|
---|
| 603 | // Recursively convert the node now without the guarded member access.
|
---|
| 604 | const access = this._visit(ast, _Mode.Expression);
|
---|
| 605 | // Remove the mapping. This is not strictly required as the converter only traverses each node
|
---|
| 606 | // once but is safer if the conversion is changed to traverse the nodes more than once.
|
---|
| 607 | this._nodeMap.delete(leftMostSafe);
|
---|
| 608 | // If we allocated a temporary, release it.
|
---|
| 609 | if (temporary) {
|
---|
| 610 | this.releaseTemporary(temporary);
|
---|
| 611 | }
|
---|
| 612 | // Produce the conditional
|
---|
| 613 | return convertToStatementIfNeeded(mode, condition.conditional(o.NULL_EXPR, access));
|
---|
| 614 | }
|
---|
| 615 | convertNullishCoalesce(ast, mode) {
|
---|
| 616 | const left = this._visit(ast.left, _Mode.Expression);
|
---|
| 617 | const right = this._visit(ast.right, _Mode.Expression);
|
---|
| 618 | const temporary = this.allocateTemporary();
|
---|
| 619 | this.releaseTemporary(temporary);
|
---|
| 620 | // Generate the following expression. It is identical to how TS
|
---|
| 621 | // transpiles binary expressions with a nullish coalescing operator.
|
---|
| 622 | // let temp;
|
---|
| 623 | // (temp = a) !== null && temp !== undefined ? temp : b;
|
---|
| 624 | return convertToStatementIfNeeded(mode, temporary.set(left)
|
---|
| 625 | .notIdentical(o.NULL_EXPR)
|
---|
| 626 | .and(temporary.notIdentical(o.literal(undefined)))
|
---|
| 627 | .conditional(temporary, right));
|
---|
| 628 | }
|
---|
| 629 | // Given an expression of the form a?.b.c?.d.e then the left most safe node is
|
---|
| 630 | // the (a?.b). The . and ?. are left associative thus can be rewritten as:
|
---|
| 631 | // ((((a?.c).b).c)?.d).e. This returns the most deeply nested safe read or
|
---|
| 632 | // safe method call as this needs to be transformed initially to:
|
---|
| 633 | // a == null ? null : a.c.b.c?.d.e
|
---|
| 634 | // then to:
|
---|
| 635 | // a == null ? null : a.b.c == null ? null : a.b.c.d.e
|
---|
| 636 | leftMostSafeNode(ast) {
|
---|
| 637 | const visit = (visitor, ast) => {
|
---|
| 638 | return (this._nodeMap.get(ast) || ast).visit(visitor);
|
---|
| 639 | };
|
---|
| 640 | return ast.visit({
|
---|
| 641 | visitUnary(ast) {
|
---|
| 642 | return null;
|
---|
| 643 | },
|
---|
| 644 | visitBinary(ast) {
|
---|
| 645 | return null;
|
---|
| 646 | },
|
---|
| 647 | visitChain(ast) {
|
---|
| 648 | return null;
|
---|
| 649 | },
|
---|
| 650 | visitConditional(ast) {
|
---|
| 651 | return null;
|
---|
| 652 | },
|
---|
| 653 | visitFunctionCall(ast) {
|
---|
| 654 | return null;
|
---|
| 655 | },
|
---|
| 656 | visitImplicitReceiver(ast) {
|
---|
| 657 | return null;
|
---|
| 658 | },
|
---|
| 659 | visitThisReceiver(ast) {
|
---|
| 660 | return null;
|
---|
| 661 | },
|
---|
| 662 | visitInterpolation(ast) {
|
---|
| 663 | return null;
|
---|
| 664 | },
|
---|
| 665 | visitKeyedRead(ast) {
|
---|
| 666 | return visit(this, ast.receiver);
|
---|
| 667 | },
|
---|
| 668 | visitKeyedWrite(ast) {
|
---|
| 669 | return null;
|
---|
| 670 | },
|
---|
| 671 | visitLiteralArray(ast) {
|
---|
| 672 | return null;
|
---|
| 673 | },
|
---|
| 674 | visitLiteralMap(ast) {
|
---|
| 675 | return null;
|
---|
| 676 | },
|
---|
| 677 | visitLiteralPrimitive(ast) {
|
---|
| 678 | return null;
|
---|
| 679 | },
|
---|
| 680 | visitMethodCall(ast) {
|
---|
| 681 | return visit(this, ast.receiver);
|
---|
| 682 | },
|
---|
| 683 | visitPipe(ast) {
|
---|
| 684 | return null;
|
---|
| 685 | },
|
---|
| 686 | visitPrefixNot(ast) {
|
---|
| 687 | return null;
|
---|
| 688 | },
|
---|
| 689 | visitNonNullAssert(ast) {
|
---|
| 690 | return null;
|
---|
| 691 | },
|
---|
| 692 | visitPropertyRead(ast) {
|
---|
| 693 | return visit(this, ast.receiver);
|
---|
| 694 | },
|
---|
| 695 | visitPropertyWrite(ast) {
|
---|
| 696 | return null;
|
---|
| 697 | },
|
---|
| 698 | visitQuote(ast) {
|
---|
| 699 | return null;
|
---|
| 700 | },
|
---|
| 701 | visitSafeMethodCall(ast) {
|
---|
| 702 | return visit(this, ast.receiver) || ast;
|
---|
| 703 | },
|
---|
| 704 | visitSafePropertyRead(ast) {
|
---|
| 705 | return visit(this, ast.receiver) || ast;
|
---|
| 706 | },
|
---|
| 707 | visitSafeKeyedRead(ast) {
|
---|
| 708 | return visit(this, ast.receiver) || ast;
|
---|
| 709 | }
|
---|
| 710 | });
|
---|
| 711 | }
|
---|
| 712 | // Returns true of the AST includes a method or a pipe indicating that, if the
|
---|
| 713 | // expression is used as the target of a safe property or method access then
|
---|
| 714 | // the expression should be stored into a temporary variable.
|
---|
| 715 | needsTemporaryInSafeAccess(ast) {
|
---|
| 716 | const visit = (visitor, ast) => {
|
---|
| 717 | return ast && (this._nodeMap.get(ast) || ast).visit(visitor);
|
---|
| 718 | };
|
---|
| 719 | const visitSome = (visitor, ast) => {
|
---|
| 720 | return ast.some(ast => visit(visitor, ast));
|
---|
| 721 | };
|
---|
| 722 | return ast.visit({
|
---|
| 723 | visitUnary(ast) {
|
---|
| 724 | return visit(this, ast.expr);
|
---|
| 725 | },
|
---|
| 726 | visitBinary(ast) {
|
---|
| 727 | return visit(this, ast.left) || visit(this, ast.right);
|
---|
| 728 | },
|
---|
| 729 | visitChain(ast) {
|
---|
| 730 | return false;
|
---|
| 731 | },
|
---|
| 732 | visitConditional(ast) {
|
---|
| 733 | return visit(this, ast.condition) || visit(this, ast.trueExp) || visit(this, ast.falseExp);
|
---|
| 734 | },
|
---|
| 735 | visitFunctionCall(ast) {
|
---|
| 736 | return true;
|
---|
| 737 | },
|
---|
| 738 | visitImplicitReceiver(ast) {
|
---|
| 739 | return false;
|
---|
| 740 | },
|
---|
| 741 | visitThisReceiver(ast) {
|
---|
| 742 | return false;
|
---|
| 743 | },
|
---|
| 744 | visitInterpolation(ast) {
|
---|
| 745 | return visitSome(this, ast.expressions);
|
---|
| 746 | },
|
---|
| 747 | visitKeyedRead(ast) {
|
---|
| 748 | return false;
|
---|
| 749 | },
|
---|
| 750 | visitKeyedWrite(ast) {
|
---|
| 751 | return false;
|
---|
| 752 | },
|
---|
| 753 | visitLiteralArray(ast) {
|
---|
| 754 | return true;
|
---|
| 755 | },
|
---|
| 756 | visitLiteralMap(ast) {
|
---|
| 757 | return true;
|
---|
| 758 | },
|
---|
| 759 | visitLiteralPrimitive(ast) {
|
---|
| 760 | return false;
|
---|
| 761 | },
|
---|
| 762 | visitMethodCall(ast) {
|
---|
| 763 | return true;
|
---|
| 764 | },
|
---|
| 765 | visitPipe(ast) {
|
---|
| 766 | return true;
|
---|
| 767 | },
|
---|
| 768 | visitPrefixNot(ast) {
|
---|
| 769 | return visit(this, ast.expression);
|
---|
| 770 | },
|
---|
| 771 | visitNonNullAssert(ast) {
|
---|
| 772 | return visit(this, ast.expression);
|
---|
| 773 | },
|
---|
| 774 | visitPropertyRead(ast) {
|
---|
| 775 | return false;
|
---|
| 776 | },
|
---|
| 777 | visitPropertyWrite(ast) {
|
---|
| 778 | return false;
|
---|
| 779 | },
|
---|
| 780 | visitQuote(ast) {
|
---|
| 781 | return false;
|
---|
| 782 | },
|
---|
| 783 | visitSafeMethodCall(ast) {
|
---|
| 784 | return true;
|
---|
| 785 | },
|
---|
| 786 | visitSafePropertyRead(ast) {
|
---|
| 787 | return false;
|
---|
| 788 | },
|
---|
| 789 | visitSafeKeyedRead(ast) {
|
---|
| 790 | return false;
|
---|
| 791 | }
|
---|
| 792 | });
|
---|
| 793 | }
|
---|
| 794 | allocateTemporary() {
|
---|
| 795 | const tempNumber = this._currentTemporary++;
|
---|
| 796 | this.temporaryCount = Math.max(this._currentTemporary, this.temporaryCount);
|
---|
| 797 | return new o.ReadVarExpr(temporaryName(this.bindingId, tempNumber));
|
---|
| 798 | }
|
---|
| 799 | releaseTemporary(temporary) {
|
---|
| 800 | this._currentTemporary--;
|
---|
| 801 | if (temporary.name != temporaryName(this.bindingId, this._currentTemporary)) {
|
---|
| 802 | throw new Error(`Temporary ${temporary.name} released out of order`);
|
---|
| 803 | }
|
---|
| 804 | }
|
---|
| 805 | /**
|
---|
| 806 | * Creates an absolute `ParseSourceSpan` from the relative `ParseSpan`.
|
---|
| 807 | *
|
---|
| 808 | * `ParseSpan` objects are relative to the start of the expression.
|
---|
| 809 | * This method converts these to full `ParseSourceSpan` objects that
|
---|
| 810 | * show where the span is within the overall source file.
|
---|
| 811 | *
|
---|
| 812 | * @param span the relative span to convert.
|
---|
| 813 | * @returns a `ParseSourceSpan` for the given span or null if no
|
---|
| 814 | * `baseSourceSpan` was provided to this class.
|
---|
| 815 | */
|
---|
| 816 | convertSourceSpan(span) {
|
---|
| 817 | if (this.baseSourceSpan) {
|
---|
| 818 | const start = this.baseSourceSpan.start.moveBy(span.start);
|
---|
| 819 | const end = this.baseSourceSpan.start.moveBy(span.end);
|
---|
| 820 | const fullStart = this.baseSourceSpan.fullStart.moveBy(span.start);
|
---|
| 821 | return new ParseSourceSpan(start, end, fullStart);
|
---|
| 822 | }
|
---|
| 823 | else {
|
---|
| 824 | return null;
|
---|
| 825 | }
|
---|
| 826 | }
|
---|
| 827 | /** Adds the name of an AST to the list of implicit receiver accesses. */
|
---|
| 828 | addImplicitReceiverAccess(name) {
|
---|
| 829 | if (this.implicitReceiverAccesses) {
|
---|
| 830 | this.implicitReceiverAccesses.add(name);
|
---|
| 831 | }
|
---|
| 832 | }
|
---|
| 833 | }
|
---|
| 834 | function flattenStatements(arg, output) {
|
---|
| 835 | if (Array.isArray(arg)) {
|
---|
| 836 | arg.forEach((entry) => flattenStatements(entry, output));
|
---|
| 837 | }
|
---|
| 838 | else {
|
---|
| 839 | output.push(arg);
|
---|
| 840 | }
|
---|
| 841 | }
|
---|
| 842 | class DefaultLocalResolver {
|
---|
| 843 | constructor(globals) {
|
---|
| 844 | this.globals = globals;
|
---|
| 845 | }
|
---|
| 846 | notifyImplicitReceiverUse() { }
|
---|
| 847 | maybeRestoreView() { }
|
---|
| 848 | getLocal(name) {
|
---|
| 849 | if (name === EventHandlerVars.event.name) {
|
---|
| 850 | return EventHandlerVars.event;
|
---|
| 851 | }
|
---|
| 852 | return null;
|
---|
| 853 | }
|
---|
| 854 | }
|
---|
| 855 | function createCurrValueExpr(bindingId) {
|
---|
| 856 | return o.variable(`currVal_${bindingId}`); // fix syntax highlighting: `
|
---|
| 857 | }
|
---|
| 858 | function createPreventDefaultVar(bindingId) {
|
---|
| 859 | return o.variable(`pd_${bindingId}`);
|
---|
| 860 | }
|
---|
| 861 | function convertStmtIntoExpression(stmt) {
|
---|
| 862 | if (stmt instanceof o.ExpressionStatement) {
|
---|
| 863 | return stmt.expr;
|
---|
| 864 | }
|
---|
| 865 | else if (stmt instanceof o.ReturnStatement) {
|
---|
| 866 | return stmt.value;
|
---|
| 867 | }
|
---|
| 868 | return null;
|
---|
| 869 | }
|
---|
| 870 | export class BuiltinFunctionCall extends cdAst.FunctionCall {
|
---|
| 871 | constructor(span, sourceSpan, args, converter) {
|
---|
| 872 | super(span, sourceSpan, null, args);
|
---|
| 873 | this.converter = converter;
|
---|
| 874 | }
|
---|
| 875 | }
|
---|
| 876 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression_converter.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/compiler_util/expression_converter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as cdAst from '../expression_parser/ast';\nimport {Identifiers} from '../identifiers';\nimport * as o from '../output/output_ast';\nimport {ParseSourceSpan} from '../parse_util';\n\nexport class EventHandlerVars {\n  static event = o.variable('$event');\n}\n\nexport interface LocalResolver {\n  getLocal(name: string): o.Expression|null;\n  notifyImplicitReceiverUse(): void;\n  globals?: Set<string>;\n  maybeRestoreView(): void;\n}\n\nexport class ConvertActionBindingResult {\n  /**\n   * Store statements which are render3 compatible.\n   */\n  render3Stmts: o.Statement[];\n  constructor(\n      /**\n       * Render2 compatible statements,\n       */\n      public stmts: o.Statement[],\n      /**\n       * Variable name used with render2 compatible statements.\n       */\n      public allowDefault: o.ReadVarExpr) {\n    /**\n     * This is bit of a hack. It converts statements which render2 expects to statements which are\n     * expected by render3.\n     *\n     * Example: `<div click=\"doSomething($event)\">` will generate:\n     *\n     * Render3:\n     * ```\n     * const pd_b:any = ((<any>ctx.doSomething($event)) !== false);\n     * return pd_b;\n     * ```\n     *\n     * but render2 expects:\n     * ```\n     * return ctx.doSomething($event);\n     * ```\n     */\n    // TODO(misko): remove this hack once we no longer support ViewEngine.\n    this.render3Stmts = stmts.map((statement: o.Statement) => {\n      if (statement instanceof o.DeclareVarStmt && statement.name == allowDefault.name &&\n          statement.value instanceof o.BinaryOperatorExpr) {\n        const lhs = statement.value.lhs as o.CastExpr;\n        return new o.ReturnStatement(lhs.value);\n      }\n      return statement;\n    });\n  }\n}\n\nexport type InterpolationFunction = (args: o.Expression[]) => o.Expression;\n\n/**\n * Converts the given expression AST into an executable output AST, assuming the expression is\n * used in an action binding (e.g. an event handler).\n */\nexport function convertActionBinding(\n    localResolver: LocalResolver|null, implicitReceiver: o.Expression, action: cdAst.AST,\n    bindingId: string, interpolationFunction?: InterpolationFunction,\n    baseSourceSpan?: ParseSourceSpan, implicitReceiverAccesses?: Set<string>,\n    globals?: Set<string>): ConvertActionBindingResult {\n  if (!localResolver) {\n    localResolver = new DefaultLocalResolver(globals);\n  }\n  const actionWithoutBuiltins = convertPropertyBindingBuiltins(\n      {\n        createLiteralArrayConverter: (argCount: number) => {\n          // Note: no caching for literal arrays in actions.\n          return (args: o.Expression[]) => o.literalArr(args);\n        },\n        createLiteralMapConverter: (keys: {key: string, quoted: boolean}[]) => {\n          // Note: no caching for literal maps in actions.\n          return (values: o.Expression[]) => {\n            const entries = keys.map((k, i) => ({\n                                       key: k.key,\n                                       value: values[i],\n                                       quoted: k.quoted,\n                                     }));\n            return o.literalMap(entries);\n          };\n        },\n        createPipeConverter: (name: string) => {\n          throw new Error(`Illegal State: Actions are not allowed to contain pipes. Pipe: ${name}`);\n        }\n      },\n      action);\n\n  const visitor = new _AstToIrVisitor(\n      localResolver, implicitReceiver, bindingId, interpolationFunction, baseSourceSpan,\n      implicitReceiverAccesses);\n  const actionStmts: o.Statement[] = [];\n  flattenStatements(actionWithoutBuiltins.visit(visitor, _Mode.Statement), actionStmts);\n  prependTemporaryDecls(visitor.temporaryCount, bindingId, actionStmts);\n\n  if (visitor.usesImplicitReceiver) {\n    localResolver.notifyImplicitReceiverUse();\n  }\n\n  const lastIndex = actionStmts.length - 1;\n  let preventDefaultVar: o.ReadVarExpr = null!;\n  if (lastIndex >= 0) {\n    const lastStatement = actionStmts[lastIndex];\n    const returnExpr = convertStmtIntoExpression(lastStatement);\n    if (returnExpr) {\n      // Note: We need to cast the result of the method call to dynamic,\n      // as it might be a void method!\n      preventDefaultVar = createPreventDefaultVar(bindingId);\n      actionStmts[lastIndex] =\n          preventDefaultVar.set(returnExpr.cast(o.DYNAMIC_TYPE).notIdentical(o.literal(false)))\n              .toDeclStmt(null, [o.StmtModifier.Final]);\n    }\n  }\n  return new ConvertActionBindingResult(actionStmts, preventDefaultVar);\n}\n\nexport interface BuiltinConverter {\n  (args: o.Expression[]): o.Expression;\n}\n\nexport interface BuiltinConverterFactory {\n  createLiteralArrayConverter(argCount: number): BuiltinConverter;\n  createLiteralMapConverter(keys: {key: string, quoted: boolean}[]): BuiltinConverter;\n  createPipeConverter(name: string, argCount: number): BuiltinConverter;\n}\n\nexport function convertPropertyBindingBuiltins(\n    converterFactory: BuiltinConverterFactory, ast: cdAst.AST): cdAst.AST {\n  return convertBuiltins(converterFactory, ast);\n}\n\nexport class ConvertPropertyBindingResult {\n  constructor(public stmts: o.Statement[], public currValExpr: o.Expression) {}\n}\n\nexport enum BindingForm {\n  // The general form of binding expression, supports all expressions.\n  General,\n\n  // Try to generate a simple binding (no temporaries or statements)\n  // otherwise generate a general binding\n  TrySimple,\n\n  // Inlines assignment of temporaries into the generated expression. The result may still\n  // have statements attached for declarations of temporary variables.\n  // This is the only relevant form for Ivy, the other forms are only used in ViewEngine.\n  Expression,\n}\n\n/**\n * Converts the given expression AST into an executable output AST, assuming the expression\n * is used in property binding. The expression has to be preprocessed via\n * `convertPropertyBindingBuiltins`.\n */\nexport function convertPropertyBinding(\n    localResolver: LocalResolver|null, implicitReceiver: o.Expression,\n    expressionWithoutBuiltins: cdAst.AST, bindingId: string, form: BindingForm,\n    interpolationFunction?: InterpolationFunction): ConvertPropertyBindingResult {\n  if (!localResolver) {\n    localResolver = new DefaultLocalResolver();\n  }\n  const visitor =\n      new _AstToIrVisitor(localResolver, implicitReceiver, bindingId, interpolationFunction);\n  const outputExpr: o.Expression = expressionWithoutBuiltins.visit(visitor, _Mode.Expression);\n  const stmts: o.Statement[] = getStatementsFromVisitor(visitor, bindingId);\n\n  if (visitor.usesImplicitReceiver) {\n    localResolver.notifyImplicitReceiverUse();\n  }\n\n  if (visitor.temporaryCount === 0 && form == BindingForm.TrySimple) {\n    return new ConvertPropertyBindingResult([], outputExpr);\n  } else if (form === BindingForm.Expression) {\n    return new ConvertPropertyBindingResult(stmts, outputExpr);\n  }\n\n  const currValExpr = createCurrValueExpr(bindingId);\n  stmts.push(currValExpr.set(outputExpr).toDeclStmt(o.DYNAMIC_TYPE, [o.StmtModifier.Final]));\n  return new ConvertPropertyBindingResult(stmts, currValExpr);\n}\n\n/**\n * Given some expression, such as a binding or interpolation expression, and a context expression to\n * look values up on, visit each facet of the given expression resolving values from the context\n * expression such that a list of arguments can be derived from the found values that can be used as\n * arguments to an external update instruction.\n *\n * @param localResolver The resolver to use to look up expressions by name appropriately\n * @param contextVariableExpression The expression representing the context variable used to create\n * the final argument expressions\n * @param expressionWithArgumentsToExtract The expression to visit to figure out what values need to\n * be resolved and what arguments list to build.\n * @param bindingId A name prefix used to create temporary variable names if they're needed for the\n * arguments generated\n * @returns An array of expressions that can be passed as arguments to instruction expressions like\n * `o.importExpr(R3.propertyInterpolate).callFn(result)`\n */\nexport function convertUpdateArguments(\n    localResolver: LocalResolver, contextVariableExpression: o.Expression,\n    expressionWithArgumentsToExtract: cdAst.AST, bindingId: string) {\n  const visitor =\n      new _AstToIrVisitor(localResolver, contextVariableExpression, bindingId, undefined);\n  const outputExpr: o.InvokeFunctionExpr =\n      expressionWithArgumentsToExtract.visit(visitor, _Mode.Expression);\n\n  if (visitor.usesImplicitReceiver) {\n    localResolver.notifyImplicitReceiverUse();\n  }\n\n  const stmts = getStatementsFromVisitor(visitor, bindingId);\n\n  // Removing the first argument, because it was a length for ViewEngine, not Ivy.\n  let args = outputExpr.args.slice(1);\n  if (expressionWithArgumentsToExtract instanceof cdAst.Interpolation) {\n    // If we're dealing with an interpolation of 1 value with an empty prefix and suffix, reduce the\n    // args returned to just the value, because we're going to pass it to a special instruction.\n    const strings = expressionWithArgumentsToExtract.strings;\n    if (args.length === 3 && strings[0] === '' && strings[1] === '') {\n      // Single argument interpolate instructions.\n      args = [args[1]];\n    } else if (args.length >= 19) {\n      // 19 or more arguments must be passed to the `interpolateV`-style instructions, which accept\n      // an array of arguments\n      args = [o.literalArr(args)];\n    }\n  }\n  return {stmts, args};\n}\n\nfunction getStatementsFromVisitor(visitor: _AstToIrVisitor, bindingId: string) {\n  const stmts: o.Statement[] = [];\n  for (let i = 0; i < visitor.temporaryCount; i++) {\n    stmts.push(temporaryDeclaration(bindingId, i));\n  }\n  return stmts;\n}\n\nfunction convertBuiltins(converterFactory: BuiltinConverterFactory, ast: cdAst.AST): cdAst.AST {\n  const visitor = new _BuiltinAstConverter(converterFactory);\n  return ast.visit(visitor);\n}\n\nfunction temporaryName(bindingId: string, temporaryNumber: number): string {\n  return `tmp_${bindingId}_${temporaryNumber}`;\n}\n\nfunction temporaryDeclaration(bindingId: string, temporaryNumber: number): o.Statement {\n  return new o.DeclareVarStmt(temporaryName(bindingId, temporaryNumber));\n}\n\nfunction prependTemporaryDecls(\n    temporaryCount: number, bindingId: string, statements: o.Statement[]) {\n  for (let i = temporaryCount - 1; i >= 0; i--) {\n    statements.unshift(temporaryDeclaration(bindingId, i));\n  }\n}\n\nenum _Mode {\n  Statement,\n  Expression\n}\n\nfunction ensureStatementMode(mode: _Mode, ast: cdAst.AST) {\n  if (mode !== _Mode.Statement) {\n    throw new Error(`Expected a statement, but saw ${ast}`);\n  }\n}\n\nfunction ensureExpressionMode(mode: _Mode, ast: cdAst.AST) {\n  if (mode !== _Mode.Expression) {\n    throw new Error(`Expected an expression, but saw ${ast}`);\n  }\n}\n\nfunction convertToStatementIfNeeded(mode: _Mode, expr: o.Expression): o.Expression|o.Statement {\n  if (mode === _Mode.Statement) {\n    return expr.toStmt();\n  } else {\n    return expr;\n  }\n}\n\nclass _BuiltinAstConverter extends cdAst.AstTransformer {\n  constructor(private _converterFactory: BuiltinConverterFactory) {\n    super();\n  }\n  override visitPipe(ast: cdAst.BindingPipe, context: any): any {\n    const args = [ast.exp, ...ast.args].map(ast => ast.visit(this, context));\n    return new BuiltinFunctionCall(\n        ast.span, ast.sourceSpan, args,\n        this._converterFactory.createPipeConverter(ast.name, args.length));\n  }\n  override visitLiteralArray(ast: cdAst.LiteralArray, context: any): any {\n    const args = ast.expressions.map(ast => ast.visit(this, context));\n    return new BuiltinFunctionCall(\n        ast.span, ast.sourceSpan, args,\n        this._converterFactory.createLiteralArrayConverter(ast.expressions.length));\n  }\n  override visitLiteralMap(ast: cdAst.LiteralMap, context: any): any {\n    const args = ast.values.map(ast => ast.visit(this, context));\n\n    return new BuiltinFunctionCall(\n        ast.span, ast.sourceSpan, args, this._converterFactory.createLiteralMapConverter(ast.keys));\n  }\n}\n\nclass _AstToIrVisitor implements cdAst.AstVisitor {\n  private _nodeMap = new Map<cdAst.AST, cdAst.AST>();\n  private _resultMap = new Map<cdAst.AST, o.Expression>();\n  private _currentTemporary: number = 0;\n  public temporaryCount: number = 0;\n  public usesImplicitReceiver: boolean = false;\n\n  constructor(\n      private _localResolver: LocalResolver, private _implicitReceiver: o.Expression,\n      private bindingId: string, private interpolationFunction: InterpolationFunction|undefined,\n      private baseSourceSpan?: ParseSourceSpan, private implicitReceiverAccesses?: Set<string>) {}\n\n  visitUnary(ast: cdAst.Unary, mode: _Mode): any {\n    let op: o.UnaryOperator;\n    switch (ast.operator) {\n      case '+':\n        op = o.UnaryOperator.Plus;\n        break;\n      case '-':\n        op = o.UnaryOperator.Minus;\n        break;\n      default:\n        throw new Error(`Unsupported operator ${ast.operator}`);\n    }\n\n    return convertToStatementIfNeeded(\n        mode,\n        new o.UnaryOperatorExpr(\n            op, this._visit(ast.expr, _Mode.Expression), undefined,\n            this.convertSourceSpan(ast.span)));\n  }\n\n  visitBinary(ast: cdAst.Binary, mode: _Mode): any {\n    let op: o.BinaryOperator;\n    switch (ast.operation) {\n      case '+':\n        op = o.BinaryOperator.Plus;\n        break;\n      case '-':\n        op = o.BinaryOperator.Minus;\n        break;\n      case '*':\n        op = o.BinaryOperator.Multiply;\n        break;\n      case '/':\n        op = o.BinaryOperator.Divide;\n        break;\n      case '%':\n        op = o.BinaryOperator.Modulo;\n        break;\n      case '&&':\n        op = o.BinaryOperator.And;\n        break;\n      case '||':\n        op = o.BinaryOperator.Or;\n        break;\n      case '==':\n        op = o.BinaryOperator.Equals;\n        break;\n      case '!=':\n        op = o.BinaryOperator.NotEquals;\n        break;\n      case '===':\n        op = o.BinaryOperator.Identical;\n        break;\n      case '!==':\n        op = o.BinaryOperator.NotIdentical;\n        break;\n      case '<':\n        op = o.BinaryOperator.Lower;\n        break;\n      case '>':\n        op = o.BinaryOperator.Bigger;\n        break;\n      case '<=':\n        op = o.BinaryOperator.LowerEquals;\n        break;\n      case '>=':\n        op = o.BinaryOperator.BiggerEquals;\n        break;\n      case '??':\n        return this.convertNullishCoalesce(ast, mode);\n      default:\n        throw new Error(`Unsupported operation ${ast.operation}`);\n    }\n\n    return convertToStatementIfNeeded(\n        mode,\n        new o.BinaryOperatorExpr(\n            op, this._visit(ast.left, _Mode.Expression), this._visit(ast.right, _Mode.Expression),\n            undefined, this.convertSourceSpan(ast.span)));\n  }\n\n  visitChain(ast: cdAst.Chain, mode: _Mode): any {\n    ensureStatementMode(mode, ast);\n    return this.visitAll(ast.expressions, mode);\n  }\n\n  visitConditional(ast: cdAst.Conditional, mode: _Mode): any {\n    const value: o.Expression = this._visit(ast.condition, _Mode.Expression);\n    return convertToStatementIfNeeded(\n        mode,\n        value.conditional(\n            this._visit(ast.trueExp, _Mode.Expression), this._visit(ast.falseExp, _Mode.Expression),\n            this.convertSourceSpan(ast.span)));\n  }\n\n  visitPipe(ast: cdAst.BindingPipe, mode: _Mode): any {\n    throw new Error(\n        `Illegal state: Pipes should have been converted into functions. Pipe: ${ast.name}`);\n  }\n\n  visitFunctionCall(ast: cdAst.FunctionCall, mode: _Mode): any {\n    const convertedArgs = this.visitAll(ast.args, _Mode.Expression);\n    let fnResult: o.Expression;\n    if (ast instanceof BuiltinFunctionCall) {\n      fnResult = ast.converter(convertedArgs);\n    } else {\n      fnResult = this._visit(ast.target!, _Mode.Expression)\n                     .callFn(convertedArgs, this.convertSourceSpan(ast.span));\n    }\n    return convertToStatementIfNeeded(mode, fnResult);\n  }\n\n  visitImplicitReceiver(ast: cdAst.ImplicitReceiver, mode: _Mode): any {\n    ensureExpressionMode(mode, ast);\n    this.usesImplicitReceiver = true;\n    return this._implicitReceiver;\n  }\n\n  visitThisReceiver(ast: cdAst.ThisReceiver, mode: _Mode): any {\n    return this.visitImplicitReceiver(ast, mode);\n  }\n\n  visitInterpolation(ast: cdAst.Interpolation, mode: _Mode): any {\n    ensureExpressionMode(mode, ast);\n    const args = [o.literal(ast.expressions.length)];\n    for (let i = 0; i < ast.strings.length - 1; i++) {\n      args.push(o.literal(ast.strings[i]));\n      args.push(this._visit(ast.expressions[i], _Mode.Expression));\n    }\n    args.push(o.literal(ast.strings[ast.strings.length - 1]));\n\n    if (this.interpolationFunction) {\n      return this.interpolationFunction(args);\n    }\n    return ast.expressions.length <= 9 ?\n        o.importExpr(Identifiers.inlineInterpolate).callFn(args) :\n        o.importExpr(Identifiers.interpolate).callFn([\n          args[0], o.literalArr(args.slice(1), undefined, this.convertSourceSpan(ast.span))\n        ]);\n  }\n\n  visitKeyedRead(ast: cdAst.KeyedRead, mode: _Mode): any {\n    const leftMostSafe = this.leftMostSafeNode(ast);\n    if (leftMostSafe) {\n      return this.convertSafeAccess(ast, leftMostSafe, mode);\n    } else {\n      return convertToStatementIfNeeded(\n          mode,\n          this._visit(ast.receiver, _Mode.Expression).key(this._visit(ast.key, _Mode.Expression)));\n    }\n  }\n\n  visitKeyedWrite(ast: cdAst.KeyedWrite, mode: _Mode): any {\n    const obj: o.Expression = this._visit(ast.receiver, _Mode.Expression);\n    const key: o.Expression = this._visit(ast.key, _Mode.Expression);\n    const value: o.Expression = this._visit(ast.value, _Mode.Expression);\n\n    if (obj === this._implicitReceiver) {\n      this._localResolver.maybeRestoreView();\n    }\n\n    return convertToStatementIfNeeded(mode, obj.key(key).set(value));\n  }\n\n  visitLiteralArray(ast: cdAst.LiteralArray, mode: _Mode): any {\n    throw new Error(`Illegal State: literal arrays should have been converted into functions`);\n  }\n\n  visitLiteralMap(ast: cdAst.LiteralMap, mode: _Mode): any {\n    throw new Error(`Illegal State: literal maps should have been converted into functions`);\n  }\n\n  visitLiteralPrimitive(ast: cdAst.LiteralPrimitive, mode: _Mode): any {\n    // For literal values of null, undefined, true, or false allow type interference\n    // to infer the type.\n    const type =\n        ast.value === null || ast.value === undefined || ast.value === true || ast.value === true ?\n        o.INFERRED_TYPE :\n        undefined;\n    return convertToStatementIfNeeded(\n        mode, o.literal(ast.value, type, this.convertSourceSpan(ast.span)));\n  }\n\n  private _getLocal(name: string, receiver: cdAst.AST): o.Expression|null {\n    if (this._localResolver.globals?.has(name) && receiver instanceof cdAst.ThisReceiver) {\n      return null;\n    }\n\n    return this._localResolver.getLocal(name);\n  }\n\n  visitMethodCall(ast: cdAst.MethodCall, mode: _Mode): any {\n    if (ast.receiver instanceof cdAst.ImplicitReceiver &&\n        !(ast.receiver instanceof cdAst.ThisReceiver) && ast.name === '$any') {\n      const args = this.visitAll(ast.args, _Mode.Expression) as any[];\n      if (args.length != 1) {\n        throw new Error(\n            `Invalid call to $any, expected 1 argument but received ${args.length || 'none'}`);\n      }\n      return (args[0] as o.Expression).cast(o.DYNAMIC_TYPE, this.convertSourceSpan(ast.span));\n    }\n\n    const leftMostSafe = this.leftMostSafeNode(ast);\n    if (leftMostSafe) {\n      return this.convertSafeAccess(ast, leftMostSafe, mode);\n    } else {\n      const args = this.visitAll(ast.args, _Mode.Expression);\n      const prevUsesImplicitReceiver = this.usesImplicitReceiver;\n      let result: any = null;\n      const receiver = this._visit(ast.receiver, _Mode.Expression);\n      if (receiver === this._implicitReceiver) {\n        const varExpr = this._getLocal(ast.name, ast.receiver);\n        if (varExpr) {\n          // Restore the previous \"usesImplicitReceiver\" state since the implicit\n          // receiver has been replaced with a resolved local expression.\n          this.usesImplicitReceiver = prevUsesImplicitReceiver;\n          result = varExpr.callFn(args);\n          this.addImplicitReceiverAccess(ast.name);\n        }\n      }\n      if (result == null) {\n        result = receiver.callMethod(ast.name, args, this.convertSourceSpan(ast.span));\n      }\n      return convertToStatementIfNeeded(mode, result);\n    }\n  }\n\n  visitPrefixNot(ast: cdAst.PrefixNot, mode: _Mode): any {\n    return convertToStatementIfNeeded(mode, o.not(this._visit(ast.expression, _Mode.Expression)));\n  }\n\n  visitNonNullAssert(ast: cdAst.NonNullAssert, mode: _Mode): any {\n    return convertToStatementIfNeeded(\n        mode, o.assertNotNull(this._visit(ast.expression, _Mode.Expression)));\n  }\n\n  visitPropertyRead(ast: cdAst.PropertyRead, mode: _Mode): any {\n    const leftMostSafe = this.leftMostSafeNode(ast);\n    if (leftMostSafe) {\n      return this.convertSafeAccess(ast, leftMostSafe, mode);\n    } else {\n      let result: any = null;\n      const prevUsesImplicitReceiver = this.usesImplicitReceiver;\n      const receiver = this._visit(ast.receiver, _Mode.Expression);\n      if (receiver === this._implicitReceiver) {\n        result = this._getLocal(ast.name, ast.receiver);\n        if (result) {\n          // Restore the previous \"usesImplicitReceiver\" state since the implicit\n          // receiver has been replaced with a resolved local expression.\n          this.usesImplicitReceiver = prevUsesImplicitReceiver;\n          this.addImplicitReceiverAccess(ast.name);\n        }\n      }\n      if (result == null) {\n        result = receiver.prop(ast.name);\n      }\n      return convertToStatementIfNeeded(mode, result);\n    }\n  }\n\n  visitPropertyWrite(ast: cdAst.PropertyWrite, mode: _Mode): any {\n    const receiver: o.Expression = this._visit(ast.receiver, _Mode.Expression);\n    const prevUsesImplicitReceiver = this.usesImplicitReceiver;\n\n    let varExpr: o.ReadPropExpr|null = null;\n    if (receiver === this._implicitReceiver) {\n      const localExpr = this._getLocal(ast.name, ast.receiver);\n      if (localExpr) {\n        if (localExpr instanceof o.ReadPropExpr) {\n          // If the local variable is a property read expression, it's a reference\n          // to a 'context.property' value and will be used as the target of the\n          // write expression.\n          varExpr = localExpr;\n          // Restore the previous \"usesImplicitReceiver\" state since the implicit\n          // receiver has been replaced with a resolved local expression.\n          this.usesImplicitReceiver = prevUsesImplicitReceiver;\n          this.addImplicitReceiverAccess(ast.name);\n        } else {\n          // Otherwise it's an error.\n          const receiver = ast.name;\n          const value = (ast.value instanceof cdAst.PropertyRead) ? ast.value.name : undefined;\n          throw new Error(`Cannot assign value \"${value}\" to template variable \"${\n              receiver}\". Template variables are read-only.`);\n        }\n      }\n    }\n    // If no local expression could be produced, use the original receiver's\n    // property as the target.\n    if (varExpr === null) {\n      varExpr = receiver.prop(ast.name);\n    }\n    return convertToStatementIfNeeded(mode, varExpr.set(this._visit(ast.value, _Mode.Expression)));\n  }\n\n  visitSafePropertyRead(ast: cdAst.SafePropertyRead, mode: _Mode): any {\n    return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);\n  }\n\n  visitSafeMethodCall(ast: cdAst.SafeMethodCall, mode: _Mode): any {\n    return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);\n  }\n\n  visitSafeKeyedRead(ast: cdAst.SafeKeyedRead, mode: _Mode): any {\n    return this.convertSafeAccess(ast, this.leftMostSafeNode(ast), mode);\n  }\n\n  visitAll(asts: cdAst.AST[], mode: _Mode): any {\n    return asts.map(ast => this._visit(ast, mode));\n  }\n\n  visitQuote(ast: cdAst.Quote, mode: _Mode): any {\n    throw new Error(`Quotes are not supported for evaluation!\n        Statement: ${ast.uninterpretedExpression} located at ${ast.location}`);\n  }\n\n  private _visit(ast: cdAst.AST, mode: _Mode): any {\n    const result = this._resultMap.get(ast);\n    if (result) return result;\n    return (this._nodeMap.get(ast) || ast).visit(this, mode);\n  }\n\n  private convertSafeAccess(\n      ast: cdAst.AST, leftMostSafe: cdAst.SafeMethodCall|cdAst.SafePropertyRead|cdAst.SafeKeyedRead,\n      mode: _Mode): any {\n    // If the expression contains a safe access node on the left it needs to be converted to\n    // an expression that guards the access to the member by checking the receiver for blank. As\n    // execution proceeds from left to right, the left most part of the expression must be guarded\n    // first but, because member access is left associative, the right side of the expression is at\n    // the top of the AST. The desired result requires lifting a copy of the left part of the\n    // expression up to test it for blank before generating the unguarded version.\n\n    // Consider, for example the following expression: a?.b.c?.d.e\n\n    // This results in the ast:\n    //         .\n    //        / \\\n    //       ?.   e\n    //      /  \\\n    //     .    d\n    //    / \\\n    //   ?.  c\n    //  /  \\\n    // a    b\n\n    // The following tree should be generated:\n    //\n    //        /---- ? ----\\\n    //       /      |      \\\n    //     a   /--- ? ---\\  null\n    //        /     |     \\\n    //       .      .     null\n    //      / \\    / \\\n    //     .  c   .   e\n    //    / \\    / \\\n    //   a   b  .   d\n    //         / \\\n    //        .   c\n    //       / \\\n    //      a   b\n    //\n    // Notice that the first guard condition is the left hand of the left most safe access node\n    // which comes in as leftMostSafe to this routine.\n\n    let guardedExpression = this._visit(leftMostSafe.receiver, _Mode.Expression);\n    let temporary: o.ReadVarExpr = undefined!;\n    if (this.needsTemporaryInSafeAccess(leftMostSafe.receiver)) {\n      // If the expression has method calls or pipes then we need to save the result into a\n      // temporary variable to avoid calling stateful or impure code more than once.\n      temporary = this.allocateTemporary();\n\n      // Preserve the result in the temporary variable\n      guardedExpression = temporary.set(guardedExpression);\n\n      // Ensure all further references to the guarded expression refer to the temporary instead.\n      this._resultMap.set(leftMostSafe.receiver, temporary);\n    }\n    const condition = guardedExpression.isBlank();\n\n    // Convert the ast to an unguarded access to the receiver's member. The map will substitute\n    // leftMostNode with its unguarded version in the call to `this.visit()`.\n    if (leftMostSafe instanceof cdAst.SafeMethodCall) {\n      this._nodeMap.set(\n          leftMostSafe,\n          new cdAst.MethodCall(\n              leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.nameSpan,\n              leftMostSafe.receiver, leftMostSafe.name, leftMostSafe.args,\n              leftMostSafe.argumentSpan));\n    } else if (leftMostSafe instanceof cdAst.SafeKeyedRead) {\n      this._nodeMap.set(\n          leftMostSafe,\n          new cdAst.KeyedRead(\n              leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.receiver, leftMostSafe.key));\n    } else {\n      this._nodeMap.set(\n          leftMostSafe,\n          new cdAst.PropertyRead(\n              leftMostSafe.span, leftMostSafe.sourceSpan, leftMostSafe.nameSpan,\n              leftMostSafe.receiver, leftMostSafe.name));\n    }\n\n    // Recursively convert the node now without the guarded member access.\n    const access = this._visit(ast, _Mode.Expression);\n\n    // Remove the mapping. This is not strictly required as the converter only traverses each node\n    // once but is safer if the conversion is changed to traverse the nodes more than once.\n    this._nodeMap.delete(leftMostSafe);\n\n    // If we allocated a temporary, release it.\n    if (temporary) {\n      this.releaseTemporary(temporary);\n    }\n\n    // Produce the conditional\n    return convertToStatementIfNeeded(mode, condition.conditional(o.NULL_EXPR, access));\n  }\n\n  private convertNullishCoalesce(ast: cdAst.Binary, mode: _Mode): any {\n    const left: o.Expression = this._visit(ast.left, _Mode.Expression);\n    const right: o.Expression = this._visit(ast.right, _Mode.Expression);\n    const temporary = this.allocateTemporary();\n    this.releaseTemporary(temporary);\n\n    // Generate the following expression. It is identical to how TS\n    // transpiles binary expressions with a nullish coalescing operator.\n    // let temp;\n    // (temp = a) !== null && temp !== undefined ? temp : b;\n    return convertToStatementIfNeeded(\n        mode,\n        temporary.set(left)\n            .notIdentical(o.NULL_EXPR)\n            .and(temporary.notIdentical(o.literal(undefined)))\n            .conditional(temporary, right));\n  }\n\n  // Given an expression of the form a?.b.c?.d.e then the left most safe node is\n  // the (a?.b). The . and ?. are left associative thus can be rewritten as:\n  // ((((a?.c).b).c)?.d).e. This returns the most deeply nested safe read or\n  // safe method call as this needs to be transformed initially to:\n  //   a == null ? null : a.c.b.c?.d.e\n  // then to:\n  //   a == null ? null : a.b.c == null ? null : a.b.c.d.e\n  private leftMostSafeNode(ast: cdAst.AST): cdAst.SafePropertyRead|cdAst.SafeMethodCall\n      |cdAst.SafeKeyedRead {\n    const visit = (visitor: cdAst.AstVisitor, ast: cdAst.AST): any => {\n      return (this._nodeMap.get(ast) || ast).visit(visitor);\n    };\n    return ast.visit({\n      visitUnary(ast: cdAst.Unary) {\n        return null;\n      },\n      visitBinary(ast: cdAst.Binary) {\n        return null;\n      },\n      visitChain(ast: cdAst.Chain) {\n        return null;\n      },\n      visitConditional(ast: cdAst.Conditional) {\n        return null;\n      },\n      visitFunctionCall(ast: cdAst.FunctionCall) {\n        return null;\n      },\n      visitImplicitReceiver(ast: cdAst.ImplicitReceiver) {\n        return null;\n      },\n      visitThisReceiver(ast: cdAst.ThisReceiver) {\n        return null;\n      },\n      visitInterpolation(ast: cdAst.Interpolation) {\n        return null;\n      },\n      visitKeyedRead(ast: cdAst.KeyedRead) {\n        return visit(this, ast.receiver);\n      },\n      visitKeyedWrite(ast: cdAst.KeyedWrite) {\n        return null;\n      },\n      visitLiteralArray(ast: cdAst.LiteralArray) {\n        return null;\n      },\n      visitLiteralMap(ast: cdAst.LiteralMap) {\n        return null;\n      },\n      visitLiteralPrimitive(ast: cdAst.LiteralPrimitive) {\n        return null;\n      },\n      visitMethodCall(ast: cdAst.MethodCall) {\n        return visit(this, ast.receiver);\n      },\n      visitPipe(ast: cdAst.BindingPipe) {\n        return null;\n      },\n      visitPrefixNot(ast: cdAst.PrefixNot) {\n        return null;\n      },\n      visitNonNullAssert(ast: cdAst.NonNullAssert) {\n        return null;\n      },\n      visitPropertyRead(ast: cdAst.PropertyRead) {\n        return visit(this, ast.receiver);\n      },\n      visitPropertyWrite(ast: cdAst.PropertyWrite) {\n        return null;\n      },\n      visitQuote(ast: cdAst.Quote) {\n        return null;\n      },\n      visitSafeMethodCall(ast: cdAst.SafeMethodCall) {\n        return visit(this, ast.receiver) || ast;\n      },\n      visitSafePropertyRead(ast: cdAst.SafePropertyRead) {\n        return visit(this, ast.receiver) || ast;\n      },\n      visitSafeKeyedRead(ast: cdAst.SafeKeyedRead) {\n        return visit(this, ast.receiver) || ast;\n      }\n    });\n  }\n\n  // Returns true of the AST includes a method or a pipe indicating that, if the\n  // expression is used as the target of a safe property or method access then\n  // the expression should be stored into a temporary variable.\n  private needsTemporaryInSafeAccess(ast: cdAst.AST): boolean {\n    const visit = (visitor: cdAst.AstVisitor, ast: cdAst.AST): boolean => {\n      return ast && (this._nodeMap.get(ast) || ast).visit(visitor);\n    };\n    const visitSome = (visitor: cdAst.AstVisitor, ast: cdAst.AST[]): boolean => {\n      return ast.some(ast => visit(visitor, ast));\n    };\n    return ast.visit({\n      visitUnary(ast: cdAst.Unary): boolean {\n        return visit(this, ast.expr);\n      },\n      visitBinary(ast: cdAst.Binary): boolean {\n        return visit(this, ast.left) || visit(this, ast.right);\n      },\n      visitChain(ast: cdAst.Chain) {\n        return false;\n      },\n      visitConditional(ast: cdAst.Conditional): boolean {\n        return visit(this, ast.condition) || visit(this, ast.trueExp) || visit(this, ast.falseExp);\n      },\n      visitFunctionCall(ast: cdAst.FunctionCall) {\n        return true;\n      },\n      visitImplicitReceiver(ast: cdAst.ImplicitReceiver) {\n        return false;\n      },\n      visitThisReceiver(ast: cdAst.ThisReceiver) {\n        return false;\n      },\n      visitInterpolation(ast: cdAst.Interpolation) {\n        return visitSome(this, ast.expressions);\n      },\n      visitKeyedRead(ast: cdAst.KeyedRead) {\n        return false;\n      },\n      visitKeyedWrite(ast: cdAst.KeyedWrite) {\n        return false;\n      },\n      visitLiteralArray(ast: cdAst.LiteralArray) {\n        return true;\n      },\n      visitLiteralMap(ast: cdAst.LiteralMap) {\n        return true;\n      },\n      visitLiteralPrimitive(ast: cdAst.LiteralPrimitive) {\n        return false;\n      },\n      visitMethodCall(ast: cdAst.MethodCall) {\n        return true;\n      },\n      visitPipe(ast: cdAst.BindingPipe) {\n        return true;\n      },\n      visitPrefixNot(ast: cdAst.PrefixNot) {\n        return visit(this, ast.expression);\n      },\n      visitNonNullAssert(ast: cdAst.PrefixNot) {\n        return visit(this, ast.expression);\n      },\n      visitPropertyRead(ast: cdAst.PropertyRead) {\n        return false;\n      },\n      visitPropertyWrite(ast: cdAst.PropertyWrite) {\n        return false;\n      },\n      visitQuote(ast: cdAst.Quote) {\n        return false;\n      },\n      visitSafeMethodCall(ast: cdAst.SafeMethodCall) {\n        return true;\n      },\n      visitSafePropertyRead(ast: cdAst.SafePropertyRead) {\n        return false;\n      },\n      visitSafeKeyedRead(ast: cdAst.SafeKeyedRead) {\n        return false;\n      }\n    });\n  }\n\n  private allocateTemporary(): o.ReadVarExpr {\n    const tempNumber = this._currentTemporary++;\n    this.temporaryCount = Math.max(this._currentTemporary, this.temporaryCount);\n    return new o.ReadVarExpr(temporaryName(this.bindingId, tempNumber));\n  }\n\n  private releaseTemporary(temporary: o.ReadVarExpr) {\n    this._currentTemporary--;\n    if (temporary.name != temporaryName(this.bindingId, this._currentTemporary)) {\n      throw new Error(`Temporary ${temporary.name} released out of order`);\n    }\n  }\n\n  /**\n   * Creates an absolute `ParseSourceSpan` from the relative `ParseSpan`.\n   *\n   * `ParseSpan` objects are relative to the start of the expression.\n   * This method converts these to full `ParseSourceSpan` objects that\n   * show where the span is within the overall source file.\n   *\n   * @param span the relative span to convert.\n   * @returns a `ParseSourceSpan` for the given span or null if no\n   * `baseSourceSpan` was provided to this class.\n   */\n  private convertSourceSpan(span: cdAst.ParseSpan) {\n    if (this.baseSourceSpan) {\n      const start = this.baseSourceSpan.start.moveBy(span.start);\n      const end = this.baseSourceSpan.start.moveBy(span.end);\n      const fullStart = this.baseSourceSpan.fullStart.moveBy(span.start);\n      return new ParseSourceSpan(start, end, fullStart);\n    } else {\n      return null;\n    }\n  }\n\n  /** Adds the name of an AST to the list of implicit receiver accesses. */\n  private addImplicitReceiverAccess(name: string) {\n    if (this.implicitReceiverAccesses) {\n      this.implicitReceiverAccesses.add(name);\n    }\n  }\n}\n\nfunction flattenStatements(arg: any, output: o.Statement[]) {\n  if (Array.isArray(arg)) {\n    (<any[]>arg).forEach((entry) => flattenStatements(entry, output));\n  } else {\n    output.push(arg);\n  }\n}\n\nclass DefaultLocalResolver implements LocalResolver {\n  constructor(public globals?: Set<string>) {}\n  notifyImplicitReceiverUse(): void {}\n  maybeRestoreView(): void {}\n  getLocal(name: string): o.Expression|null {\n    if (name === EventHandlerVars.event.name) {\n      return EventHandlerVars.event;\n    }\n    return null;\n  }\n}\n\nfunction createCurrValueExpr(bindingId: string): o.ReadVarExpr {\n  return o.variable(`currVal_${bindingId}`);  // fix syntax highlighting: `\n}\n\nfunction createPreventDefaultVar(bindingId: string): o.ReadVarExpr {\n  return o.variable(`pd_${bindingId}`);\n}\n\nfunction convertStmtIntoExpression(stmt: o.Statement): o.Expression|null {\n  if (stmt instanceof o.ExpressionStatement) {\n    return stmt.expr;\n  } else if (stmt instanceof o.ReturnStatement) {\n    return stmt.value;\n  }\n  return null;\n}\n\nexport class BuiltinFunctionCall extends cdAst.FunctionCall {\n  constructor(\n      span: cdAst.ParseSpan, sourceSpan: cdAst.AbsoluteSourceSpan, args: cdAst.AST[],\n      public converter: BuiltinConverter) {\n    super(span, sourceSpan, null, args);\n  }\n}\n"]} |
---|