Skip to content

Commit 7dd64d3

Browse files
committed
Properly narrow union types containing string and number
1 parent 4c581de commit 7dd64d3

File tree

2 files changed

+25
-9
lines changed

2 files changed

+25
-9
lines changed

src/compiler/checker.ts

Lines changed: 24 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -8461,6 +8461,28 @@ namespace ts {
84618461
return f(type) ? type : neverType;
84628462
}
84638463

8464+
function mapType(type: Type, f: (t: Type) => Type): Type {
8465+
return type.flags & TypeFlags.Union ? getUnionType(map((<UnionType>type).types, f)) : f(type);
8466+
}
8467+
8468+
function extractTypesOfKind(type: Type, kind: TypeFlags) {
8469+
return filterType(type, t => (t.flags & kind) !== 0);
8470+
}
8471+
8472+
// Return a new type in which occurrences of the string and number primitive types in
8473+
// typeWithPrimitives have been replaced with occurrences of string literals and numeric
8474+
// literals in typeWithLiterals, respectively.
8475+
function replacePrimitivesWithLiterals(typeWithPrimitives: Type, typeWithLiterals: Type) {
8476+
if (isTypeSubsetOf(stringType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, TypeFlags.StringLiteral) ||
8477+
isTypeSubsetOf(numberType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, TypeFlags.NumberLiteral)) {
8478+
return mapType(typeWithPrimitives, t =>
8479+
t.flags & TypeFlags.String ? extractTypesOfKind(typeWithLiterals, TypeFlags.String | TypeFlags.StringLiteral) :
8480+
t.flags & TypeFlags.Number ? extractTypesOfKind(typeWithLiterals, TypeFlags.Number | TypeFlags.NumberLiteral) :
8481+
t);
8482+
}
8483+
return typeWithPrimitives;
8484+
}
8485+
84648486
function isIncomplete(flowType: FlowType) {
84658487
return flowType.flags === 0;
84668488
}
@@ -8791,16 +8813,12 @@ namespace ts {
87918813
assumeTrue ? TypeFacts.EQUndefined : TypeFacts.NEUndefined;
87928814
return getTypeWithFacts(type, facts);
87938815
}
8794-
if (type.flags & TypeFlags.String && isTypeOfKind(valueType, TypeFlags.StringLiteral) ||
8795-
type.flags & TypeFlags.Number && isTypeOfKind(valueType, TypeFlags.NumberLiteral)) {
8796-
return assumeTrue? valueType : type;
8797-
}
87988816
if (type.flags & TypeFlags.NotUnionOrUnit) {
87998817
return type;
88008818
}
88018819
if (assumeTrue) {
88028820
const narrowedType = filterType(type, t => areTypesComparable(t, valueType));
8803-
return narrowedType.flags & TypeFlags.Never ? type : narrowedType;
8821+
return narrowedType.flags & TypeFlags.Never ? type : replacePrimitivesWithLiterals(narrowedType, valueType);
88048822
}
88058823
if (isUnitType(valueType)) {
88068824
const regularType = getRegularTypeOfLiteralType(valueType);
@@ -8849,9 +8867,7 @@ namespace ts {
88498867
const discriminantType = getUnionType(clauseTypes);
88508868
const caseType =
88518869
discriminantType.flags & TypeFlags.Never ? neverType :
8852-
type.flags & TypeFlags.String && isTypeOfKind(discriminantType, TypeFlags.StringLiteral) ? discriminantType :
8853-
type.flags & TypeFlags.Number && isTypeOfKind(discriminantType, TypeFlags.NumberLiteral) ? discriminantType :
8854-
filterType(type, t => isTypeComparableTo(discriminantType, t));
8870+
replacePrimitivesWithLiterals(filterType(type, t => isTypeComparableTo(discriminantType, t)), discriminantType);
88558871
if (!hasDefaultClause) {
88568872
return caseType;
88578873
}

src/compiler/types.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2644,7 +2644,7 @@ namespace ts {
26442644
// 'Narrowable' types are types where narrowing actually narrows.
26452645
// This *should* be every type other than null, undefined, void, and never
26462646
Narrowable = Any | StructuredType | TypeParameter | StringLike | NumberLike | BooleanLike | ESSymbol,
2647-
NotUnionOrUnit = Any | String | Number | ESSymbol | ObjectType,
2647+
NotUnionOrUnit = Any | ESSymbol | ObjectType,
26482648
/* @internal */
26492649
RequiresWidening = ContainsWideningType | ContainsObjectLiteral,
26502650
/* @internal */

0 commit comments

Comments
 (0)