diff --git a/compiler/src/dotty/tools/dotc/typer/Namer.scala b/compiler/src/dotty/tools/dotc/typer/Namer.scala index 91a7a03abb36..58406c306a22 100644 --- a/compiler/src/dotty/tools/dotc/typer/Namer.scala +++ b/compiler/src/dotty/tools/dotc/typer/Namer.scala @@ -1196,8 +1196,22 @@ class Namer { typer: Typer => if mbr.isType then val forwarderName = checkNoConflict(alias.toTypeName, isPrivate = false, span) var target = pathType.select(sym) - if target.typeParams.nonEmpty then - target = target.etaExpand(target.typeParams) + val tparams = target.typeParams + if tparams.nonEmpty then + // like `target = target.etaExpand(target.typeParams)` + // except call `asSeenFrom` to fix class type parameter bounds + // e.g. in pos/i18569: + // `Test#F` should have `M2.A` or `Test.A` as bounds, not `M1#A`. + target = HKTypeLambda(tparams.map(_.paramName))( + tl => tparams.map { + case p: Symbol => + val info = p.info.asSeenFrom(pathType, sym.owner) + HKTypeLambda.toPInfo(tl.integrate(tparams, info)) + case p => + val info = p.paramInfo + HKTypeLambda.toPInfo(tl.integrate(tparams, info)) + }, + tl => tl.integrate(tparams, target.appliedTo(tparams.map(_.paramRef)))) newSymbol( cls, forwarderName, Exported | Final, diff --git a/tests/pos/i18569.scala b/tests/pos/i18569.scala new file mode 100644 index 000000000000..1457f6fbe5b6 --- /dev/null +++ b/tests/pos/i18569.scala @@ -0,0 +1,11 @@ +trait M1: + trait A + trait F[T <: A] + type G[T <: A] = F[T] + +object M2 extends M1 + +trait Test: + export M2.* + def y: F[A] + def z: G[A]