Add missing getReducedType call in getConditionalTypeInstantiation (#48061)

This commit is contained in:
Wesley Wigham
2022-02-28 13:41:10 -08:00
committed by GitHub
parent 41b981c998
commit 03c11c8f70
5 changed files with 161 additions and 1 deletions

View File

@@ -16854,7 +16854,7 @@ namespace ts {
// distributive conditional type T extends U ? X : Y is instantiated with A | B for T, the
// result is (A extends U ? X : Y) | (B extends U ? X : Y).
result = distributionType && checkType !== distributionType && distributionType.flags & (TypeFlags.Union | TypeFlags.Never) ?
mapTypeWithAlias(distributionType, t => getConditionalType(root, prependTypeMapping(checkType, t, newMapper)), aliasSymbol, aliasTypeArguments) :
mapTypeWithAlias(getReducedType(distributionType), t => getConditionalType(root, prependTypeMapping(checkType, t, newMapper)), aliasSymbol, aliasTypeArguments) :
getConditionalType(root, newMapper, aliasSymbol, aliasTypeArguments);
root.instantiations!.set(id, result);
}

View File

@@ -0,0 +1,25 @@
//// [objectAssignLikeNonUnionResult.ts]
interface Interface {
field: number;
}
const defaultValue: Interface = { field: 1 };
declare function assign<T, U>(target: T, source: U): T & U;
// Displayed type: Interface & { field: number }
// Underlying type: Something else...
const data1 = assign(defaultValue, Date.now() > 3 ? { field: 2 } : {});
type ExtractRawComponent<T> = T extends { __raw: infer C } ? [L1: T, L2: C] : [R1: T];
type t1 = ExtractRawComponent<typeof data1>;
// ???
type Explode<T> = T extends { x: infer A } ? [A] : 'X';
// 'X' | [unknown] -- why?
type e1 = Explode<typeof data1>;
//// [objectAssignLikeNonUnionResult.js]
var defaultValue = { field: 1 };
// Displayed type: Interface & { field: number }
// Underlying type: Something else...
var data1 = assign(defaultValue, Date.now() > 3 ? { field: 2 } : {});

View File

@@ -0,0 +1,64 @@
=== tests/cases/compiler/objectAssignLikeNonUnionResult.ts ===
interface Interface {
>Interface : Symbol(Interface, Decl(objectAssignLikeNonUnionResult.ts, 0, 0))
field: number;
>field : Symbol(Interface.field, Decl(objectAssignLikeNonUnionResult.ts, 0, 21))
}
const defaultValue: Interface = { field: 1 };
>defaultValue : Symbol(defaultValue, Decl(objectAssignLikeNonUnionResult.ts, 3, 5))
>Interface : Symbol(Interface, Decl(objectAssignLikeNonUnionResult.ts, 0, 0))
>field : Symbol(field, Decl(objectAssignLikeNonUnionResult.ts, 3, 33))
declare function assign<T, U>(target: T, source: U): T & U;
>assign : Symbol(assign, Decl(objectAssignLikeNonUnionResult.ts, 3, 45))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 5, 24))
>U : Symbol(U, Decl(objectAssignLikeNonUnionResult.ts, 5, 26))
>target : Symbol(target, Decl(objectAssignLikeNonUnionResult.ts, 5, 30))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 5, 24))
>source : Symbol(source, Decl(objectAssignLikeNonUnionResult.ts, 5, 40))
>U : Symbol(U, Decl(objectAssignLikeNonUnionResult.ts, 5, 26))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 5, 24))
>U : Symbol(U, Decl(objectAssignLikeNonUnionResult.ts, 5, 26))
// Displayed type: Interface & { field: number }
// Underlying type: Something else...
const data1 = assign(defaultValue, Date.now() > 3 ? { field: 2 } : {});
>data1 : Symbol(data1, Decl(objectAssignLikeNonUnionResult.ts, 9, 5))
>assign : Symbol(assign, Decl(objectAssignLikeNonUnionResult.ts, 3, 45))
>defaultValue : Symbol(defaultValue, Decl(objectAssignLikeNonUnionResult.ts, 3, 5))
>Date.now : Symbol(DateConstructor.now, Decl(lib.es5.d.ts, --, --))
>Date : Symbol(Date, Decl(lib.es5.d.ts, --, --), Decl(lib.es5.d.ts, --, --), Decl(lib.es5.d.ts, --, --), Decl(lib.scripthost.d.ts, --, --))
>now : Symbol(DateConstructor.now, Decl(lib.es5.d.ts, --, --))
>field : Symbol(field, Decl(objectAssignLikeNonUnionResult.ts, 9, 53))
type ExtractRawComponent<T> = T extends { __raw: infer C } ? [L1: T, L2: C] : [R1: T];
>ExtractRawComponent : Symbol(ExtractRawComponent, Decl(objectAssignLikeNonUnionResult.ts, 9, 71))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 11, 25))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 11, 25))
>__raw : Symbol(__raw, Decl(objectAssignLikeNonUnionResult.ts, 11, 41))
>C : Symbol(C, Decl(objectAssignLikeNonUnionResult.ts, 11, 54))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 11, 25))
>C : Symbol(C, Decl(objectAssignLikeNonUnionResult.ts, 11, 54))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 11, 25))
type t1 = ExtractRawComponent<typeof data1>;
>t1 : Symbol(t1, Decl(objectAssignLikeNonUnionResult.ts, 11, 86))
>ExtractRawComponent : Symbol(ExtractRawComponent, Decl(objectAssignLikeNonUnionResult.ts, 9, 71))
>data1 : Symbol(data1, Decl(objectAssignLikeNonUnionResult.ts, 9, 5))
// ???
type Explode<T> = T extends { x: infer A } ? [A] : 'X';
>Explode : Symbol(Explode, Decl(objectAssignLikeNonUnionResult.ts, 12, 44))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 15, 13))
>T : Symbol(T, Decl(objectAssignLikeNonUnionResult.ts, 15, 13))
>x : Symbol(x, Decl(objectAssignLikeNonUnionResult.ts, 15, 29))
>A : Symbol(A, Decl(objectAssignLikeNonUnionResult.ts, 15, 38))
>A : Symbol(A, Decl(objectAssignLikeNonUnionResult.ts, 15, 38))
// 'X' | [unknown] -- why?
type e1 = Explode<typeof data1>;
>e1 : Symbol(e1, Decl(objectAssignLikeNonUnionResult.ts, 15, 55))
>Explode : Symbol(Explode, Decl(objectAssignLikeNonUnionResult.ts, 12, 44))
>data1 : Symbol(data1, Decl(objectAssignLikeNonUnionResult.ts, 9, 5))

