diff --git a/crates/oxc_transformer/src/common/helper_loader.rs b/crates/oxc_transformer/src/common/helper_loader.rs index cefbe44f6a35ef..eb8218e21e67f8 100644 --- a/crates/oxc_transformer/src/common/helper_loader.rs +++ b/crates/oxc_transformer/src/common/helper_loader.rs @@ -159,6 +159,7 @@ pub enum Helper { ClassPrivateFieldLooseKey, ClassPrivateFieldLooseBase, SuperPropGet, + SuperPropSet, } impl Helper { @@ -183,6 +184,7 @@ impl Helper { Self::ClassPrivateFieldLooseKey => "classPrivateFieldLooseKey", Self::ClassPrivateFieldLooseBase => "classPrivateFieldLooseBase", Self::SuperPropGet => "superPropGet", + Self::SuperPropSet => "superPropSet", } } } diff --git a/crates/oxc_transformer/src/es2022/class_properties/private_field.rs b/crates/oxc_transformer/src/es2022/class_properties/private_field.rs index cd2f257c8d780e..0e7c5ae86b8d5a 100644 --- a/crates/oxc_transformer/src/es2022/class_properties/private_field.rs +++ b/crates/oxc_transformer/src/es2022/class_properties/private_field.rs @@ -1608,7 +1608,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> { /// * Anything else `foo()` -> `_foo = foo()`, `_foo` /// /// Returns 2 `Expression`s. The first must be inserted into output first. - fn duplicate_object( + pub(super) fn duplicate_object( &self, object: Expression<'a>, ctx: &mut TraverseCtx<'a>, diff --git a/crates/oxc_transformer/src/es2022/class_properties/static_prop_init.rs b/crates/oxc_transformer/src/es2022/class_properties/static_prop_init.rs index c8024dc6a39a17..2d85e959838206 100644 --- a/crates/oxc_transformer/src/es2022/class_properties/static_prop_init.rs +++ b/crates/oxc_transformer/src/es2022/class_properties/static_prop_init.rs @@ -195,9 +195,15 @@ impl<'a, 'ctx, 'v> VisitMut<'a> for StaticInitializerVisitor<'a, 'ctx, 'v> { self.transform_call_expression_if_super_member_expression(call_expr); self.class_properties.transform_call_expression(expr, self.ctx); } + // `super.prop = value`, `super.prop += value`, `super.prop ??= value` or // `object.#prop = value`, `object.#prop += value`, `object.#prop ??= value` etc Expression::AssignmentExpression(_) => { - self.class_properties.transform_assignment_expression(expr, self.ctx); + self.transform_assignment_expression_if_super_member_assignment_target(expr); + // Check again if it's an assignment expression, because it could have been transformed + // to other expression. + if matches!(expr, Expression::AssignmentExpression(_)) { + self.class_properties.transform_assignment_expression(expr, self.ctx); + } } // `object.#prop++`, `--object.#prop` Expression::UpdateExpression(_) => { @@ -521,4 +527,14 @@ impl<'a, 'ctx, 'v> StaticInitializerVisitor<'a, 'ctx, 'v> { .transform_call_expression_for_super_member_expr(call_expr, self.ctx); } } + + fn transform_assignment_expression_if_super_member_assignment_target( + &mut self, + expr: &mut Expression<'a>, + ) { + if self.this_depth == 0 { + self.class_properties + .transform_assignment_expression_for_super_assignment_target(expr, self.ctx); + } + } } diff --git a/crates/oxc_transformer/src/es2022/class_properties/supers.rs b/crates/oxc_transformer/src/es2022/class_properties/supers.rs index cbeabac882dc42..247010c22ca046 100644 --- a/crates/oxc_transformer/src/es2022/class_properties/supers.rs +++ b/crates/oxc_transformer/src/es2022/class_properties/supers.rs @@ -125,6 +125,133 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> { arguments.push(Argument::from(array)); } + /// Transform assignment expression where the left-hand side is a member expression with `super`. + /// + /// * `super.prop = value` + /// -> `_superPropSet(_Class, prop, value, _Class)` + /// * `super.prop += value` + /// -> `_superPropSet(_Class, prop, _superPropGet(_Class, prop, _Class) + value, _Class)` + /// * `super.prop &&= value` + /// -> `_superPropGet(_Class, prop, _Class) && _superPropSet(_Class, prop, value, _Class)` + // + // `#[inline]` so can bail out fast without a function call if `left` is not a member expression + // with `super` as member expression object (fairly rare). + // Actual transform is broken out into separate functions. + pub(super) fn transform_assignment_expression_for_super_assignment_target( + &mut self, + expr: &mut Expression<'a>, + ctx: &mut TraverseCtx<'a>, + ) { + let Expression::AssignmentExpression(assign_expr) = expr else { unreachable!() }; + match &assign_expr.left { + AssignmentTarget::StaticMemberExpression(member) if member.object.is_super() => { + self.transform_assignment_expression_for_super_static_member_expr(expr, ctx); + } + AssignmentTarget::ComputedMemberExpression(member) if member.object.is_super() => { + self.transform_assignment_expression_for_super_computed_member_expr(expr, ctx); + } + _ => {} + }; + } + + /// Transform assignment expression where the left-hand side is a static member expression + /// with `super`. + // + // `#[inline]` so that compiler sees that `expr` is an `Expression::AssignmentExpression`. + #[inline] + fn transform_assignment_expression_for_super_static_member_expr( + &mut self, + expr: &mut Expression<'a>, + ctx: &mut TraverseCtx<'a>, + ) { + let Expression::AssignmentExpression(assign_expr) = ctx.ast.move_expression(expr) else { + unreachable!() + }; + let AssignmentExpression { span, operator, right: value, left } = assign_expr.unbox(); + let AssignmentTarget::StaticMemberExpression(member) = left else { unreachable!() }; + let property = ctx.ast.expression_string_literal( + member.property.span, + member.property.name.clone(), + None, + ); + *expr = + self.transform_super_assignment_expression_impl(span, operator, property, value, ctx); + } + + /// Transform assignment expression where the left-hand side is a computed member expression + /// with `super`. + /// + // `#[inline]` so that compiler sees that `expr` is an `Expression::AssignmentExpression`. + #[inline] + fn transform_assignment_expression_for_super_computed_member_expr( + &mut self, + expr: &mut Expression<'a>, + ctx: &mut TraverseCtx<'a>, + ) { + let Expression::AssignmentExpression(assign_expr) = ctx.ast.move_expression(expr) else { + unreachable!() + }; + let AssignmentExpression { span, operator, right: value, left } = assign_expr.unbox(); + let AssignmentTarget::ComputedMemberExpression(member) = left else { unreachable!() }; + let property = member.unbox().expression; + *expr = + self.transform_super_assignment_expression_impl(span, operator, property, value, ctx); + } + + /// Transform assignment expression with instance private prop as assignee. + /// + /// * `super.prop = value` + /// -> `_superPropSet(_Class, _prop, value, _Class, 1)` + /// * `super.prop += value` + /// -> `_superPropSet(_Class, _prop, _superPropGet(_Class, _prop, _Class) + value, 1)` + /// * `super.prop &&= value` + /// -> `_superPropGet(_Class, _prop, _Class) && _superPropSet(_Class, _prop, value, _Class, 1)` + fn transform_super_assignment_expression_impl( + &mut self, + span: Span, + operator: AssignmentOperator, + property: Expression<'a>, + value: Expression<'a>, + ctx: &mut TraverseCtx<'a>, + ) -> Expression<'a> { + if operator == AssignmentOperator::Assign { + // `super.prop = value` -> `_superPropSet(_Class, _prop, value, _Class, 1)` + self.create_super_prop_set(span, property, value, ctx) + } else { + // Make 2 copies of `object` + let (property1, property2) = self.duplicate_object(property, ctx); + + if let Some(operator) = operator.to_binary_operator() { + // `super.prop += value` + // -> `_superPropSet(_Class, _prop, _superPropGet(_Class, _prop, _Class) + value, _Class, 1)` + + // `_superPropGet(_Class, _prop, _Class)` + let get_call = self.create_super_prop_get(SPAN, property2, false, ctx); + + // `_superPropGet(_Class, _prop, _Class) + value` + let value = ctx.ast.expression_binary(SPAN, get_call, operator, value); + + // `_superPropSet(_Class, _prop, _superPropGet(_Class, _prop, _Class) + value, 1)` + self.create_super_prop_set(span, property1, value, ctx) + } else if let Some(operator) = operator.to_logical_operator() { + // `super.prop &&= value` + // -> `_superPropGet(_Class, _prop, _Class) && _superPropSet(_Class, _prop, value, _Class, 1)` + + // `_superPropGet(_Class, _prop, _Class)` + let get_call = self.create_super_prop_get(SPAN, property1, false, ctx); + + // `_superPropSet(_Class, _prop, value, _Class, 1)` + let set_call = self.create_super_prop_set(span, property2, value, ctx); + + // `_superPropGet(_Class, _prop, _Class) && _superPropSet(_Class, _prop, value, _Class, 1)` + ctx.ast.expression_logical(span, get_call, operator, set_call) + } else { + // The above covers all types of `AssignmentOperator` + unreachable!(); + } + } + } + /// Member: /// `_superPropGet(_Class, prop, _Class)` /// @@ -155,4 +282,28 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> { // `_superPropGet(_Class, prop, _Class)` or `_superPropGet(_Class, prop, _Class, 2)` self.ctx.helper_call_expr(Helper::SuperPropGet, span, arguments, ctx) } + + /// `_superPropSet(_Class, prop, value, _Class, 1)` + fn create_super_prop_set( + &mut self, + span: Span, + property: Expression<'a>, + value: Expression<'a>, + ctx: &mut TraverseCtx<'a>, + ) -> Expression<'a> { + let temp_binding = self.current_class_mut().bindings.get_or_init_static_binding(ctx); + let arguments = ctx.ast.vec_from_array([ + Argument::from(temp_binding.create_read_expression(ctx)), + Argument::from(property), + Argument::from(value), + Argument::from(temp_binding.create_read_expression(ctx)), + Argument::from(ctx.ast.expression_numeric_literal( + SPAN, + 1.0, + None, + NumberBase::Decimal, + )), + ]); + self.ctx.helper_call_expr(Helper::SuperPropSet, span, arguments, ctx) + } } diff --git a/tasks/transform_conformance/snapshots/babel_exec.snap.md b/tasks/transform_conformance/snapshots/babel_exec.snap.md index afa017ab6f152f..8820341f83c907 100644 --- a/tasks/transform_conformance/snapshots/babel_exec.snap.md +++ b/tasks/transform_conformance/snapshots/babel_exec.snap.md @@ -1,132 +1,8 @@ commit: 54a8389f node: v22.12.0 +filter: fixtures/babel +include: **/*.{test,spec}.?(c|m)[jt]s?(x) +exclude: **/node_modules/**, **/dist/**, **/cypress/**, **/.{idea,git,cache,output,temp}/**, **/{karma,rollup,webpack,vite,vitest,jest,ava,babel,nyc,cypress,tsup,build,eslint,prettier}.config.*, "" -Passed: 191 of 215 (88.84%) - -Failures: - -./fixtures/babel/babel-plugin-transform-arrow-functions-test-fixtures-arrow-functions-implicit-var-arguments-exec.test.js -'eval' and 'arguments' cannot be used as a binding identifier in strict mode - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-assumption-setPublicClassFields-static-infer-name-exec.test.js -AssertionError: expected '_Class' to be 'Foo' // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-assumption-setPublicClassFields-static-infer-name-exec.test.js:8:19 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-nested-class-super-call-in-decorator-exec.test.js -AssertionError: expected undefined to be 'hello' // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-nested-class-super-call-in-decorator-exec.test.js:21:28 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-nested-class-super-property-in-accessor-key-exec.test.js -Unexpected token `[`. Expected * for generator, private key, identifier or async - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-nested-class-super-property-in-decorator-exec.test.js -AssertionError: expected undefined to be 'hello' // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-nested-class-super-property-in-decorator-exec.test.js:22:28 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-nested-class-computed-redeclared-exec.test.js -Private field '#foo' must be declared in an enclosing class - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-nested-class-extends-computed-exec.test.js -AssertionError: expected [Function] to not throw an error but 'TypeError: attempted to use private f…' was thrown - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:1438:21) - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:923:17) - at Proxy.methodWrapper (./node_modules/.pnpm/chai@5.1.2/node_modules/chai/chai.js:1610:25) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-nested-class-extends-computed-exec.test.js:36:9 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js -TypeError: Cannot convert undefined or null to object - at hasOwnProperty () - at _classPrivateFieldBase (./node_modules/.pnpm/@babel+runtime@7.26.0/node_modules/@babel/runtime/helpers/classPrivateFieldLooseBase.js:2:26) - at value (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js:63:11) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js:44:198 - at j (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js:45:6) - at Function.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js:52:11) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-in-function-param-with-transform-exec.test.js:71:6 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-member-optional-call-with-transform-exec.test.js -TypeError: Cannot convert undefined or null to object - at hasOwnProperty () - at _classPrivateFieldBase (./node_modules/.pnpm/@babel+runtime@7.26.0/node_modules/@babel/runtime/helpers/classPrivateFieldLooseBase.js:2:26) - at value (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-member-optional-call-with-transform-exec.test.js:123:11) - at Function.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-member-optional-call-with-transform-exec.test.js:24:134) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-optional-chain-member-optional-call-with-transform-exec.test.js:131:6 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-exec.test.js -TypeError: Cannot read properties of undefined (reading 'bind') - at Foo.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-exec.test.js:20:59) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-exec.test.js:78:12 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-with-transform-exec.test.js -TypeError: Cannot read properties of undefined (reading 'bind') - at Foo.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-with-transform-exec.test.js:20:59) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-loose-parenthesized-optional-member-call-with-transform-exec.test.js:78:12 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-nested-class-computed-redeclared-exec.test.js -Private field '#foo' must be declared in an enclosing class - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-nested-class-extends-computed-exec.test.js -AssertionError: expected [Function] to not throw an error but 'TypeError: Private element is not pre…' was thrown - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:1438:21) - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:923:17) - at Proxy.methodWrapper (./node_modules/.pnpm/chai@5.1.2/node_modules/chai/chai.js:1610:25) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-nested-class-extends-computed-exec.test.js:31:9 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-static-shadow-exec.test.js -TypeError: e.has is not a function - at _assertClassBrand (./node_modules/.pnpm/@babel+runtime@7.26.0/node_modules/@babel/runtime/helpers/assertClassBrand.js:2:44) - at func (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-static-shadow-exec.test.js:10:12) - at Function.method (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-static-shadow-exec.test.js:12:11) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-private-static-shadow-exec.test.js:16:14 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-computed-toPrimitive-exec.test.js -AssertionError: expected [Function] to throw error including '@@toPrimitive must return a primitive…' but got 'Cannot convert object to primitive va…' - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:1438:21) - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:923:17) - at Proxy.methodWrapper (./node_modules/.pnpm/chai@5.1.2/node_modules/chai/chai.js:1610:25) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-computed-toPrimitive-exec.test.js:37:5 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-delete-super-property-exec.test.js -AssertionError: expected function to throw an error, but it didn't - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-delete-super-property-exec.test.js:25:5 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-loose-static-infer-name-exec.test.js -AssertionError: expected '_Class' to be 'Foo' // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-loose-static-infer-name-exec.test.js:8:19 - -./fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-static-infer-name-exec.test.js -AssertionError: expected '_Class' to be 'Foo' // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-class-properties-test-fixtures-public-static-infer-name-exec.test.js:9:19 - -./fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-assumption-noDocumentAll-parenthesized-expression-member-call-exec.test.js -TypeError: Cannot read properties of undefined (reading 'x') - at m (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-assumption-noDocumentAll-parenthesized-expression-member-call-exec.test.js:10:16) - at Foo.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-assumption-noDocumentAll-parenthesized-expression-member-call-exec.test.js:25:63) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-assumption-noDocumentAll-parenthesized-expression-member-call-exec.test.js:68:12 - -./fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-exec.test.js -TypeError: Cannot read properties of undefined (reading 'x') - at m (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-exec.test.js:10:16) - at Foo.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-exec.test.js:25:63) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-exec.test.js:68:12 - -./fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-loose-exec.test.js -TypeError: Cannot read properties of undefined (reading 'x') - at m (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-loose-exec.test.js:10:16) - at Foo.test (./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-loose-exec.test.js:25:63) - at ./tasks/transform_conformance/fixtures/babel/babel-plugin-transform-optional-chaining-test-fixtures-general-parenthesized-expression-member-call-loose-exec.test.js:68:12 - -./fixtures/babel/babel-preset-env-test-fixtures-plugins-integration-issue-15170-exec.test.js -AssertionError: expected [Function] to not throw an error but 'ReferenceError: x is not defined' was thrown - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:1438:21) - at Proxy. (./node_modules/.pnpm/@vitest+expect@2.1.2/node_modules/@vitest/expect/dist/index.js:923:17) - at Proxy.methodWrapper (./node_modules/.pnpm/chai@5.1.2/node_modules/chai/chai.js:1610:25) - at ./tasks/transform_conformance/fixtures/babel/babel-preset-env-test-fixtures-plugins-integration-issue-15170-exec.test.js:6:9 - -./fixtures/babel/babel-preset-env-test-fixtures-sanity-check-es2015-constants-exec.test.js -TypeError: Assignment to constant variable. - at ./tasks/transform_conformance/fixtures/babel/babel-preset-env-test-fixtures-sanity-check-es2015-constants-exec.test.js:5:6 - -./fixtures/babel/babel-preset-env-test-fixtures-sanity-regex-dot-all-exec.test.js -AssertionError: expected false to be true // Object.is equality - at ./tasks/transform_conformance/fixtures/babel/babel-preset-env-test-fixtures-sanity-regex-dot-all-exec.test.js:10:37 +No test files found, exiting with code 1 diff --git a/tasks/transform_conformance/snapshots/oxc.snap.md b/tasks/transform_conformance/snapshots/oxc.snap.md index b44ec0405f484f..8ae859d65a050a 100644 --- a/tasks/transform_conformance/snapshots/oxc.snap.md +++ b/tasks/transform_conformance/snapshots/oxc.snap.md @@ -1,6 +1,6 @@ commit: 54a8389f -Passed: 112/126 +Passed: 113/127 # All Passed: * babel-plugin-transform-class-static-block @@ -16,7 +16,7 @@ Passed: 112/126 * regexp -# babel-plugin-transform-class-properties (13/15) +# babel-plugin-transform-class-properties (14/16) * typescript/optional-call/input.ts Symbol reference IDs mismatch for "X": after transform: SymbolId(0): [ReferenceId(0), ReferenceId(2), ReferenceId(6), ReferenceId(11), ReferenceId(16)] diff --git a/tasks/transform_conformance/snapshots/oxc_exec.snap.md b/tasks/transform_conformance/snapshots/oxc_exec.snap.md index 23e62427ba1c38..1c3b4e2a74fc8a 100644 --- a/tasks/transform_conformance/snapshots/oxc_exec.snap.md +++ b/tasks/transform_conformance/snapshots/oxc_exec.snap.md @@ -1,5 +1,8 @@ commit: 54a8389f node: v22.12.0 +filter: fixtures/oxc +include: **/*.{test,spec}.?(c|m)[jt]s?(x) +exclude: **/node_modules/**, **/dist/**, **/cypress/**, **/.{idea,git,cache,output,temp}/**, **/{karma,rollup,webpack,vite,vitest,jest,ava,babel,nyc,cypress,tsup,build,eslint,prettier}.config.*, "" -Passed: 3 of 3 (100.00%) +No test files found, exiting with code 1 diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/input.js b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/input.js new file mode 100644 index 00000000000000..057eec64eb2861 --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/input.js @@ -0,0 +1,49 @@ +const ident = "A"; +class Outer { + static A = 0; + static B = () => { + super.A += 1; + super.A -= 1; + super.A &&= 1; + super.A ||= 1; + super.A = 1; + + super[ident] += 1; + super[ident] -= 1; + super[ident] &&= 1; + super[ident] ||= 1; + super[ident] = 1; + + class Inner { + method() { + // Don't transform + super.A += 1; + super.A -= 1; + super.A &&= 1; + super.A ||= 1; + super.A = 1; + + super[ident] += 1; + super[ident] -= 1; + super[ident] &&= 1; + super[ident] ||= 1; + super[ident] = 1; + } + + static staticMethod() { + // Don't transform + super.A += 1; + super.A -= 1; + super.A &&= 1; + super.A ||= 1; + super.A = 1; + + super[ident] += 1; + super[ident] -= 1; + super[ident] &&= 1; + super[ident] ||= 1; + super[ident] = 1; + } + } + }; +} diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/output.js b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/output.js new file mode 100644 index 00000000000000..41d7e78447782b --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/static-super-assignment-expression/output.js @@ -0,0 +1,72 @@ +var _Outer; +const ident = "A"; +class Outer {} +_Outer = Outer; +babelHelpers.defineProperty(Outer, "A", 0); +babelHelpers.defineProperty(Outer, "B", () => { + babelHelpers.superPropSet( + _Outer, + "A", + babelHelpers.superPropGet(_Outer, "A", _Outer) + 1, + _Outer, + 1, + ); + babelHelpers.superPropSet( + _Outer, + "A", + babelHelpers.superPropGet(_Outer, "A", _Outer) - 1, + _Outer, + 1, + ); + babelHelpers.superPropGet(_Outer, "A", _Outer) && + babelHelpers.superPropSet(_Outer, "A", 1, _Outer, 1); + babelHelpers.superPropGet(_Outer, "A", _Outer) || + babelHelpers.superPropSet(_Outer, "A", 1, _Outer, 1); + babelHelpers.superPropSet(_Outer, "A", 1, _Outer, 1); + babelHelpers.superPropSet( + _Outer, + ident, + babelHelpers.superPropGet(_Outer, ident, _Outer) + 1, + _Outer, + 1, + ); + babelHelpers.superPropSet( + _Outer, + ident, + babelHelpers.superPropGet(_Outer, ident, _Outer) - 1, + _Outer, + 1, + ); + babelHelpers.superPropGet(_Outer, ident, _Outer) && + babelHelpers.superPropSet(_Outer, ident, 1, _Outer, 1); + babelHelpers.superPropGet(_Outer, ident, _Outer) || + babelHelpers.superPropSet(_Outer, ident, 1, _Outer, 1); + babelHelpers.superPropSet(_Outer, ident, 1, _Outer, 1); + + class Inner { + method() { + super.A += 1; + super.A -= 1; + super.A &&= 1; + super.A ||= 1; + super.A = 1; + super[ident] += 1; + super[ident] -= 1; + super[ident] &&= 1; + super[ident] ||= 1; + super[ident] = 1; + } + static staticMethod() { + super.A += 1; + super.A -= 1; + super.A &&= 1; + super.A ||= 1; + super.A = 1; + super[ident] += 1; + super[ident] -= 1; + super[ident] &&= 1; + super[ident] ||= 1; + super[ident] = 1; + } + } +});