@angular/compiler 17.3.0-rc.0 → 17.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -29,10 +29,19 @@ export function transformTwoWayBindingSet(job) {
29
29
  }
30
30
  }
31
31
  function wrapSetOperation(target, value) {
32
+ // ASSUMPTION: here we're assuming that `ReadVariableExpr` will be a reference
33
+ // to a local template variable. This appears to be the case at the time of writing.
34
+ // If the expression is targeting a variable read, we only emit the `twoWayBindingSet` since
35
+ // the fallback would be attempting to write into a constant. Invalid usages will be flagged
36
+ // during template type checking.
37
+ if (target instanceof ir.ReadVariableExpr) {
38
+ return ng.twoWayBindingSet(target, value);
39
+ }
32
40
  return ng.twoWayBindingSet(target, value).or(target.set(value));
33
41
  }
34
42
  function isReadExpression(value) {
35
- return value instanceof o.ReadPropExpr || value instanceof o.ReadKeyExpr;
43
+ return value instanceof o.ReadPropExpr || value instanceof o.ReadKeyExpr ||
44
+ value instanceof ir.ReadVariableExpr;
36
45
  }
37
46
  function wrapAction(target, value) {
38
47
  // The only officially supported expressions inside of a two-way binding are read expressions.
@@ -76,4 +85,4 @@ function wrapAction(target, value) {
76
85
  }
77
86
  throw new Error(`Unsupported expression in two-way action binding.`);
78
87
  }
