Skip to content

Commit

Permalink
Allow decorator on private members and class expressions (#14548)
Browse files Browse the repository at this point in the history
  • Loading branch information
fisker committed Mar 21, 2023
1 parent ddf3b43 commit 44aef0b
Show file tree
Hide file tree
Showing 8 changed files with 97 additions and 91 deletions.
24 changes: 24 additions & 0 deletions changelog_unreleased/typescript/14548.md
@@ -0,0 +1,24 @@
#### Allow decorators on private members and class expressions (#14548 by @fisker)

<!-- prettier-ignore -->
```ts
// Input
class A {
@decorator()
#privateMethod () {}
}

// Prettier stable
SyntaxError: Decorators are not valid here. (2:3)
1 | class A {
> 2 | @decorator()
| ^^^^^^^^^^^^
3 | #privateMethod () {}
4 | }

// Prettier main
class A {
@decorator()
#privateMethod() {}
}
```
30 changes: 17 additions & 13 deletions src/language-js/parse/postprocess/typescript.js
Expand Up @@ -30,6 +30,20 @@ function throwErrorOnTsNode(node, message) {
throwSyntaxError({ loc: { start, end } }, message);
}

function nodeCanBeDecorated(node) {
const ts = require("typescript");

return [true, false].some((useLegacyDecorators) =>
// @ts-expect-error -- internal?
ts.nodeCanBeDecorated(
useLegacyDecorators,
node,
node.parent,
node.parent.parent
)
);
}

// Invalid decorators are removed since `@typescript-eslint/typescript-estree` v4
// https://github.com/typescript-eslint/typescript-eslint/pull/2375
// There is a `checkGrammarDecorators` in `typescript` package, consider use it directly in future
Expand All @@ -44,16 +58,7 @@ function throwErrorForInvalidDecorator(node) {
const { SyntaxKind } = ts;
for (const modifier of modifiers) {
if (ts.isDecorator(modifier)) {
const legacyDecorators = true;
if (
// @ts-expect-error -- internal?
!ts.nodeCanBeDecorated(
legacyDecorators,
node,
node.parent,
node.parent.parent
)
) {
if (!nodeCanBeDecorated(node)) {
if (
node.kind === SyntaxKind.MethodDeclaration &&
// @ts-expect-error -- internal?
Expand All @@ -67,9 +72,8 @@ function throwErrorForInvalidDecorator(node) {
throwErrorOnTsNode(modifier, "Decorators are not valid here.");
}
} else if (
legacyDecorators &&
(node.kind === SyntaxKind.GetAccessor ||
node.kind === SyntaxKind.SetAccessor)
node.kind === SyntaxKind.GetAccessor ||
node.kind === SyntaxKind.SetAccessor
) {
// @ts-expect-error -- internal?
const accessors = ts.getAllAccessorDeclarations(
Expand Down
Expand Up @@ -24,14 +24,6 @@ exports[`arguments.js [flow] format 1`] = `
3 |"
`;

exports[`arguments.js [typescript] format 1`] = `
"Decorators are not valid here. (1:13)
> 1 | console.log(@deco class Foo {})
| ^^^^^
2 | console.log(@deco class {})
3 |"
`;

exports[`arguments.js - {"semi":false} [acorn] format 1`] = `
"Unexpected character '@' (1:13)
> 1 | console.log(@deco class Foo {})
Expand All @@ -56,14 +48,6 @@ exports[`arguments.js - {"semi":false} [flow] format 1`] = `
3 |"
`;

exports[`arguments.js - {"semi":false} [typescript] format 1`] = `
"Decorators are not valid here. (1:13)
> 1 | console.log(@deco class Foo {})
| ^^^^^
2 | console.log(@deco class {})
3 |"
`;

exports[`arguments.js - {"semi":false} format 1`] = `
====================================options=====================================
parsers: ["babel", "flow", "typescript"]
Expand Down Expand Up @@ -144,15 +128,6 @@ exports[`class-expression.js [flow] format 1`] = `
4 | (@deco class Foo {});"
`;

exports[`class-expression.js [typescript] format 1`] = `
"Decorators are not valid here. (1:13)
> 1 | const a1 = (@deco class Foo {});
| ^^^^^
2 | const a2 = (@deco class {});
3 |
4 | (@deco class Foo {});"
`;

exports[`class-expression.js - {"semi":false} [acorn] format 1`] = `
"Unexpected character '@' (1:13)
> 1 | const a1 = (@deco class Foo {});
Expand Down Expand Up @@ -180,15 +155,6 @@ exports[`class-expression.js - {"semi":false} [flow] format 1`] = `
4 | (@deco class Foo {});"
`;

exports[`class-expression.js - {"semi":false} [typescript] format 1`] = `
"Decorators are not valid here. (1:13)
> 1 | const a1 = (@deco class Foo {});
| ^^^^^
2 | const a2 = (@deco class {});
3 |
4 | (@deco class Foo {});"
`;

exports[`class-expression.js - {"semi":false} format 1`] = `
====================================options=====================================
parsers: ["babel", "flow", "typescript"]
Expand Down Expand Up @@ -336,14 +302,6 @@ exports[`member-expression.js [flow] format 1`] = `
3 |"
`;

exports[`member-expression.js [typescript] format 1`] = `
"Decorators are not valid here. (1:2)
> 1 | (@deco class Foo {}).name;
| ^^^^^
2 | (@deco class {}).name;
3 |"
`;

exports[`member-expression.js - {"semi":false} [acorn] format 1`] = `
"Unexpected character '@' (1:2)
> 1 | (@deco class Foo {}).name;
Expand All @@ -368,14 +326,6 @@ exports[`member-expression.js - {"semi":false} [flow] format 1`] = `
3 |"
`;

exports[`member-expression.js - {"semi":false} [typescript] format 1`] = `
"Decorators are not valid here. (1:2)
> 1 | (@deco class Foo {}).name;
| ^^^^^
2 | (@deco class {}).name;
3 |"
`;

exports[`member-expression.js - {"semi":false} format 1`] = `
====================================options=====================================
parsers: ["babel", "flow", "typescript"]
Expand Down Expand Up @@ -448,15 +398,6 @@ exports[`super-class.js [flow] format 1`] = `
4 |"
`;

exports[`super-class.js [typescript] format 1`] = `
"Decorators are not valid here. (1:20)
> 1 | class Foo extends (@deco class Foo {}){}
| ^^^^^
2 |
3 | class Foo extends (@deco class {}){}
4 |"
`;

exports[`super-class.js - {"semi":false} [acorn] format 1`] = `
"Unexpected character '@' (1:20)
> 1 | class Foo extends (@deco class Foo {}){}
Expand Down Expand Up @@ -484,15 +425,6 @@ exports[`super-class.js - {"semi":false} [flow] format 1`] = `
4 |"
`;

exports[`super-class.js - {"semi":false} [typescript] format 1`] = `
"Decorators are not valid here. (1:20)
> 1 | class Foo extends (@deco class Foo {}){}
| ^^^^^
2 |
3 | class Foo extends (@deco class {}){}
4 |"
`;

exports[`super-class.js - {"semi":false} format 1`] = `
====================================options=====================================
parsers: ["babel", "flow", "typescript"]
Expand Down
1 change: 0 additions & 1 deletion tests/format/js/decorators/class-expression/jsfmt.spec.js
@@ -1,6 +1,5 @@
const errors = {
flow: true,
typescript: true,
acorn: true,
espree: true,
};
Expand Down
Expand Up @@ -497,6 +497,48 @@ class MyContainerComponent {
================================================================================
`;
exports[`legacy.ts format 1`] = `
====================================options=====================================
parsers: ["typescript"]
printWidth: 80
| printWidth
=====================================input======================================
[
@decorator() class {},
@decorator() class A {},
];
class A {
@decorator() accessor #field;
}
class B {
@decorator() #field () {}
}
=====================================output=====================================
[
(
@decorator()
class {}
),
(
@decorator()
class A {}
),
];
class A {
@decorator() accessor #field;
}
class B {
@decorator() #field() {}
}
================================================================================
`;
exports[`mobx.ts format 1`] = `
====================================options=====================================
parsers: ["typescript"]
Expand Down
12 changes: 12 additions & 0 deletions tests/format/typescript/decorators/legacy.ts
@@ -0,0 +1,12 @@
[
@decorator() class {},
@decorator() class A {},
];

class A {
@decorator() accessor #field;
}

class B {
@decorator() #field () {}
}
Expand Up @@ -49,7 +49,7 @@ exports[`3.ts [typescript] format 1`] = `
2 |"
`;
exports[`4.ts [babel-ts] format 1`] = `
exports[`4.ts format 1`] = `
====================================options=====================================
parsers: ["typescript"]
printWidth: 80
Expand All @@ -66,13 +66,6 @@ printWidth: 80
================================================================================
`;
exports[`4.ts [typescript] format 1`] = `
"Decorators are not valid here. (1:2)
> 1 | (@f<<T>(v: T) => void>() class {});
| ^^^^^^^^^^^^^^^^^^^^^^^
2 |"
`;
exports[`5.tsx [babel-ts] format 1`] = `
====================================options=====================================
parsers: ["typescript"]
Expand Down
@@ -1,3 +1,3 @@
run_spec(__dirname, ["typescript"], {
errors: { typescript: ["3.ts", "4.ts", "5.tsx"] },
errors: { typescript: ["3.ts", "5.tsx"] },
});

0 comments on commit 44aef0b

Please sign in to comment.