@@ -296,8 +296,8 @@ namespace ts {
296
296
createPromiseType,
297
297
createArrayType,
298
298
getBooleanType: () => booleanType,
299
- getFalseType: () => falseType,
300
- getTrueType: () => trueType,
299
+ getFalseType: (fresh? ) => fresh ? falseType : regularFalseType ,
300
+ getTrueType: (fresh? ) => fresh ? trueType : regularTrueType ,
301
301
getVoidType: () => voidType,
302
302
getUndefinedType: () => undefinedType,
303
303
getNullType: () => nullType,
@@ -405,9 +405,22 @@ namespace ts {
405
405
const nullWideningType = strictNullChecks ? nullType : createIntrinsicType(TypeFlags.Null | TypeFlags.ContainsWideningType, "null");
406
406
const stringType = createIntrinsicType(TypeFlags.String, "string");
407
407
const numberType = createIntrinsicType(TypeFlags.Number, "number");
408
- const falseType = createIntrinsicType(TypeFlags.BooleanLiteral, "false");
409
- const trueType = createIntrinsicType(TypeFlags.BooleanLiteral, "true");
410
- const booleanType = createBooleanType([falseType, trueType]);
408
+ const falseType = createIntrinsicType(TypeFlags.BooleanLiteral, "false") as FreshableIntrinsicType;
409
+ const regularFalseType = createIntrinsicType(TypeFlags.BooleanLiteral, "false") as FreshableIntrinsicType;
410
+ const trueType = createIntrinsicType(TypeFlags.BooleanLiteral, "true") as FreshableIntrinsicType;
411
+ const regularTrueType = createIntrinsicType(TypeFlags.BooleanLiteral, "true") as FreshableIntrinsicType;
412
+ falseType.flags |= TypeFlags.FreshLiteral;
413
+ trueType.flags |= TypeFlags.FreshLiteral;
414
+ trueType.regularType = regularTrueType;
415
+ regularTrueType.freshType = trueType;
416
+ falseType.regularType = regularFalseType;
417
+ regularFalseType.freshType = falseType;
418
+ const booleanType = createBooleanType([regularFalseType, regularTrueType]);
419
+ // Also mark all combinations of fresh/regular booleans as "Boolean" so they print as `boolean` instead of `true | false`
420
+ // (The union is cached, so simply doing the marking here is sufficient)
421
+ createBooleanType([regularFalseType, trueType]);
422
+ createBooleanType([falseType, regularTrueType]);
423
+ createBooleanType([falseType, trueType]);
411
424
const esSymbolType = createIntrinsicType(TypeFlags.ESSymbol, "symbol");
412
425
const voidType = createIntrinsicType(TypeFlags.Void, "void");
413
426
const neverType = createIntrinsicType(TypeFlags.Never, "never");
@@ -4170,7 +4183,7 @@ namespace ts {
4170
4183
const baseType = t.flags & TypeFlags.BooleanLiteral ? booleanType : getBaseTypeOfEnumLiteralType(<LiteralType>t);
4171
4184
if (baseType.flags & TypeFlags.Union) {
4172
4185
const count = (<UnionType>baseType).types.length;
4173
- if (i + count <= types.length && types[i + count - 1] === ( <UnionType>baseType).types[count - 1]) {
4186
+ if (i + count <= types.length && getRegularTypeOfLiteralType( types[i + count - 1]) === getRegularTypeOfLiteralType(( <UnionType>baseType).types[count - 1]) ) {
4174
4187
result.push(baseType);
4175
4188
i += count - 1;
4176
4189
continue;
@@ -6157,7 +6170,7 @@ namespace ts {
6157
6170
if (type.flags & TypeFlags.UniqueESSymbol) {
6158
6171
return `__@${type.symbol.escapedName}@${getSymbolId(type.symbol)}` as __String;
6159
6172
}
6160
- if (type.flags & TypeFlags.StringOrNumberLiteral ) {
6173
+ if (type.flags & ( TypeFlags.StringLiteral | TypeFlags.NumberLiteral) ) {
6161
6174
return escapeLeadingUnderscores("" + (<LiteralType>type).value);
6162
6175
}
6163
6176
return Debug.fail();
@@ -8810,7 +8823,7 @@ namespace ts {
8810
8823
t.flags & TypeFlags.StringLiteral && includes & TypeFlags.String ||
8811
8824
t.flags & TypeFlags.NumberLiteral && includes & TypeFlags.Number ||
8812
8825
t.flags & TypeFlags.UniqueESSymbol && includes & TypeFlags.ESSymbol ||
8813
- t.flags & TypeFlags.StringOrNumberLiteral && t.flags & TypeFlags.FreshLiteral && containsType(types, (<LiteralType>t).regularType);
8826
+ t.flags & TypeFlags.Literal && t.flags & TypeFlags.FreshLiteral && containsType(types, (<LiteralType>t).regularType);
8814
8827
if (remove) {
8815
8828
orderedRemoveItemAt(types, i);
8816
8829
}
@@ -9810,8 +9823,8 @@ namespace ts {
9810
9823
}
9811
9824
9812
9825
function getFreshTypeOfLiteralType(type: Type): Type {
9813
- if (type.flags & TypeFlags.StringOrNumberLiteral && !(type.flags & TypeFlags.FreshLiteral)) {
9814
- if (!(<LiteralType>type).freshType) {
9826
+ if (type.flags & TypeFlags.Literal && !(type.flags & TypeFlags.FreshLiteral)) {
9827
+ if (!(<LiteralType>type).freshType) { // NOTE: Safe because all freshable intrinsics always have fresh types already
9815
9828
const freshType = createLiteralType(type.flags | TypeFlags.FreshLiteral, (<LiteralType>type).value, (<LiteralType>type).symbol);
9816
9829
freshType.regularType = <LiteralType>type;
9817
9830
(<LiteralType>type).freshType = freshType;
@@ -9822,7 +9835,7 @@ namespace ts {
9822
9835
}
9823
9836
9824
9837
function getRegularTypeOfLiteralType(type: Type): Type {
9825
- return type.flags & TypeFlags.StringOrNumberLiteral && type.flags & TypeFlags.FreshLiteral ? (<LiteralType>type).regularType :
9838
+ return type.flags & TypeFlags.Literal && type.flags & TypeFlags.FreshLiteral ? (<LiteralType>type).regularType :
9826
9839
type.flags & TypeFlags.Union ? getUnionType(sameMap((<UnionType>type).types, getRegularTypeOfLiteralType)) :
9827
9840
type;
9828
9841
}
@@ -11035,11 +11048,11 @@ namespace ts {
11035
11048
}
11036
11049
11037
11050
function isTypeRelatedTo(source: Type, target: Type, relation: Map<RelationComparisonResult>) {
11038
- if (source.flags & TypeFlags.StringOrNumberLiteral && source.flags & TypeFlags.FreshLiteral) {
11039
- source = (<LiteralType >source).regularType;
11051
+ if (source.flags & TypeFlags.Literal && source.flags & TypeFlags.FreshLiteral) {
11052
+ source = (<FreshableType >source).regularType;
11040
11053
}
11041
- if (target.flags & TypeFlags.StringOrNumberLiteral && target.flags & TypeFlags.FreshLiteral) {
11042
- target = (<LiteralType >target).regularType;
11054
+ if (target.flags & TypeFlags.Literal && target.flags & TypeFlags.FreshLiteral) {
11055
+ target = (<FreshableType >target).regularType;
11043
11056
}
11044
11057
if (source === target ||
11045
11058
relation === comparableRelation && !(target.flags & TypeFlags.Never) && isSimpleTypeRelatedTo(target, source, relation) ||
@@ -11194,11 +11207,11 @@ namespace ts {
11194
11207
* * Ternary.False if they are not related.
11195
11208
*/
11196
11209
function isRelatedTo(source: Type, target: Type, reportErrors = false, headMessage?: DiagnosticMessage): Ternary {
11197
- if (source.flags & TypeFlags.StringOrNumberLiteral && source.flags & TypeFlags.FreshLiteral) {
11198
- source = (<LiteralType >source).regularType;
11210
+ if (source.flags & TypeFlags.Literal && source.flags & TypeFlags.FreshLiteral) {
11211
+ source = (<FreshableType >source).regularType;
11199
11212
}
11200
- if (target.flags & TypeFlags.StringOrNumberLiteral && target.flags & TypeFlags.FreshLiteral) {
11201
- target = (<LiteralType >target).regularType;
11213
+ if (target.flags & TypeFlags.Literal && target.flags & TypeFlags.FreshLiteral) {
11214
+ target = (<FreshableType >target).regularType;
11202
11215
}
11203
11216
if (source.flags & TypeFlags.Substitution) {
11204
11217
source = relation === definitelyAssignableRelation ? (<SubstitutionType>source).typeVariable : (<SubstitutionType>source).substitute;
@@ -12766,7 +12779,7 @@ namespace ts {
12766
12779
return type.flags & TypeFlags.EnumLiteral && type.flags & TypeFlags.FreshLiteral ? getBaseTypeOfEnumLiteralType(<LiteralType>type) :
12767
12780
type.flags & TypeFlags.StringLiteral && type.flags & TypeFlags.FreshLiteral ? stringType :
12768
12781
type.flags & TypeFlags.NumberLiteral && type.flags & TypeFlags.FreshLiteral ? numberType :
12769
- type.flags & TypeFlags.BooleanLiteral ? booleanType :
12782
+ type.flags & TypeFlags.BooleanLiteral && type.flags & TypeFlags.FreshLiteral ? booleanType :
12770
12783
type.flags & TypeFlags.Union ? getUnionType(sameMap((<UnionType>type).types, getWidenedLiteralType)) :
12771
12784
type;
12772
12785
}
@@ -12820,7 +12833,7 @@ namespace ts {
12820
12833
return type.flags & TypeFlags.Union ? getFalsyFlagsOfTypes((<UnionType>type).types) :
12821
12834
type.flags & TypeFlags.StringLiteral ? (<LiteralType>type).value === "" ? TypeFlags.StringLiteral : 0 :
12822
12835
type.flags & TypeFlags.NumberLiteral ? (<LiteralType>type).value === 0 ? TypeFlags.NumberLiteral : 0 :
12823
- type.flags & TypeFlags.BooleanLiteral ? type === falseType ? TypeFlags.BooleanLiteral : 0 :
12836
+ type.flags & TypeFlags.BooleanLiteral ? ( type === falseType || type === regularFalseType) ? TypeFlags.BooleanLiteral : 0 :
12824
12837
type.flags & TypeFlags.PossiblyFalsy;
12825
12838
}
12826
12839
@@ -12837,7 +12850,8 @@ namespace ts {
12837
12850
function getDefinitelyFalsyPartOfType(type: Type): Type {
12838
12851
return type.flags & TypeFlags.String ? emptyStringType :
12839
12852
type.flags & TypeFlags.Number ? zeroType :
12840
- type.flags & TypeFlags.Boolean || type === falseType ? falseType :
12853
+ type.flags & TypeFlags.Boolean || type === regularFalseType ? regularFalseType :
12854
+ type === falseType ? falseType :
12841
12855
type.flags & (TypeFlags.Void | TypeFlags.Undefined | TypeFlags.Null) ||
12842
12856
type.flags & TypeFlags.StringLiteral && (<LiteralType>type).value === "" ||
12843
12857
type.flags & TypeFlags.NumberLiteral && (<LiteralType>type).value === 0 ? type :
@@ -14092,7 +14106,10 @@ namespace ts {
14092
14106
if (assignedType.flags & TypeFlags.Never) {
14093
14107
return assignedType;
14094
14108
}
14095
- const reducedType = filterType(declaredType, t => typeMaybeAssignableTo(assignedType, t));
14109
+ let reducedType = filterType(declaredType, t => typeMaybeAssignableTo(assignedType, t));
14110
+ if (assignedType.flags & (TypeFlags.FreshLiteral | TypeFlags.Literal)) {
14111
+ reducedType = mapType(reducedType, getFreshTypeOfLiteralType); // Ensure that if the assignment is a fresh type, that we narrow to fresh types
14112
+ }
14096
14113
// Our crude heuristic produces an invalid result in some cases: see GH#26130.
14097
14114
// For now, when that happens, we give up and don't narrow at all. (This also
14098
14115
// means we'll never narrow for erroneous assignments where the assigned type
@@ -14145,8 +14162,8 @@ namespace ts {
14145
14162
}
14146
14163
if (flags & TypeFlags.BooleanLike) {
14147
14164
return strictNullChecks ?
14148
- type === falseType ? TypeFacts.FalseStrictFacts : TypeFacts.TrueStrictFacts :
14149
- type === falseType ? TypeFacts.FalseFacts : TypeFacts.TrueFacts;
14165
+ ( type === falseType || type === regularFalseType) ? TypeFacts.FalseStrictFacts : TypeFacts.TrueStrictFacts :
14166
+ ( type === falseType || type === regularFalseType) ? TypeFacts.FalseFacts : TypeFacts.TrueFacts;
14150
14167
}
14151
14168
if (flags & TypeFlags.Object) {
14152
14169
return isFunctionObjectType(<ObjectType>type) ?
@@ -28351,19 +28368,20 @@ namespace ts {
28351
28368
function isLiteralConstDeclaration(node: VariableDeclaration | PropertyDeclaration | PropertySignature | ParameterDeclaration): boolean {
28352
28369
if (isDeclarationReadonly(node) || isVariableDeclaration(node) && isVarConst(node)) {
28353
28370
const type = getTypeOfSymbol(getSymbolOfNode(node));
28354
- return !!(type.flags & TypeFlags.StringOrNumberLiteral && type.flags & TypeFlags.FreshLiteral);
28371
+ return !!(type.flags & TypeFlags.Literal && type.flags & TypeFlags.FreshLiteral);
28355
28372
}
28356
28373
return false;
28357
28374
}
28358
28375
28359
- function literalTypeToNode(type: LiteralType, enclosing: Node): Expression {
28360
- const enumResult = type.flags & TypeFlags.EnumLiteral && nodeBuilder.symbolToExpression(type.symbol, SymbolFlags.Value, enclosing);
28361
- return enumResult || createLiteral(type.value);
28376
+ function literalTypeToNode(type: FreshableType, enclosing: Node): Expression {
28377
+ const enumResult = type.flags & TypeFlags.EnumLiteral ? nodeBuilder.symbolToExpression(type.symbol, SymbolFlags.Value, enclosing)
28378
+ : type === trueType ? createTrue() : type === falseType && createFalse();
28379
+ return enumResult || createLiteral((type as LiteralType).value);
28362
28380
}
28363
28381
28364
28382
function createLiteralConstValue(node: VariableDeclaration | PropertyDeclaration | PropertySignature | ParameterDeclaration) {
28365
28383
const type = getTypeOfSymbol(getSymbolOfNode(node));
28366
- return literalTypeToNode(<LiteralType >type, node);
28384
+ return literalTypeToNode(<FreshableType >type, node);
28367
28385
}
28368
28386
28369
28387
function createResolver(): EmitResolver {
@@ -29735,7 +29753,7 @@ namespace ts {
29735
29753
29736
29754
function checkAmbientInitializer(node: VariableDeclaration | PropertyDeclaration | PropertySignature) {
29737
29755
if (node.initializer) {
29738
- const isInvalidInitializer = !(isStringOrNumberLiteralExpression(node.initializer) || isSimpleLiteralEnumReference(node.initializer));
29756
+ const isInvalidInitializer = !(isStringOrNumberLiteralExpression(node.initializer) || isSimpleLiteralEnumReference(node.initializer) || node.initializer.kind === SyntaxKind.TrueKeyword || node.initializer.kind === SyntaxKind.FalseKeyword );
29739
29757
const isConstOrReadonly = isDeclarationReadonly(node) || isVariableDeclaration(node) && isVarConst(node);
29740
29758
if (isConstOrReadonly && !node.type) {
29741
29759
if (isInvalidInitializer) {
0 commit comments