fix: make generics with runtime props in defineComponent work (fix #11374) #13119
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
fixes #11374
relates to #12761 (comment), #7963 (comment)
Problem
In #7963 was introduced support for generics in defineComponent, but it doesn't work as everyone wants it to work. Specifically, if the
props
option passed, the generic type disappears and all the prop types become anyIn other words, even the simplest component with generics like shown below is impossible to make without losing types
Why doesn't it work? Let's look at one of the overloads of defineComponent:
Here we can see the
Props
generic type, thesetup
argument usingProps
and theoptions
argument also usingProps
I guess when we add a generic type to the
setup
function, it makes the function less "pure / straightforward" to infer theProps
type, and typescript choosesProps
from theoptions.props
instead. But because it only contains names of the props, all values are preserved asany
fromProps
extendsSolution
We need to tell typescript not to infer
Props
from theoptions.props
field, so we can use a native utility typeNoInfer
:Initially I've come up with another solution which doesn't rely on that new typescript
NoInfer
type.It works by separating
Props
into two generics — originalProps
andDeclaredProps
— and using them differently.Type
DeclaredProps extends (keyof Props)[]
can have less keys than inProps
, but not moreAnother format for passing props
Array of prop names isn't the only way to tell what props we want to use, there is also a format with objects and validations, which unfortunately I couldn't beat:
Objects are much harder than arrays, you can't describe an object with no more fields than in another object using extends as easy as an array. Ok, maybe you can, but then you'll run into another problem: it's not easy (if even possible) to check if an object with primitive types matches another object with generic types
I tried some variants, but ended up with an idea that it just doesn't worth it and there's no reason to use this format with generics. So this format works as before: no generics and
any
instead of passed types