@@ -6598,8 +6598,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
6598
6598
}
6599
6599
if (
6600
6600
context.flags & NodeBuilderFlags.GenerateNamesForShadowedTypeParams &&
6601
- type.flags & TypeFlags.TypeParameter &&
6602
- !isTypeSymbolAccessible(type.symbol, context.enclosingDeclaration)
6601
+ type.flags & TypeFlags.TypeParameter
6603
6602
) {
6604
6603
const name = typeParameterToName(type, context);
6605
6604
context.approximateLength += idText(name).length;
@@ -7464,7 +7463,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
7464
7463
&& signature.declaration
7465
7464
&& signature.declaration !== context.enclosingDeclaration
7466
7465
&& !isInJSFile(signature.declaration)
7467
- && some(expandedParams)
7466
+ && ( some(expandedParams) || some(signature.typeParameters) )
7468
7467
) {
7469
7468
// As a performance optimization, reuse the same fake scope within this chain.
7470
7469
// This is especially needed when we are working on an excessively deep type;
@@ -7482,32 +7481,64 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
7482
7481
// Note that we only check the most immediate enclosingDeclaration; the only place we
7483
7482
// could potentially add another fake scope into the chain is right here, so we don't
7484
7483
// traverse all ancestors.
7485
- const existingFakeScope = getNodeLinks(context.enclosingDeclaration).fakeScopeForSignatureDeclaration ? context.enclosingDeclaration : undefined;
7486
- Debug.assertOptionalNode(existingFakeScope, isBlock);
7484
+ pushFakeScope(
7485
+ "params",
7486
+ add => {
7487
+ for (const param of expandedParams) {
7488
+ add(param.escapedName, param);
7489
+ }
7490
+ },
7491
+ );
7487
7492
7488
- const locals = existingFakeScope?.locals ?? createSymbolTable();
7493
+ if (context.flags & NodeBuilderFlags.GenerateNamesForShadowedTypeParams) {
7494
+ // TODO(jakebailey): should this instead be done before walking type parameters?
7495
+ pushFakeScope(
7496
+ "typeParams",
7497
+ add => {
7498
+ for (const typeParam of signature.typeParameters ?? emptyArray) {
7499
+ const typeParamName = typeParameterToName(typeParam, context).escapedText;
7500
+ add(typeParamName, typeParam.symbol);
7501
+ }
7502
+ },
7503
+ );
7504
+ }
7489
7505
7490
- let newLocals: __String[] | undefined;
7491
- for (const param of expandedParams) {
7492
- if (!locals.has(param.escapedName)) {
7493
- newLocals = append(newLocals, param.escapedName);
7494
- locals.set(param.escapedName, param);
7506
+ function pushFakeScope(kind: "params" | "typeParams", addAll: (addSymbol: (name: __String, symbol: Symbol) => void) => void) {
7507
+ // We only ever need to look two declarations upward.
7508
+ Debug.assert(context.enclosingDeclaration);
7509
+ let existingFakeScope: Node | undefined;
7510
+ if (getNodeLinks(context.enclosingDeclaration).fakeScopeForSignatureDeclaration === kind) {
7511
+ existingFakeScope = context.enclosingDeclaration;
7495
7512
}
7496
- }
7513
+ else if (context.enclosingDeclaration.parent && getNodeLinks(context.enclosingDeclaration.parent).fakeScopeForSignatureDeclaration === kind) {
7514
+ existingFakeScope = context.enclosingDeclaration.parent;
7515
+ }
7516
+ Debug.assertOptionalNode(existingFakeScope, isBlock);
7497
7517
7498
- if (newLocals) {
7499
- function removeNewLocals() {
7518
+ const locals = existingFakeScope?.locals ?? createSymbolTable();
7519
+ let newLocals: __String[] | undefined;
7520
+ addAll((name, symbol) => {
7521
+ if (!locals.has(name)) {
7522
+ newLocals = append(newLocals, name);
7523
+ locals.set(name, symbol);
7524
+ }
7525
+ });
7526
+ if (!newLocals) return;
7527
+
7528
+ const oldCleanup = cleanup;
7529
+ function undo() {
7500
7530
forEach(newLocals, s => locals.delete(s));
7531
+ oldCleanup?.();
7501
7532
}
7502
7533
7503
7534
if (existingFakeScope) {
7504
- cleanup = removeNewLocals ;
7535
+ cleanup = undo ;
7505
7536
}
7506
7537
else {
7507
7538
// Use a Block for this; the type of the node doesn't matter so long as it
7508
7539
// has locals, and this is cheaper/easier than using a function-ish Node.
7509
7540
const fakeScope = parseNodeFactory.createBlock(emptyArray);
7510
- getNodeLinks(fakeScope).fakeScopeForSignatureDeclaration = true ;
7541
+ getNodeLinks(fakeScope).fakeScopeForSignatureDeclaration = kind ;
7511
7542
fakeScope.locals = locals;
7512
7543
7513
7544
const saveEnclosingDeclaration = context.enclosingDeclaration;
@@ -7516,7 +7547,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
7516
7547
7517
7548
cleanup = () => {
7518
7549
context.enclosingDeclaration = saveEnclosingDeclaration;
7519
- removeNewLocals ();
7550
+ undo ();
7520
7551
};
7521
7552
}
7522
7553
}
@@ -8062,13 +8093,10 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
8062
8093
}
8063
8094
}
8064
8095
8065
- function typeParameterShadowsNameInScope (escapedName: __String, context: NodeBuilderContext, type: TypeParameter) {
8096
+ function typeParameterShadowsOtherTypeParameterInScope (escapedName: __String, context: NodeBuilderContext, type: TypeParameter) {
8066
8097
const result = resolveName(context.enclosingDeclaration, escapedName, SymbolFlags.Type, /*nameNotFoundMessage*/ undefined, escapedName, /*isUse*/ false);
8067
- if (result) {
8068
- if (result.flags & SymbolFlags.TypeParameter && result === type.symbol) {
8069
- return false;
8070
- }
8071
- return true;
8098
+ if (result && result.flags & SymbolFlags.TypeParameter) {
8099
+ return result !== type.symbol;
8072
8100
}
8073
8101
return false;
8074
8102
}
@@ -8088,7 +8116,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
8088
8116
const rawtext = result.escapedText as string;
8089
8117
let i = context.typeParameterNamesByTextNextNameCount?.get(rawtext) || 0;
8090
8118
let text = rawtext;
8091
- while (context.typeParameterNamesByText?.has(text) || typeParameterShadowsNameInScope (text as __String, context, type)) {
8119
+ while (context.typeParameterNamesByText?.has(text) || typeParameterShadowsOtherTypeParameterInScope (text as __String, context, type)) {
8092
8120
i++;
8093
8121
text = `${rawtext}_${i}`;
8094
8122
}
@@ -8101,7 +8129,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
8101
8129
// `i` we've used thus far, to save work later
8102
8130
(context.typeParameterNamesByTextNextNameCount ||= new Map()).set(rawtext, i);
8103
8131
(context.typeParameterNames ||= new Map()).set(getTypeId(type), result);
8104
- (context.typeParameterNamesByText ||= new Set()).add(rawtext );
8132
+ (context.typeParameterNamesByText ||= new Set()).add(text );
8105
8133
}
8106
8134
return result;
8107
8135
}
@@ -8280,7 +8308,10 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
8280
8308
}
8281
8309
8282
8310
function getEnclosingDeclarationIgnoringFakeScope(enclosingDeclaration: Node) {
8283
- return getNodeLinks(enclosingDeclaration).fakeScopeForSignatureDeclaration ? enclosingDeclaration.parent : enclosingDeclaration;
8311
+ while (getNodeLinks(enclosingDeclaration).fakeScopeForSignatureDeclaration) {
8312
+ enclosingDeclaration = enclosingDeclaration.parent;
8313
+ }
8314
+ return enclosingDeclaration;
8284
8315
}
8285
8316
8286
8317
/**
@@ -8360,7 +8391,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
8360
8391
}
8361
8392
if (isIdentifier(node)) {
8362
8393
const type = getDeclaredTypeOfSymbol(sym);
8363
- const name = sym.flags & SymbolFlags.TypeParameter && !isTypeSymbolAccessible(type.symbol, context.enclosingDeclaration) ? typeParameterToName(type, context) : factory.cloneNode(node);
8394
+ const name = sym.flags & SymbolFlags.TypeParameter ? typeParameterToName(type, context) : factory.cloneNode(node);
8364
8395
name.symbol = sym; // for quickinfo, which uses identifier symbol information
8365
8396
return { introducesError, node: setEmitFlags(setOriginalNode(name, node), EmitFlags.NoAsciiEscaping) };
8366
8397
}
0 commit comments