-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
Support locally scoped type alias nodes #23188
Comments
I think it's a great idea. I've wanted to propose something like this before but I wasn't able to imagine a workable syntax. I do think the syntax is a little hard to follow but I'd be willing to accept that in exchange for the benefits you mentioned. I see myself using this all the time. |
I think the difficulties in reading it largely come from two places:
You've opened my mind to the possibilities though so if you're open to some bikeshedding, what about: type Foo<T, K extends string> with <TVal = T[K]> = TVal extends string ? {x: TVal} : never; and type MyBox<T> with <K = keyof T, TVal = T[K]> = {
host: T,
getValue(k: K): TVal,
setValue(k: K, v: TVal): TVal
}; It allows visual nesting between the top level |
@kpdonn I'd like it to be a type node and not an alias variant, so it can be used with embedded conditional types, eg type MethodReturnMap<T> = {
[K in keyof T]: type Val = T[K] in Val extends () => infer R ? R : never
} This way it can always be used to force a conditional type to distribute over unions. |
Okay I can see that, that's even more powerful than I was thinking. I do have a hard time with really wanting to read the Edit: Maybe |
I like the type MethodReturnMap<T> = {
[K in keyof T]: with(Val = T[K]) { Val extends () => infer R ? R : never }
} IMO this makes it more readable and may already look familiar due to the similarities to JS with statements (don't know if that's desirable). |
What you have there looks like a block or an object type. If we're drawing analogies to value spaces we'd want something that works like an expression. Think let expressions in Haskell. |
Am I right in thinking that semantically the proposal is much like an anonymous type abstraction that is immediately applied? Similar to how
then the type level version is:
This would be consistent with local aliases causing distribution. I think the main concern with the syntax would be users getting confused between these two: type T = number | string;
type Foo<X> = T extends number ? X : never;
type Bar<X> = type T = number | string in T extends number ? X : never; Is there a reason why |
Thinking more about my personal readability difficulties, I keep having problems with type Foo<T, K extends string> = type TVal = ( T[K] in TVal extends string ? {x: TVal} : never ) or type Foo<T, K extends string> = type TVal = ( T[K] in TVal ) extends string ? {x: TVal} : never instead of the correct type Foo<T, K extends string> = type ( TVal = T[K] ) in ( TVal extends string ? {x: TVal} : never ) Another syntax suggestionThinking about that plus @jack-williams concern about the different meanings of
Examplestype Foo<T, K extends string> = type <TVal = T[K]> TVal extends string ? {x: TVal} : never; type MyBox<T> = type <K = keyof T, TVal = T[K]> {
host: T,
getValue(k: K): TVal,
setValue(k: K, v: TVal): TVal
}; type MethodReturnMap<T> = {
[K in keyof T]: type <Val = T[K]> Val extends () => infer R ? R : never
} type Bar<X> = type <T = number | string> T extends number ? X : never; ThoughtsThat syntax resolves my readability difficulties. It doesn't seem like there would need to be any extra delimiter after the I think that syntax would also help with @jack-williams comment by making it more obvious that the type Bar<X> = type <T = number | string> T extends number ? X : never; is analogous to type DistributeHelper<X, T> = T extends number ? X : never;
type Foo<X> = DistributeHelper<X, string | number> and not analogous to type T = number | string;
type Foo<X> = T extends number ? X : never; FinallyJust want to re-emphasize that I like the proposal a lot regardless of syntax. So if you don't think my suggestion is an improvement then I'd still be very happy with the original. |
In that PR (#32525) I've suggested a functionality similar to what's being suggested here. Basically, the idea is to extend syntax of type expressions with type Entries<Obj extends object> =
Array<[ Key, Obj[Key] ]> where Key is keyof Obj; type Entries<Obj extends object> =
Array<[ Key, Value ]> where Key is keyof Obj, Value is Obj[Key]; The full syntax is this (
What do you think about this kind of syntax? |
Thanks to @jack-williams for pointing me to the correct issue. I closed #40194 in favour of this. I found myself thinking of it as a conceptual combination of Mapped Types and explicit distributive logic. If this is correct, the syntax proposal I sketched using both
The problem case I mentioned in the original issue would therefore read like...
If I followed it correctly, the OP's problem case...
...would look like...
...or limiting to the constrained, mapped type, union case
|
Related is #41470 which thinks about this in terms of an anonymous namespace so that you can have arbitrary locals. |
I'm suggesting an alternative syntax for locally scoped type alias: type Foo<T, K extends string> = <TVal = T[K]> TVal extends string ? {x: TVal} : never With two locals: type Foo<T> = <K = keyof T, TVal = T[K]> TVal extends string ? {x: TVal} : never This syntax does not introduce new keywords, and is compatible with jsx as the When using type Normal<T> = `${T}${T}`
type Distributive<T> = <K in T>`${K}${K}`
type NormalResult = Normal<'a' | 'b'> // 'aa' | 'ab' | 'ba' | 'bb'
type DistributiveResult =Distributive<'a' | 'b'> // 'aa' | 'bb' Currently, types are distributive only in conditional types , forcing users to write meaningless conditional types like |
We mentioned this at a previous design meeting while discussing conditional types but I don't think an issue was opened for it (nor can I find mention of it within some design meeting notes). We think it'd be useful for both conditional types (to make a bare type reference/parameter) and in complex types (to reduce duplication) to enable a kind of type-alias-as-a-type-node syntax. Something that allows rewriting code like this:
as
or
as
this allows elision of unnecessary information (no need to write both a constraint and identical default), and prevents usages from accidentally providing an extra type parameter when they shouldn't (because the defaulted parameter was used effectively as a const). I believe @bterlson had mentioned wanting something like this in-person, too.
As for proposed syntax, I'd say:
LocalTypeAlias
is aTypeNode
(so is valid anywhere aTypeNode
is).LocalTypeAlias
is parsed as a requiredtype
keword, then a comma separated list ofLocalTypeAliasAssignment
s (with at least one element) - the type assignment list, followed byin
and an arbitraryTypeNode
- the subject of the assignments.LocalTypeAliasAssignment
is anIdentifier
followed by=
followed by aTypeNode
. (does this need to be restricted to parse in a human-understandable way?)For semantics:
LocalTypeAlias
is a local scope around its subject type node. It binds a declaration for a type parameter for each identifier in each assignment in its type alias assignment list, constrained to the assigned value, and establishes aTypeMapper
to map those type parameters to their assigned value as well (similar to an instantiated generic type alias). (Should they be allowed to be mutually referential? Other parameter lists are, so I don't see why not.) These aliases are made to be type parameters (and not raw aliases like a nongeneric type alias declaration) so they interact favorably with conditional types - eg, they are a way to force a conditional type to iterate over a union for any type node without introducing another top-level alias.Thoughts?
The text was updated successfully, but these errors were encountered: