diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractExecutableMemberWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractExecutableMemberWriter.java index a3174e3e8be..39c15a82313 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractExecutableMemberWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/AbstractExecutableMemberWriter.java @@ -133,13 +133,13 @@ public abstract class AbstractExecutableMemberWriter extends AbstractMemberWrite /** * Add the parameter for the executable member. * - * @param member the member to write parameter for. * @param param the parameter that needs to be written. + * @param paramType the type of the parameter. * @param isVarArg true if this is a link to var arg. * @param tree the content tree to which the parameter information will be added. */ - protected void addParam(ExecutableElement member, VariableElement param, TypeMirror paramType, - boolean isVarArg, Content tree) { + protected void addParam(VariableElement param, TypeMirror paramType, boolean isVarArg, + Content tree) { Content link = writer.getLink(new HtmlLinkInfo(configuration, EXECUTABLE_MEMBER_PARAM, paramType).varargs(isVarArg)); tree.add(link); @@ -249,7 +249,7 @@ public abstract class AbstractExecutableMemberWriter extends AbstractMemberWrite .add(" "); } } - addParam(member, param, paramType, + addParam(param, paramType, (paramstart == parameters.size() - 1) && member.isVarArgs(), paramTree); break; } @@ -268,7 +268,7 @@ public abstract class AbstractExecutableMemberWriter extends AbstractMemberWrite .add(" "); } } - addParam(member, parameters.get(i), instMeth.getParameterTypes().get(i), + addParam(parameters.get(i), instMeth.getParameterTypes().get(i), (i == parameters.size() - 1) && member.isVarArgs(), paramTree); } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java index 0c8182e7a84..c95b735b709 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDocletWriter.java @@ -1075,13 +1075,11 @@ public class HtmlDocletWriter { } else if (refMemName == null) { // Must be a class reference since refClass is not null and refMemName is null. if (labelContent.isEmpty()) { - if (!refClass.getTypeParameters().isEmpty() && seeText.contains("<")) { - // If this is a generic type link try to use the TypeMirror representation. - TypeMirror refType = ch.getReferencedType(see); - if (refType != null) { - return plainOrCode(isLinkPlain, getLink( - new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, refType))); - } + TypeMirror referencedType = ch.getReferencedType(see); + if (utils.isGenericType(referencedType)) { + // This is a generic type link, use the TypeMirror representation. + return plainOrCode(isLinkPlain, getLink( + new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, referencedType))); } labelContent = plainOrCode(isLinkPlain, Text.of(utils.getSimpleName(refClass))); } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlLinkFactory.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlLinkFactory.java index 805a7c3d519..ee09e99cea6 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlLinkFactory.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlLinkFactory.java @@ -107,8 +107,7 @@ public class HtmlLinkFactory extends LinkFactory { if (utils.isIncluded(typeElement)) { if (configuration.isGeneratedDoc(typeElement) && !utils.hasHiddenTag(typeElement)) { DocPath filename = getPath(classLinkInfo); - if (linkInfo.linkToSelf || - !(docPaths.forName(typeElement)).equals(m_writer.filename)) { + if (linkInfo.linkToSelf || typeElement != m_writer.getCurrentPageElement()) { link.add(m_writer.links.createLink( filename.fragment(classLinkInfo.where), label, diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java index 89635a3c585..65bcdce9181 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/CommentHelper.java @@ -192,6 +192,9 @@ public class CommentHelper { } public TypeMirror getType(ReferenceTree rtree) { + // Workaround for JDK-8269706 + if (path == null || dcTree == null || rtree == null) + return null; DocTreePath docTreePath = DocTreePath.getPath(path, dcTree, rtree); if (docTreePath != null) { DocTrees doctrees = configuration.docEnv.getDocTrees(); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/Utils.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/Utils.java index 4bcc5bf3acc..652f61caaa7 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/Utils.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/Utils.java @@ -973,6 +973,21 @@ public class Utils { return null; } + /** + * Returns true if {@code type} or any of its enclosing types has non-empty type arguments. + * @param type the type + * @return {@code true} if type arguments were found + */ + public boolean isGenericType(TypeMirror type) { + while (type instanceof DeclaredType dt) { + if (!dt.getTypeArguments().isEmpty()) { + return true; + } + type = dt.getEnclosingType(); + } + return false; + } + /** * TODO: FIXME: port to javax.lang.model * Find a class within the context of this class. Search order: qualified name, in this class diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkFactory.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkFactory.java index cb1b16a4f88..c5e67990221 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkFactory.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkFactory.java @@ -166,6 +166,16 @@ public abstract class LinkFactory { @Override public Content visitDeclared(DeclaredType type, LinkInfo linkInfo) { + TypeMirror enc = type.getEnclosingType(); + if (enc instanceof DeclaredType dt && utils.isGenericType(dt)) { + // If an enclosing type has type parameters render them as separate links as + // otherwise this information is lost. On the other hand, plain enclosing types + // are not linked separately as they are easy to reach from the nested type. + setEnclosingTypeLinkInfo(linkInfo, dt); + visitDeclared(dt, linkInfo); + link.add("."); + setEnclosingTypeLinkInfo(linkInfo, type); + } link.add(getTypeAnnotationLinks(linkInfo)); linkInfo.typeElement = utils.asTypeElement(type); link.add(getClassLink(linkInfo)); @@ -195,6 +205,12 @@ public abstract class LinkFactory { linkInfo.skipPreview = false; } + private void setEnclosingTypeLinkInfo(LinkInfo linkinfo, DeclaredType enclosing) { + linkinfo.typeElement = null; + linkinfo.label = null; + linkinfo.type = enclosing; + } + /** * Returns a link to the given class. * diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkInfo.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkInfo.java index 1907ad426d1..9094e04a087 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkInfo.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/links/LinkInfo.java @@ -27,10 +27,12 @@ package jdk.javadoc.internal.doclets.toolkit.util.links; import javax.lang.model.element.ExecutableElement; import javax.lang.model.element.TypeElement; +import javax.lang.model.type.DeclaredType; import javax.lang.model.type.TypeMirror; import jdk.javadoc.internal.doclets.toolkit.BaseConfiguration; import jdk.javadoc.internal.doclets.toolkit.Content; +import jdk.javadoc.internal.doclets.toolkit.util.Utils; /** * Encapsulates information about a link. @@ -125,7 +127,11 @@ public abstract class LinkInfo { return label; } else if (isLinkable()) { Content tlabel = newContent(); - tlabel.add(configuration.utils.getSimpleName(typeElement)); + Utils utils = configuration.utils; + tlabel.add(type instanceof DeclaredType dt && utils.isGenericType(dt.getEnclosingType()) + // If enclosing type is rendered as separate links only use own class name + ? typeElement.getSimpleName().toString() + : configuration.utils.getSimpleName(typeElement)); return tlabel; } else { Content tlabel = newContent(); diff --git a/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/TestGenericTypeLink.java b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/TestGenericTypeLink.java index cd289202c7a..fbfbb5f3978 100644 --- a/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/TestGenericTypeLink.java +++ b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/TestGenericTypeLink.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8177280 8262992 + * @bug 8177280 8262992 8259499 * @summary see and link tag syntax should allow generic types * @library ../../lib * @modules jdk.javadoc/jdk.javadoc.internal.tool @@ -118,6 +118,40 @@ public class TestGenericTypeLink extends JavadocTester { """ ); + checkOutput("pkg1/A.Inner.html", true, + """ +
A<\
+ /a>
+ A<Object,<\
+ wbr>RuntimeExcepti\
+ on>.Inner""");
}
/**
diff --git a/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/A.java b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/A.java
index e665cab02e5..bae55f24dbd 100644
--- a/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/A.java
+++ b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/A.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -60,6 +60,16 @@ public class A {
*/
public void otherMethod(Map list, double d) {}
+ /**
+ * Here's a generic link: {@link A