Skip to content

Commit 975ba5d

Browse files
committed
Add fallback logic for generating signatures for unions of array members
1 parent 218180d commit 975ba5d

9 files changed

+1217
-2508
lines changed

src/compiler/checker.ts

+30-1
Original file line numberDiff line numberDiff line change
@@ -14265,7 +14265,36 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
1426514265
* maps primitive types and type parameters are to their apparent types.
1426614266
*/
1426714267
function getSignaturesOfType(type: Type, kind: SignatureKind): readonly Signature[] {
14268-
return getSignaturesOfStructuredType(getReducedApparentType(type), kind);
14268+
const result = getSignaturesOfStructuredType(getReducedApparentType(type), kind);
14269+
if (kind === SignatureKind.Call && !length(result) && type.flags & TypeFlags.Union) {
14270+
if ((type as UnionType).arrayFallbackSignatures) {
14271+
return (type as UnionType).arrayFallbackSignatures!;
14272+
}
14273+
// If the union is all different instantiations of a member of the global array type...
14274+
let memberName: __String;
14275+
if (everyType(type, t => !!t.symbol?.parent && isArrayOrTupleSymbol(t.symbol.parent) && (!memberName ? (memberName = t.symbol.escapedName, true) : memberName === t.symbol.escapedName))) {
14276+
// Transform the type from `(A[] | B[])["member"]` to `(A | B)[]["member"]` (since we pretend array is covariant anyway)
14277+
const arrayArg = mapType(type, t => getMappedType((isReadonlyArraySymbol(t.symbol.parent) ? globalReadonlyArrayType : globalArrayType).typeParameters![0], (t as AnonymousType).mapper!));
14278+
const arrayType = createArrayType(arrayArg, someType(type, t => isReadonlyArraySymbol(t.symbol.parent)));
14279+
return (type as UnionType).arrayFallbackSignatures = getSignaturesOfType(getTypeOfPropertyOfType(arrayType, memberName!)!, kind);
14280+
}
14281+
(type as UnionType).arrayFallbackSignatures = result;
14282+
}
14283+
return result;
14284+
}
14285+
14286+
function isArrayOrTupleSymbol(symbol: Symbol | undefined) {
14287+
if (!symbol || !globalArrayType.symbol || !globalReadonlyArrayType.symbol) {
14288+
return false;
14289+
}
14290+
return !!getSymbolIfSameReference(symbol, globalArrayType.symbol) || !!getSymbolIfSameReference(symbol, globalReadonlyArrayType.symbol);
14291+
}
14292+
14293+
function isReadonlyArraySymbol(symbol: Symbol | undefined) {
14294+
if (!symbol || !globalReadonlyArrayType.symbol) {
14295+
return false;
14296+
}
14297+
return !!getSymbolIfSameReference(symbol, globalReadonlyArrayType.symbol);
1426914298
}
1427014299

1427114300
function findIndexInfo(indexInfos: readonly IndexInfo[], keyType: Type) {

src/compiler/types.ts

+2
Original file line numberDiff line numberDiff line change
@@ -6438,6 +6438,8 @@ export interface UnionType extends UnionOrIntersectionType {
64386438
keyPropertyName?: __String; // Property with unique unit type that exists in every object/intersection in union type
64396439
/** @internal */
64406440
constituentMap?: Map<TypeId, Type>; // Constituents keyed by unit type discriminants
6441+
/** @internal */
6442+
arrayFallbackSignatures?: readonly Signature[]; // Special remapped signature list for unions of arrays
64416443
}
64426444

64436445
export interface IntersectionType extends UnionOrIntersectionType {

0 commit comments

Comments
 (0)