79
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform_two_way_binding_set.js","sourceRoot":"","sources":["../../../../../../../../../../packages/compiler/src/template/pipeline/src/phases/transform_two_way_binding_set.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 o from '../../../../output/output_ast';\nimport * as ir from '../../ir';\nimport * as ng from '../instruction';\nimport type {CompilationJob} from '../compilation';\n\n/**\n * Transforms a `TwoWayBindingSet` expression into an expression that either\n * sets a value through the `twoWayBindingSet` instruction or falls back to setting\n * the value directly. E.g. the expression `TwoWayBindingSet(target, value)` becomes:\n * `ng.twoWayBindingSet(target, value) || (target = value)`.\n */\nexport function transformTwoWayBindingSet(job: CompilationJob): void {\n  for (const unit of job.units) {\n    for (const op of unit.create) {\n      if (op.kind === ir.OpKind.TwoWayListener) {\n        ir.transformExpressionsInOp(op, (expr) => {\n          if (expr instanceof ir.TwoWayBindingSetExpr) {\n            return wrapAction(expr.target, expr.value);\n          }\n          return expr;\n        }, ir.VisitorContextFlag.InChildOperation);\n      }\n    }\n  }\n}\n\nfunction wrapSetOperation(target: o.ReadPropExpr|o.ReadKeyExpr, value: o.Expression): o.Expression {\n  return ng.twoWayBindingSet(target, value).or(target.set(value));\n}\n\nfunction isReadExpression(value: unknown): value is o.ReadPropExpr|o.ReadKeyExpr {\n  return value instanceof o.ReadPropExpr || value instanceof o.ReadKeyExpr;\n}\n\nfunction wrapAction(target: o.Expression, value: o.Expression): o.Expression {\n  // The only officially supported expressions inside of a two-way binding are read expressions.\n  if (isReadExpression(target)) {\n    return wrapSetOperation(target, value);\n  }\n\n  // However, historically the expression parser was handling two-way events by appending `=$event`\n  // to the raw string before attempting to parse it. This has led to bugs over the years (see\n  // #37809) and to unintentionally supporting unassignable events in the two-way binding. The\n  // logic below aims to emulate the old behavior while still supporting the new output format\n  // which uses `twoWayBindingSet`. Note that the generated code doesn't necessarily make sense\n  // based on what the user wrote, for example the event binding for `[(value)]=\"a ? b : c\"`\n  // would produce `ctx.a ? ctx.b : ctx.c = $event`. We aim to reproduce what the parser used\n  // to generate before #54154.\n  if (target instanceof o.BinaryOperatorExpr && isReadExpression(target.rhs)) {\n    // `a && b` -> `ctx.a && twoWayBindingSet(ctx.b, $event) || (ctx.b = $event)`\n    return new o.BinaryOperatorExpr(\n        target.operator, target.lhs, wrapSetOperation(target.rhs, value));\n  }\n\n  // Note: this also supports nullish coalescing expressions which\n  // would've been downleveled to ternary expressions by this point.\n  if (target instanceof o.ConditionalExpr && isReadExpression(target.falseCase)) {\n    // `a ? b : c` -> `ctx.a ? ctx.b : twoWayBindingSet(ctx.c, $event) || (ctx.c = $event)`\n    return new o.ConditionalExpr(\n        target.condition, target.trueCase, wrapSetOperation(target.falseCase, value));\n  }\n\n  // `!!a` -> `twoWayBindingSet(ctx.a, $event) || (ctx.a = $event)`\n  // Note: previously we'd actually produce `!!(ctx.a = $event)`, but the wrapping\n  // node doesn't affect the result so we don't need to carry it over.\n  if (target instanceof o.NotExpr) {\n    let expr = target.condition;\n\n    while (true) {\n      if (expr instanceof o.NotExpr) {\n        expr = expr.condition;\n      } else {\n        if (isReadExpression(expr)) {\n          return wrapSetOperation(expr, value);\n        }\n        break;\n      }\n    }\n  }\n\n  throw new Error(`Unsupported expression in two-way action binding.`);\n}\n"]}
88
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform_two_way_binding_set.js","sourceRoot":"","sources":["../../../../../../../../../../packages/compiler/src/template/pipeline/src/phases/transform_two_way_binding_set.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 o from '../../../../output/output_ast';\nimport * as ir from '../../ir';\nimport * as ng from '../instruction';\nimport type {CompilationJob} from '../compilation';\n\n/**\n * Transforms a `TwoWayBindingSet` expression into an expression that either\n * sets a value through the `twoWayBindingSet` instruction or falls back to setting\n * the value directly. E.g. the expression `TwoWayBindingSet(target, value)` becomes:\n * `ng.twoWayBindingSet(target, value) || (target = value)`.\n */\nexport function transformTwoWayBindingSet(job: CompilationJob): void {\n  for (const unit of job.units) {\n    for (const op of unit.create) {\n      if (op.kind === ir.OpKind.TwoWayListener) {\n        ir.transformExpressionsInOp(op, (expr) => {\n          if (expr instanceof ir.TwoWayBindingSetExpr) {\n            return wrapAction(expr.target, expr.value);\n          }\n          return expr;\n        }, ir.VisitorContextFlag.InChildOperation);\n      }\n    }\n  }\n}\n\nfunction wrapSetOperation(\n    target: o.ReadPropExpr|o.ReadKeyExpr|ir.ReadVariableExpr, value: o.Expression): o.Expression {\n  // ASSUMPTION: here we're assuming that `ReadVariableExpr` will be a reference\n  // to a local template variable. This appears to be the case at the time of writing.\n  // If the expression is targeting a variable read, we only emit the `twoWayBindingSet` since\n  // the fallback would be attempting to write into a constant. Invalid usages will be flagged\n  // during template type checking.\n  if (target instanceof ir.ReadVariableExpr) {\n    return ng.twoWayBindingSet(target, value);\n  }\n\n  return ng.twoWayBindingSet(target, value).or(target.set(value));\n}\n\nfunction isReadExpression(value: unknown): value is o.ReadPropExpr|o.ReadKeyExpr|\n    ir.ReadVariableExpr {\n  return value instanceof o.ReadPropExpr || value instanceof o.ReadKeyExpr ||\n      value instanceof ir.ReadVariableExpr;\n}\n\nfunction wrapAction(target: o.Expression, value: o.Expression): o.Expression {\n  // The only officially supported expressions inside of a two-way binding are read expressions.\n  if (isReadExpression(target)) {\n    return wrapSetOperation(target, value);\n  }\n\n  // However, historically the expression parser was handling two-way events by appending `=$event`\n  // to the raw string before attempting to parse it. This has led to bugs over the years (see\n  // #37809) and to unintentionally supporting unassignable events in the two-way binding. The\n  // logic below aims to emulate the old behavior while still supporting the new output format\n  // which uses `twoWayBindingSet`. Note that the generated code doesn't necessarily make sense\n  // based on what the user wrote, for example the event binding for `[(value)]=\"a ? b : c\"`\n  // would produce `ctx.a ? ctx.b : ctx.c = $event`. We aim to reproduce what the parser used\n  // to generate before #54154.\n  if (target instanceof o.BinaryOperatorExpr && isReadExpression(target.rhs)) {\n    // `a && b` -> `ctx.a && twoWayBindingSet(ctx.b, $event) || (ctx.b = $event)`\n    return new o.BinaryOperatorExpr(\n        target.operator, target.lhs, wrapSetOperation(target.rhs, value));\n  }\n\n  // Note: this also supports nullish coalescing expressions which\n  // would've been downleveled to ternary expressions by this point.\n  if (target instanceof o.ConditionalExpr && isReadExpression(target.falseCase)) {\n    // `a ? b : c` -> `ctx.a ? ctx.b : twoWayBindingSet(ctx.c, $event) || (ctx.c = $event)`\n    return new o.ConditionalExpr(\n        target.condition, target.trueCase, wrapSetOperation(target.falseCase, value));\n  }\n\n  // `!!a` -> `twoWayBindingSet(ctx.a, $event) || (ctx.a = $event)`\n  // Note: previously we'd actually produce `!!(ctx.a = $event)`, but the wrapping\n  // node doesn't affect the result so we don't need to carry it over.\n  if (target instanceof o.NotExpr) {\n    let expr = target.condition;\n\n    while (true) {\n      if (expr instanceof o.NotExpr) {\n        expr = expr.condition;\n      } else {\n        if (isReadExpression(expr)) {\n          return wrapSetOperation(expr, value);\n        }\n        break;\n      }\n    }\n  }\n\n  throw new Error(`Unsupported expression in two-way action binding.`);\n}\n"]}
@@ -11,5 +11,5 @@
11
11
  * Entry point for all public APIs of the compiler package.
12
12
  */
13
13
  import { Version } from './util';
14
- export const VERSION = new Version('17.3.0-rc.0');
14
+ export const VERSION = new Version('17.3.1');
15
15
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvbXBpbGVyL3NyYy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUVIOzs7O0dBSUc7QUFFSCxPQUFPLEVBQUMsT0FBTyxFQUFDLE1BQU0sUUFBUSxDQUFDO0FBRS9CLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxJQUFJLE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbi8qKlxuICogQG1vZHVsZVxuICogQGRlc2NyaXB0aW9uXG4gKiBFbnRyeSBwb2ludCBmb3IgYWxsIHB1YmxpYyBBUElzIG9mIHRoZSBjb21waWxlciBwYWNrYWdlLlxuICovXG5cbmltcG9ydCB7VmVyc2lvbn0gZnJvbSAnLi91dGlsJztcblxuZXhwb3J0IGNvbnN0IFZFUlNJT04gPSBuZXcgVmVyc2lvbignMC4wLjAtUExBQ0VIT0xERVInKTtcbiJdfQ==
@@ -1,5 +1,5 @@
1
1
  /**
2
- * @license Angular v17.3.0-rc.0
2
+ * @license Angular v17.3.1
3
3
  * (c) 2010-2022 Google LLC. https://angular.io/
4
4
  * License: MIT
5
5
  */
@@ -23169,10 +23169,19 @@ function transformTwoWayBindingSet(job) {
23169
23169
  }
23170
23170
  }
23171
23171
  function wrapSetOperation(target, value) {
23172
+ // ASSUMPTION: here we're assuming that `ReadVariableExpr` will be a reference
23173
+ // to a local template variable. This appears to be the case at the time of writing.
23174
+ // If the expression is targeting a variable read, we only emit the `twoWayBindingSet` since
23175
+ // the fallback would be attempting to write into a constant. Invalid usages will be flagged
23176
+ // during template type checking.
23177
+ if (target instanceof ReadVariableExpr) {
23178
+ return twoWayBindingSet(target, value);
23179
+ }
23172
23180
  return twoWayBindingSet(target, value).or(target.set(value));
23173
23181
  }
23174
23182
  function isReadExpression(value) {
23175
- return value instanceof ReadPropExpr || value instanceof ReadKeyExpr;
23183
+ return value instanceof ReadPropExpr || value instanceof ReadKeyExpr ||
23184
+ value instanceof ReadVariableExpr;
23176
23185
  }
23177
23186
  function wrapAction(target, value) {
23178
23187
  // The only officially supported expressions inside of a two-way binding are read expressions.
@@ -25328,15 +25337,23 @@ function ingestControlFlowInsertionPoint(unit, xref, node) {
25328
25337
  root = child;
25329
25338
  }
25330
25339
  }
25331
- // If we've found a single root node, its tag name and *static* attributes can be copied
25332
- // to the surrounding template to be used for content projection. Note that it's important
25333
- // that we don't copy any bound attributes since they don't participate in content projection
25334
- // and they can be used in directive matching (in the case of `Template.templateAttrs`).
25340
+ // If we've found a single root node, its tag name and attributes can be
25341
+ // copied to the surrounding template to be used for content projection.
25335
25342
  if (root !== null) {
25343
+ // Collect the static attributes for content projection purposes.
25336
25344
  for (const attr of root.attributes) {
25337
25345
  const securityContext = domSchema.securityContext(NG_TEMPLATE_TAG_NAME$1, attr.name, true);
25338
25346
  unit.update.push(createBindingOp(xref, BindingKind.Attribute, attr.name, literal(attr.value), null, securityContext, true, false, null, asMessage(attr.i18n), attr.sourceSpan));
25339
25347
  }
25348
+ // Also collect the inputs since they participate in content projection as well.
25349
+ // Note that TDB used to collect the outputs as well, but it wasn't passing them into
25350
+ // the template instruction. Here we just don't collect them.
25351
+ for (const attr of root.inputs) {
25352
+ if (attr.type !== 4 /* e.BindingType.Animation */ && attr.type !== 1 /* e.BindingType.Attribute */) {
25353
+ const securityContext = domSchema.securityContext(NG_TEMPLATE_TAG_NAME$1, attr.name, true);
25354
+ unit.create.push(createExtractedAttributeOp(xref, BindingKind.Property, null, attr.name, null, null, null, securityContext));
25355
+ }
25356
+ }
25340
25357
  const tagName = root instanceof Element$1 ? root.name : root.tagName;
25341
25358
  // Don't pass along `ng-template` tag name since it enables directive matching.
25342
25359
  return tagName === NG_TEMPLATE_TAG_NAME$1 ? null : tagName;
@@ -26156,6 +26173,9 @@ function parseForLoopParameters(block, errors, bindingParser) {
26156
26173
  }
26157
26174
  else {
26158
26175
  const expression = parseBlockParameterToBinding(param, bindingParser, trackMatch[1]);
26176
+ if (expression.ast instanceof EmptyExpr$1) {
26177
+ errors.push(new ParseError(param.sourceSpan, '@for loop must have a "track" expression'));
26178
+ }
26159
26179
  const keywordSpan = new ParseSourceSpan(param.sourceSpan.start, param.sourceSpan.start.moveBy('track'.length));
26160
26180
  result.trackBy = { expression, keywordSpan };
26161
26181
  }
@@ -32806,7 +32826,7 @@ function publishFacade(global) {
32806
32826
  * @description
32807
32827
  * Entry point for all public APIs of the compiler package.
32808
32828
  */
32809
- const VERSION = new Version('17.3.0-rc.0');
32829
+ const VERSION = new Version('17.3.1');
32810
32830
 
32811
32831
  class CompilerConfig {
32812
32832
  constructor({ defaultEncapsulation = ViewEncapsulation.Emulated, preserveWhitespaces, strictInjectionParameters } = {}) {
@@ -34374,7 +34394,7 @@ const MINIMUM_PARTIAL_LINKER_VERSION$5 = '12.0.0';
34374
34394
  function compileDeclareClassMetadata(metadata) {
34375
34395
  const definitionMap = new DefinitionMap();
34376
34396
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION$5));
34377
- definitionMap.set('version', literal('17.3.0-rc.0'));
34397
+ definitionMap.set('version', literal('17.3.1'));
34378
34398
  definitionMap.set('ngImport', importExpr(Identifiers.core));
34379
34399
  definitionMap.set('type', metadata.type);
34380
34400
  definitionMap.set('decorators', metadata.decorators);
@@ -34470,7 +34490,7 @@ function createDirectiveDefinitionMap(meta) {
34470
34490
  const definitionMap = new DefinitionMap();
34471
34491
  const minVersion = getMinimumVersionForPartialOutput(meta);
34472
34492
  definitionMap.set('minVersion', literal(minVersion));
34473
- definitionMap.set('version', literal('17.3.0-rc.0'));
34493
+ definitionMap.set('version', literal('17.3.1'));
34474
34494
  // e.g. `type: MyDirective`
34475
34495
  definitionMap.set('type', meta.type.value);
34476
34496
  if (meta.isStandalone) {
@@ -34862,7 +34882,7 @@ const MINIMUM_PARTIAL_LINKER_VERSION$4 = '12.0.0';
34862
34882
  function compileDeclareFactoryFunction(meta) {
34863
34883
  const definitionMap = new DefinitionMap();
34864
34884
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION$4));
34865
- definitionMap.set('version', literal('17.3.0-rc.0'));
34885
+ definitionMap.set('version', literal('17.3.1'));
34866
34886
  definitionMap.set('ngImport', importExpr(Identifiers.core));
34867
34887
  definitionMap.set('type', meta.type.value);
34868
34888
  definitionMap.set('deps', compileDependencies(meta.deps));
@@ -34897,7 +34917,7 @@ function compileDeclareInjectableFromMetadata(meta) {
34897
34917
  function createInjectableDefinitionMap(meta) {
34898
34918
  const definitionMap = new DefinitionMap();
34899
34919
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION$3));
34900
- definitionMap.set('version', literal('17.3.0-rc.0'));
34920
+ definitionMap.set('version', literal('17.3.1'));
34901
34921
  definitionMap.set('ngImport', importExpr(Identifiers.core));
34902
34922
  definitionMap.set('type', meta.type.value);
34903
34923
  // Only generate providedIn property if it has a non-null value
@@ -34948,7 +34968,7 @@ function compileDeclareInjectorFromMetadata(meta) {
34948
34968
  function createInjectorDefinitionMap(meta) {
34949
34969
  const definitionMap = new DefinitionMap();
34950
34970
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION$2));
34951
- definitionMap.set('version', literal('17.3.0-rc.0'));
34971
+ definitionMap.set('version', literal('17.3.1'));
34952
34972
  definitionMap.set('ngImport', importExpr(Identifiers.core));
34953
34973
  definitionMap.set('type', meta.type.value);
34954
34974
  definitionMap.set('providers', meta.providers);
@@ -34981,7 +35001,7 @@ function createNgModuleDefinitionMap(meta) {
34981
35001
  throw new Error('Invalid path! Local compilation mode should not get into the partial compilation path');
34982
35002
  }
34983
35003
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION$1));
34984
- definitionMap.set('version', literal('17.3.0-rc.0'));
35004
+ definitionMap.set('version', literal('17.3.1'));
34985
35005
  definitionMap.set('ngImport', importExpr(Identifiers.core));
34986
35006
  definitionMap.set('type', meta.type.value);
34987
35007
  // We only generate the keys in the metadata if the arrays contain values.
@@ -35032,7 +35052,7 @@ function compileDeclarePipeFromMetadata(meta) {
35032
35052
  function createPipeDefinitionMap(meta) {
35033
35053
  const definitionMap = new DefinitionMap();
35034
35054
  definitionMap.set('minVersion', literal(MINIMUM_PARTIAL_LINKER_VERSION));
35035
- definitionMap.set('version', literal('17.3.0-rc.0'));
35055
+ definitionMap.set('version', literal('17.3.1'));
35036
35056
  definitionMap.set('ngImport', importExpr(Identifiers.core));
35037
35057
  // e.g. `type: MyPipe`
35038
35058
  definitionMap.set('type', meta.type.value);