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, + """ +
+
See Also:
+
+ +
+
"""); + + checkOutput("pkg1/C.html", true, + """ + Description copied from class: A<\ + /a> +
Here's a generic link: 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.Inner} + */ + public void overriddenMethod() {} + + /** + * @see A.Inner + * @see A.Inner, A.SomeException> + */ + class Inner {} } diff --git a/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/C.java b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/C.java new file mode 100644 index 00000000000..400eaa0970b --- /dev/null +++ b/test/langtools/jdk/javadoc/doclet/testGenericTypeLink/pkg1/C.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 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 + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package pkg1; + +import java.util.Map; + +public class C extends A { + + @Override + public void overriddenMethod() {} + +} diff --git a/test/langtools/jdk/javadoc/doclet/testMethodSignature/TestMethodSignature.java b/test/langtools/jdk/javadoc/doclet/testMethodSignature/TestMethodSignature.java index 77acf2b168a..afb2e93a01f 100644 --- a/test/langtools/jdk/javadoc/doclet/testMethodSignature/TestMethodSignature.java +++ b/test/langtools/jdk/javadoc/doclet/testMethodSignature/TestMethodSignature.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8214126 8241470 + * @bug 8214126 8241470 8259499 * @summary Method signatures not formatted correctly in browser * @library ../../lib/ * @modules jdk.javadoc/jdk.javadoc.internal.tool @@ -123,7 +123,32 @@ public class TestMethodSignature extends JavadocTester { throws java.lang.IllegalArgumentException, java.lang.IllegalStateException
-
Generic method with eight type args and annotation.
"""); +
Generic method with eight type args and annotation.
""", + """ +
public C.Generic&l\ + t;java.lang.Integer>.Inne\ + r nestedGeneric1(C.Generic<\ + java.lang.Integer>.Inner<\ + /a> i, + C.Generic<C>.Inner j)
""", + """ +
public C.Generic&l\ + t;C.F0<C>>.Inner.Foo nestedGeneric2(C.Generic<java.lang\ + .Integer>.Inner.Foo f, + C.Generic<C.F0<C\ + >>.Inner.Foo g)
"""); } } diff --git a/test/langtools/jdk/javadoc/doclet/testMethodSignature/pkg/C.java b/test/langtools/jdk/javadoc/doclet/testMethodSignature/pkg/C.java index 604299e2e16..3fe053ba3f8 100644 --- a/test/langtools/jdk/javadoc/doclet/testMethodSignature/pkg/C.java +++ b/test/langtools/jdk/javadoc/doclet/testMethodSignature/pkg/C.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 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 @@ -156,4 +156,33 @@ public class C { F0 t7, F0 t8) throws IllegalArgumentException, IllegalStateException { return null; } + + /** + * Inner classes with type arguments in enclosing classes. + * + * @param i param i + * @param j param j + * @return return value + */ + public Generic.Inner nestedGeneric1(Generic.Inner i, Generic.Inner j) { return i; } + + /** + * Inner classes with type arguments in enclosing classes. + * + * @param f param f + * @param g param g + * @return return value + */ + public Generic>.Inner.Foo nestedGeneric2(Generic.Inner.Foo f, Generic>.Inner.Foo g) { return g; } + + /** + * Generic class with multiple inner classes. + * @param type parameter + */ + public static class Generic { + public class Inner { + T data; + public class Foo {} + } + } }