Make it possible to infer superclass type parameters / constructor overload from the `super` call
See original GitHub issueSearch Terms
superclass base class generic type infer contextual infer parent generic super constructor
Suggestion
Make it possible to infer superclass type parameters / constructor overload from the super
call
Use Cases
I’m working on an intermediate library with typings for a legacy codebase. That intermediate library is used in third party to use the functionality from the main codebase (called adapters). There are several supported ways to access this functionality:
const foo1 = Adapter(options);
const foo2 = new Adapter(options);
class Foo3 extends Adapter { ... }
- The constructor must call the superclass constructor with the options.
Now the problem is that the Adapter
“class” (actually an ES5-style “function” class) has some properties that only exist if specific options are passed to its constructor. I’m able to model this behavior for case 1 and 2, but not 3 - at least not without manually specifying the type.
Here’s what I have so far. The code is split into three parts: main codebase, intermediate library, 3rd party / user code. (Playground link)
// legacy code, cannot change:
declare class InternalAdapter {
// actually an ES5-style class function, can be called with and without new!
someProp1: any;
someProp2: any;
// I want to override this type
cacheObj: Record<string, any> | undefined;
}
// ========================
// My code, can change:
interface AdapterOptions {
name: string;
cache?: boolean;
}
type AdapterInstance<T extends AdapterOptions> = T extends {
cache: true;
}
? Omit<InternalAdapter, "cacheObj"> & {
cacheObj: Exclude<InternalAdapter["cacheObj"], undefined>;
}
: Omit<InternalAdapter, "cacheObj">;
interface AdapterConstructor {
new (adapterName: string): AdapterInstance<{name: string}>;
new <T extends AdapterOptions>(adapterOptions: T): AdapterInstance<T>;
(adapterName: string): AdapterInstance<{name: string}>;
<T extends AdapterOptions>(adapterOptions: T): AdapterInstance<T>;
}
declare const Adapter: AdapterConstructor;
// ========================
// User code, should be as simple as possible
const name = "foobar";
const options = { name };
const test1 = Adapter(name);
test1.cacheObj; // does not exist, expected
const test2 = new Adapter(name);
test2.cacheObj; // does not exist, expected
const test3 = Adapter(options);
test3.cacheObj; // does not exist, expected
const test4 = new Adapter(options);
test4.cacheObj; // does not exist, expected
const test5 = new Adapter({ ...options, cache: true });
test5.cacheObj; // exists, expected
// Here, the problems start:
class Test6 extends Adapter<AdapterOptions> {
constructor(options: AdapterOptions) {
super(options);
this.cacheObj; // does not exist, expected
}
}
class Test7 extends Adapter<AdapterOptions> {
constructor(options: AdapterOptions) {
super({ ...options, cache: true });
this.cacheObj; // does not exist, unexpected
}
}
class Test8 extends Adapter<AdapterOptions & {cache: true}> {
constructor(options: AdapterOptions) {
super({ ...options, cache: true });
this.cacheObj; // exists, but I have to duplicate the type
}
}
Notice how the class definitions are all awkward. In Test6
I have to duplicate the generic type AdapterOptions
or the super call will default to the string constructor.
In Test7
this is actually wrong, because the generic type overrides the conditional behavior for the cacheObj
. This can be fixed like in Test8
, but that is really ugly. IMO, TypeScript should be able to infer the type arguments from a constructor call if that type argument matches the class’ type argument
Examples
Show how this would be used and what the behavior would be - Ideally it should be like this:
class Test9 extends Adapter {
constructor(options: AdapterOptions) {
super({ ...options, cache: true });
this.cacheObj; // should exist
}
}
The super
call would be used to infer that the 2nd constructor overload is the correct one:
new <T extends AdapterOptions>(adapterOptions: T): AdapterInstance<T>;
and therefore the instance type would be AdapterInstance<AdapterOptions & {cache: true}>
.
Checklist
My suggestion meets these guidelines:
- This wouldn’t be a breaking change in existing TypeScript/JavaScript code (not sure actually. I’d guess not?)
- This wouldn’t change the runtime behavior of existing JavaScript code
- This could be implemented without emitting different JS based on the types of the expressions
- This isn’t a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, etc.)
- This feature would agree with the rest of TypeScript’s Design Goals.
Issue Analytics
- State:
- Created 4 years ago
- Reactions:23
- Comments:13 (3 by maintainers)
Top GitHub Comments
This missing feature has been a thorn in my side for some time and seems like an essential language addition. As a counter point to @tadhgmister 's counter point, the code as written should be broken.
The extended type T is more specific than number, therefor usages of T should be expected to be more specific than the default assigned type. In the example
new Box(0)
would disallow settingvalue = 1
, while the subclass is allowed to do this using the same initial value. I would argue the current behavior presents like a bug.Aside from correctness, inferring generics from super calls is essential because to the best of my knowledge there is currently no way to obtain the type of that parameter otherwise. Because super is a keyword with special behavior, if the intended behavior is for the ancestor class to receive the exact inferred type there’s not an alternate path via property access, inferred optional types or any other variety of black magic typing.
This particular issue makes it considerably more difficult to migrate a project containing hundreds of thousands of lines to Typescript where it would otherwise be a relatively painless transition. Please consider approving the proposal and I would be happy to look into submitting a pull request.
To clarify here’s a proposed use case:
+1 to this
TypeScript, applied to certain use cases, e.g. subclassing, feels very verbose and unpleasant to work with at the moment. This is one of a few things needed to combat such verbosity.