View File

@@ -0,0 +1,53 @@
=== tests/cases/compiler/objectAssignLikeNonUnionResult.ts ===
interface Interface {
field: number;
>field : number
}
const defaultValue: Interface = { field: 1 };
>defaultValue : Interface
>{ field: 1 } : { field: number; }
>field : number
>1 : 1
declare function assign<T, U>(target: T, source: U): T & U;
>assign : <T, U>(target: T, source: U) => T & U
>target : T
>source : U
// Displayed type: Interface & { field: number }
// Underlying type: Something else...
const data1 = assign(defaultValue, Date.now() > 3 ? { field: 2 } : {});
>data1 : Interface & { field: number; }
>assign(defaultValue, Date.now() > 3 ? { field: 2 } : {}) : Interface & { field: number; }
>assign : <T, U>(target: T, source: U) => T & U
>defaultValue : Interface
>Date.now() > 3 ? { field: 2 } : {} : { field: number; } | {}
>Date.now() > 3 : boolean
>Date.now() : number
>Date.now : () => number
>Date : DateConstructor
>now : () => number
>3 : 3
>{ field: 2 } : { field: number; }
>field : number
>2 : 2
>{} : {}
type ExtractRawComponent<T> = T extends { __raw: infer C } ? [L1: T, L2: C] : [R1: T];
>ExtractRawComponent : ExtractRawComponent<T>
>__raw : C
type t1 = ExtractRawComponent<typeof data1>;
>t1 : [R1: Interface & { field: number; }]
>data1 : Interface & { field: number; }
// ???
type Explode<T> = T extends { x: infer A } ? [A] : 'X';
>Explode : Explode<T>
>x : A
// 'X' | [unknown] -- why?
type e1 = Explode<typeof data1>;
>e1 : "X"
>data1 : Interface & { field: number; }

View File

@@ -0,0 +1,18 @@
interface Interface {
field: number;
}
const defaultValue: Interface = { field: 1 };
declare function assign<T, U>(target: T, source: U): T & U;
// Displayed type: Interface & { field: number }
// Underlying type: Something else...
const data1 = assign(defaultValue, Date.now() > 3 ? { field: 2 } : {});
type ExtractRawComponent<T> = T extends { __raw: infer C } ? [L1: T, L2: C] : [R1: T];
type t1 = ExtractRawComponent<typeof data1>;
// ???
type Explode<T> = T extends { x: infer A } ? [A] : 'X';
// 'X' | [unknown] -- why?
type e1 = Explode<typeof data1>;