-
-
Notifications
You must be signed in to change notification settings - Fork 381
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add prefer-class-fields
rule
#2512
base: main
Are you sure you want to change the base?
Conversation
fixup "contructor" typos
const isThisAssignmentExpression = node => { | ||
if ( | ||
node.type !== 'ExpressionStatement' | ||
|| node.expression.type !== 'AssignmentExpression' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The operator didn't checked
class Foo {
constructor() {
this.foo += 'foo';
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You're right, it should be ignored. Fixed here: 21e3cf1
|
||
const lhs = node.expression.left; | ||
|
||
if (!lhs.object || lhs.object.type !== 'ThisExpression') { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The left property didn't checked
class Foo {
constructor() {
this[foo.bar] = 'foo';
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think in this case we should ignore any kind of reporting - foo.bar
might be dynamic and checking if it is or not is a rabbit hole with potentially lots of nested jumps.
I can handle cases like this['foo'] = 'foo';
though
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added a valid test case to cover the dynamic case: 633829c
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
And here: 0e6afb3 added support for simple strings & template strings.
I ended up removing the support for strings/template strings. With support for multiple "name formats" the whole process looks like this:
- check whether string is actually translatable into other "name formats",
- iterate through all of the formats and remove class fields that are already there:
foo='foo'
,['foo']='foo'
, etc. - move field declaration from constructor to the class itself:
foo='foo'
Supporting those IMO brings lots of unespected edge cases in the points 1 & 2. It's hard to predict which string and how can be represented. Simple this.foo
can be written as this['foo']
, this["foo"]
or this[`foo`]
, but string with kebab-case like: "foo-something" can be only used as this['foo-something']
, this["foo-something"]
or this[`foo-something`]
. For every of those "name formats" there are other characters that are restricted (and I don't want even start about the escaped characters 😄 ).
It's just too much of a hassle and places that can potentially break
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I didn't mean to ask support for strings, please test
class Foo {
constructor() {
this[foo] = 'foo';
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Don't worry, I didn't understood it well then :)
Done: f9caf1b
What should we do if the assignment is unreachable? class Foo {
constructor(x) {
if (x) {return}
this.foo = 'foo';
}
} |
yield fixer.insertTextAfterRange( | ||
classBodyStartRange, | ||
`\n${indent}${node.expression.left.property.name} = ${node.expression.right.raw};`, | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The fix also changes code if the property already exists.
class Foo {
foo = 'bar';
constructor() {
this.foo = 'foo';
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
True. IMO in this case the output should be:
class Foo {
foo = 'foo';
constructor() {
}
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done: a7edb37
docs/rules/prefer-class-fields.md
Outdated
|
||
```js | ||
class Foo { | ||
constructor() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
tab-indentation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's dynamically set, so it cannot be statically defined. It should not be reported by this rule. |
Hey @fregante! Or maybe we can just agree that once the work in this PR is done you'll maybe just run your linter on it once? |
06d643a
to
3a9c5f8
Compare
Handled stopping of the static analysis on unsupported case: 3376845 |
d907827
to
77f4954
Compare
Okay, I fixed all test/linting errors connected with my rule - should I fix other as well? They seems to be connected with some default options missing in already existing rules? |
Adds
prefer-class-fields
rule which turns:into:
Fixes #314
IssueHunt Summary
Referenced issues
This pull request has been submitted to:
prefer-class-fields