-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
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
Enhancement: no-unused-vars option for allowing type-only declare class
expressions
#10918
Comments
declare class
expressionsdeclare class
expressions
declare class
expressionsdeclare class
expressions
My question would be "how often are you doing this in your codebase?" Are we talking every file? One in 10 files? One in a hundred? One in a thousand? Anecdotally I've very rarely seen this pattern of dynamic class creation. And quickly searching public github repos shows just 3460 cases. I just don't think that this passes the "broadly applicable" acceptance test. But I'm looking for more info to convince me otherwise. |
Personally I have 237 classes of the form: declare abstract class AnyCombatTracker extends CombatTracker<ApplicationOptions> {
constructor(arg0: never, ...args: never[]);
} This is for 647 files and a total of 463 classes that match This is of course a bit on the extreme end and I don't expect most libraries would standardize something like this. I've done this in my library to help users side-step common subclassing issues they run into where they want a bound like To focus on mixins, I have 16 mixin functions in the JS repo I've been typing that use this hoisted class trick. It makes maintaining them much easier because I can keep them in the same form as the original JS (property order) etc. I do think mixins are somewhat niche because, frankly, they're a bit of an anti-pattern and given the rough edges in TypeScript's support for mixins it can scare people away. Though for what it's worth searching for declare const Events : <T extends AnyConstructor<Base>>(base : T) => AnyConstructor<InstanceType<T> & EventsMixin>
type MixinHelperFunc = <A extends AnyConstructor, T>(required: [A], arg: T) => T extends AnyFunction<infer M> ? M : never
export const Mixin1Func = <T extends AnyConstructor<Base>>(base: T) =>
/**
* Internal class of the Mixin1
*/
class Mixin1Class extends base {
property1 = "init";
method1(arg: Mixin1Type[]): Mixin1Type[] {
return [...arg, this];
}
}; The last one is an example of a A more accurate survey of cases where you might want to hoist classes to the top level when writing I'm unfortunately not sure there is a good regex because if you want to look for mixins the best approach would be looking for a function that contains a class that extends one of the parameters. Maybe |
If you include
namespace Foo {
class C1 extends Bar { ... }
}
function Baz() {
class C2 extends Bar { ... }
return new C2();
} Doing a quick search to also include cases of There's ultimately no perfect regex to match mixins and there's no easy way to syntactically search all of GH. But we don't want the exact numbers -- a rough heuristic of scale is what I'm interested in. All in all we're sitting at a rough heuristic showing there's possibly single-digit-thousands declarations of mixins with that number maybe being in the "low teens". In the scope of TS patterns and TS code that scale is a grain of sand on a beach, really. I'm personally unconvinced that this pattern is common enough for us to build and maintain an entire option to support. cc @typescript-eslint/triage-team -- do we have any other thoughts or views? |
I'm also not convinced. Maybe this is a good case for |
Before You File a Proposal Please Confirm You Have Done The Following...
My proposal is suitable for this project
Link to the rule's documentation
https://eslint.org/docs/latest/rules/no-unused-vars
Description
While
no-unused-var
triggering on most values that are only used in types makes sense, this breaks down when doing advanced things with classes. I propose an option to allowdeclare class
expressions to be counted as used by types.I run into this frequently when writing declaration files, specifically because I write type-only subclasses to widen the class (the most frequent example is to create a bound that allows abstract classes and changed constructors). I recognize this is niche, a more universal example would be typing JS mixins or in general functions that return internally-scoped classes.
For example if you encounter a JS function like:
It is most nicely typed as:
For those who want to comply with this rule, some fake-value classes can be successfully turned into types, though it is more verbose:
Can be converted to:
This already is a mild sacrifice because it's more verbose, forces you to separate instance and static props (even if they make more intuitive sense to order them interspersed), and you lose the ability to write
typeof SomeClass
to mean the class, which to me feels more intuitive if you're used to regular classes.However there's no valid transformations when:
private
,protected
,readonly
, andinternal
number
, getstring
) and it also breaks subclassing as you can't substitute a getter/setter pair for a property.internal
,private
, orprotected
constructor. You could simply leave out the constructor inSomeClassConstructor
but this gives worse diagnostics.While I'm quite comfortable with classes and translating them to regular interfaces, I also imagine most people will find the syntax for generic classes annoying. It also becomes more awkward when the base class expression is complex like mixins, especially when generic passthrough is involved:
You can transform
Mixed
to:This code is obviously ugly and it doesn't generalize well to more complex mixins, especially where the generic parameters and parameters differ significantly.
Fail
Pass
Additional Info
No response
The text was updated successfully, but these errors were encountered: