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 c30ad115ec6..09e243c0835 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 @@ -386,7 +386,7 @@ public class HtmlDocletWriter { * @param context the enclosing context * @return a TagletWriter */ - public TagletWriter getTagletWriterInstance(TagletWriterImpl.Context context) { + public TagletWriterImpl getTagletWriterInstance(TagletWriterImpl.Context context) { return new TagletWriterImpl(this, context); } @@ -971,316 +971,6 @@ public class HtmlDocletWriter { return label; } - public Content seeTagToContent(Element element, DocTree see, TagletWriterImpl.Context context) { - Kind kind = see.getKind(); - CommentHelper ch = utils.getCommentHelper(element); - String tagName = ch.getTagName(see); - - String refText; - List label; - switch (kind) { - case LINK, LINK_PLAIN -> { - // {@link[plain] reference label...} - LinkTree lt = (LinkTree) see; - var linkRef = lt.getReference(); - if (linkRef == null) { - messages.warning(ch.getDocTreePath(see),"doclet.link.no_reference"); - return invalidTagOutput(resources.getText("doclet.tag.invalid_input", lt.toString()), - Optional.empty()); - } - refText = linkRef.toString(); - label = lt.getLabel(); - } - - case SEE -> { - List ref = ((SeeTree) see).getReference(); - assert !ref.isEmpty(); - DocTree ref0 = ref.get(0); - switch (ref0.getKind()) { - case TEXT, START_ELEMENT -> { - // @see "Reference" - // @see ... - return commentTagsToContent(element, ref, false, false); - } - case REFERENCE -> { - // @see reference label... - refText = ref0.toString(); - label = ref.subList(1, ref.size()); - } - case ERRONEOUS -> { - return invalidTagOutput(resources.getText("doclet.tag.invalid_input", - ref0.toString()), - Optional.empty()); - } - default -> - throw new IllegalStateException(ref0.getKind().toString()); - } - } - - default -> - throw new IllegalStateException(kind.toString()); - } - - boolean isLinkPlain = kind == LINK_PLAIN; - Content labelContent = plainOrCode(isLinkPlain, - commentTagsToContent(element, label, context)); - - // The signature from the @see tag. We will output this text when a label is not specified. - Content text = plainOrCode(isLinkPlain, - Text.of(Objects.requireNonNullElse(ch.getReferencedSignature(see), ""))); - - TypeElement refClass = ch.getReferencedClass(see); - Element refMem = ch.getReferencedMember(see); - String refMemName = ch.getReferencedMemberName(see); - - if (refMemName == null && refMem != null) { - refMemName = refMem.toString(); - } - if (refClass == null) { - ModuleElement refModule = ch.getReferencedModule(see); - if (refModule != null && utils.isIncluded(refModule)) { - return getModuleLink(refModule, labelContent.isEmpty() ? text : labelContent); - } - //@see is not referencing an included class - PackageElement refPackage = ch.getReferencedPackage(see); - if (refPackage != null && utils.isIncluded(refPackage)) { - //@see is referencing an included package - if (labelContent.isEmpty()) - labelContent = plainOrCode(isLinkPlain, - Text.of(refPackage.getQualifiedName())); - return getPackageLink(refPackage, labelContent); - } else { - // @see is not referencing an included class, module or package. Check for cross links. - String refModuleName = ch.getReferencedModuleName(see); - DocLink elementCrossLink = (refPackage != null) ? getCrossPackageLink(refPackage) : - (configuration.extern.isModule(refModuleName)) - ? getCrossModuleLink(utils.elementUtils.getModuleElement(refModuleName)) - : null; - if (elementCrossLink != null) { - // Element cross link found - return links.createExternalLink(elementCrossLink, - (labelContent.isEmpty() ? text : labelContent)); - } else { - // No cross link found so print warning - if (!configuration.isDocLintReferenceGroupEnabled()) { - messages.warning(ch.getDocTreePath(see), - "doclet.see.class_or_package_not_found", - "@" + tagName, - refText); - } - return invalidTagOutput(resources.getText("doclet.tag.invalid", tagName), - Optional.of(labelContent.isEmpty() ? text: labelContent)); - } - } - } else if (refMemName == null) { - // Must be a class reference since refClass is not null and refMemName is null. - if (labelContent.isEmpty()) { - 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))); - } - return getLink(new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, refClass) - .label(labelContent)); - } else if (refMem == null) { - // Must be a member reference since refClass is not null and refMemName is not null. - // However, refMem is null, so this referenced member does not exist. - return (labelContent.isEmpty() ? text: labelContent); - } else { - // Must be a member reference since refClass is not null and refMemName is not null. - // refMem is not null, so this @see tag must be referencing a valid member. - TypeElement containing = utils.getEnclosingTypeElement(refMem); - - // Find the enclosing type where the method is actually visible - // in the inheritance hierarchy. - ExecutableElement overriddenMethod = null; - if (refMem.getKind() == ElementKind.METHOD) { - VisibleMemberTable vmt = configuration.getVisibleMemberTable(containing); - overriddenMethod = vmt.getOverriddenMethod((ExecutableElement)refMem); - - if (overriddenMethod != null) - containing = utils.getEnclosingTypeElement(overriddenMethod); - } - if (refText.trim().startsWith("#") && - ! (utils.isPublic(containing) || utils.isLinkable(containing))) { - // Since the link is relative and the holder is not even being - // documented, this must be an inherited link. Redirect it. - // The current class either overrides the referenced member or - // inherits it automatically. - if (this instanceof ClassWriterImpl writer) { - containing = writer.getTypeElement(); - } else if (!utils.isPublic(containing)) { - messages.warning( - ch.getDocTreePath(see), "doclet.see.class_or_package_not_accessible", - tagName, utils.getFullyQualifiedName(containing)); - } else { - if (!configuration.isDocLintReferenceGroupEnabled()) { - messages.warning( - ch.getDocTreePath(see), "doclet.see.class_or_package_not_found", - tagName, refText); - } - } - } - if (configuration.currentTypeElement != containing) { - refMemName = (utils.isConstructor(refMem)) - ? refMemName - : utils.getSimpleName(containing) + "." + refMemName; - } - if (utils.isExecutableElement(refMem)) { - if (refMemName.indexOf('(') < 0) { - refMemName += utils.makeSignature((ExecutableElement) refMem, null, true); - } - if (overriddenMethod != null) { - // The method to actually link. - refMem = overriddenMethod; - } - } - - return getDocLink(HtmlLinkInfo.Kind.SEE_TAG, containing, - refMem, (labelContent.isEmpty() - ? plainOrCode(isLinkPlain, Text.of(refMemName)) - : labelContent), null, false); - } - } - - // TODO: this method and seeTagToContent share much of the code; consider factoring common pieces out - public Content linkToContent(Element referrer, Element target, String targetSignature, String text) { - CommentHelper ch = utils.getCommentHelper(referrer); - - boolean isLinkPlain = false; // TODO: for now - Content labelContent = plainOrCode(isLinkPlain, Text.of(text)); - - TypeElement refClass = ch.getReferencedClass(target); - Element refMem = ch.getReferencedMember(target); - String refMemName = ch.getReferencedMemberName(targetSignature); - - if (refMemName == null && refMem != null) { - refMemName = refMem.toString(); - } - if (refClass == null) { - ModuleElement refModule = ch.getReferencedModule(target); - if (refModule != null && utils.isIncluded(refModule)) { - return getModuleLink(refModule, labelContent); - } - //@see is not referencing an included class - PackageElement refPackage = ch.getReferencedPackage(target); - if (refPackage != null && utils.isIncluded(refPackage)) { - //@see is referencing an included package - if (labelContent.isEmpty()) - labelContent = plainOrCode(isLinkPlain, - Text.of(refPackage.getQualifiedName())); - return getPackageLink(refPackage, labelContent); - } else { - // @see is not referencing an included class, module or package. Check for cross links. - String refModuleName = ch.getReferencedModuleName(targetSignature); - DocLink elementCrossLink = (refPackage != null) ? getCrossPackageLink(refPackage) : - (configuration.extern.isModule(refModuleName)) - ? getCrossModuleLink(utils.elementUtils.getModuleElement(refModuleName)) - : null; - if (elementCrossLink != null) { - // Element cross link found - return links.createExternalLink(elementCrossLink, labelContent); - } else { - // No cross link found so print warning -// TODO: -// messages.warning(ch.getDocTreePath(see), -// "doclet.see.class_or_package_not_found", -// "@" + tagName, -// seeText); - return labelContent; - } - } - } 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() && targetSignature.contains("<")) { - // If this is a generic type link try to use the TypeMirror representation. - -// TODO: -// TypeMirror refType = ch.getReferencedType(target); - TypeMirror refType = target.asType(); - - if (refType != null) { - return plainOrCode(isLinkPlain, getLink( - new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, refType))); - } - } - labelContent = plainOrCode(isLinkPlain, Text.of(utils.getSimpleName(refClass))); - } - return getLink(new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, refClass) - .label(labelContent)); - } else if (refMem == null) { - // Must be a member reference since refClass is not null and refMemName is not null. - // However, refMem is null, so this referenced member does not exist. - return labelContent; - } else { - // Must be a member reference since refClass is not null and refMemName is not null. - // refMem is not null, so this @see tag must be referencing a valid member. - TypeElement containing = utils.getEnclosingTypeElement(refMem); - - // Find the enclosing type where the method is actually visible - // in the inheritance hierarchy. - ExecutableElement overriddenMethod = null; - if (refMem.getKind() == ElementKind.METHOD) { - VisibleMemberTable vmt = configuration.getVisibleMemberTable(containing); - overriddenMethod = vmt.getOverriddenMethod((ExecutableElement)refMem); - - if (overriddenMethod != null) - containing = utils.getEnclosingTypeElement(overriddenMethod); - } - if (targetSignature.trim().startsWith("#") && - ! (utils.isPublic(containing) || utils.isLinkable(containing))) { - // Since the link is relative and the holder is not even being - // documented, this must be an inherited link. Redirect it. - // The current class either overrides the referenced member or - // inherits it automatically. - if (this instanceof ClassWriterImpl writer) { - containing = writer.getTypeElement(); - } else if (!utils.isPublic(containing)) { -// TODO: -// messages.warning( -// ch.getDocTreePath(see), "doclet.see.class_or_package_not_accessible", -// tagName, utils.getFullyQualifiedName(containing)); - } else { -// TODO: -// messages.warning( -// ch.getDocTreePath(see), "doclet.see.class_or_package_not_found", -// tagName, seeText); - } - } - if (configuration.currentTypeElement != containing) { - refMemName = (utils.isConstructor(refMem)) - ? refMemName - : utils.getSimpleName(containing) + "." + refMemName; - } - if (utils.isExecutableElement(refMem)) { - if (refMemName.indexOf('(') < 0) { - refMemName += utils.makeSignature((ExecutableElement) refMem, null, true); - } - if (overriddenMethod != null) { - // The method to actually link. - refMem = overriddenMethod; - } - } - - return getDocLink(HtmlLinkInfo.Kind.SEE_TAG, containing, - refMem, (labelContent.isEmpty() - ? plainOrCode(isLinkPlain, Text.of(text)) - : labelContent), null, false); - } - } - - private String removeTrailingSlash(String s) { - return s.endsWith("/") ? s.substring(0, s.length() -1) : s; - } - - private Content plainOrCode(boolean plain, Content body) { - return (plain || body.isEmpty()) ? body : HtmlTree.CODE(body); - } - /** * Add the inline comment. * @@ -1660,8 +1350,8 @@ public class HtmlDocletWriter { } content.add(label); } else { - Content c = seeTagToContent(element, node, context.within(node)); - content.add(c); + TagletWriterImpl t = getTagletWriterInstance(context.within(node)); + content.add(t.linkTagOutput(element, node)); } return false; } @@ -1674,12 +1364,6 @@ public class HtmlDocletWriter { return false; } - @Override - public Boolean visitSee(SeeTree node, Content content) { - content.add(seeTagToContent(element, node, context)); - return false; - } - @Override public Boolean visitStartElement(StartElementTree node, Content content) { Content attrs = new ContentBuilder(); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/TagletWriterImpl.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/TagletWriterImpl.java index 7caf038f4d9..1637d8fbef4 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/TagletWriterImpl.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/TagletWriterImpl.java @@ -29,8 +29,10 @@ import java.util.ArrayList; import java.util.EnumSet; import java.util.HashSet; import java.util.List; +import java.util.Objects; import java.util.Optional; import java.util.Set; +import java.util.function.BiConsumer; import java.util.function.Predicate; import javax.lang.model.element.Element; @@ -46,6 +48,7 @@ import javax.lang.model.util.SimpleElementVisitor14; import com.sun.source.doctree.DeprecatedTree; import com.sun.source.doctree.DocTree; import com.sun.source.doctree.IndexTree; +import com.sun.source.doctree.LinkTree; import com.sun.source.doctree.LiteralTree; import com.sun.source.doctree.ParamTree; import com.sun.source.doctree.ReturnTree; @@ -66,6 +69,7 @@ import jdk.javadoc.internal.doclets.formats.html.markup.Text; import jdk.javadoc.internal.doclets.toolkit.BaseConfiguration; import jdk.javadoc.internal.doclets.toolkit.Content; import jdk.javadoc.internal.doclets.toolkit.DocletElement; +import jdk.javadoc.internal.doclets.toolkit.Messages; import jdk.javadoc.internal.doclets.toolkit.Resources; import jdk.javadoc.internal.doclets.toolkit.builders.SerializedFormBuilder; import jdk.javadoc.internal.doclets.toolkit.taglets.ParamTaglet; @@ -79,6 +83,9 @@ import jdk.javadoc.internal.doclets.toolkit.util.DocPaths; import jdk.javadoc.internal.doclets.toolkit.util.IndexItem; import jdk.javadoc.internal.doclets.toolkit.util.Utils; import jdk.javadoc.internal.doclets.toolkit.util.Utils.PreviewFlagProvider; +import jdk.javadoc.internal.doclets.toolkit.util.VisibleMemberTable; + +import static com.sun.source.doctree.DocTree.Kind.LINK_PLAIN; /** * The taglet writer that writes HTML. @@ -143,6 +150,9 @@ public class TagletWriterImpl extends TagletWriter { private final HtmlOptions options; private final Utils utils; private final Resources resources; + + private final Messages messages; + private final Contents contents; private final Context context; @@ -186,6 +196,7 @@ public class TagletWriterImpl extends TagletWriter { configuration = htmlWriter.configuration; options = configuration.getOptions(); utils = configuration.utils; + messages = configuration.messages; resources = configuration.getDocResources(); contents = configuration.getContents(); } @@ -270,6 +281,32 @@ public class TagletWriterImpl extends TagletWriter { return result; } + @Override + public Content linkTagOutput(Element element, LinkTree tag) { + CommentHelper ch = utils.getCommentHelper(element); + + var linkRef = tag.getReference(); + if (linkRef == null) { + messages.warning(ch.getDocTreePath(tag), "doclet.link.no_reference"); + return invalidTagOutput(resources.getText("doclet.tag.invalid_input", tag.toString()), + Optional.empty()); + } + + DocTree.Kind kind = tag.getKind(); + String tagName = ch.getTagName(tag); + String refSignature = ch.getReferencedSignature(linkRef); + + return linkSeeReferenceOutput(element, + tag, + refSignature, + ch.getReferencedElement(tag), + tagName, + (kind == LINK_PLAIN), + htmlWriter.commentTagsToContent(element, tag.getLabel(), context), + (key, args) -> messages.warning(ch.getDocTreePath(tag), key, args) + ); + } + @Override protected Content literalTagOutput(Element element, LiteralTree tag) { return Text.of(utils.normalizeNewlines(tag.getBody().getBody())); @@ -315,8 +352,9 @@ public class TagletWriterImpl extends TagletWriter { @Override public Content seeTagOutput(Element holder, List seeTags) { List links = new ArrayList<>(); - for (DocTree dt : seeTags) { - links.add(htmlWriter.seeTagToContent(holder, dt, context.within(dt))); + for (SeeTree dt : seeTags) { + TagletWriterImpl t = new TagletWriterImpl(htmlWriter, context.within(dt)); + links.add(t.seeTagOutput(holder, dt)); } if (utils.isVariableElement(holder) && ((VariableElement)holder).getConstantValue() != null && htmlWriter instanceof ClassWriterImpl writer) { @@ -363,6 +401,203 @@ public class TagletWriterImpl extends TagletWriter { return s.length() > SEE_TAG_MAX_INLINE_LENGTH || s.contains(","); } + /** + * {@return the output for a single {@code @see} tag} + * + * @param element the element that has the documentation comment containing this tag + * @param seeTag the tag + */ + private Content seeTagOutput(Element element, SeeTree seeTag) { + List ref = seeTag.getReference(); + assert !ref.isEmpty(); + DocTree ref0 = ref.get(0); + switch (ref0.getKind()) { + case TEXT, START_ELEMENT -> { + // @see "Reference" + // @see ... + return htmlWriter.commentTagsToContent(element, ref, false, false); + } + + case REFERENCE -> { + // @see reference label... + CommentHelper ch = utils.getCommentHelper(element); + String tagName = ch.getTagName(seeTag); + String refSignature = ch.getReferencedSignature(ref0); + List label = ref.subList(1, ref.size()); + + return linkSeeReferenceOutput(element, + seeTag, + refSignature, + ch.getReferencedElement(seeTag), + tagName, + false, + htmlWriter.commentTagsToContent(element, label, context), + (key, args) -> messages.warning(ch.getDocTreePath(seeTag), key, args) + ); + } + + case ERRONEOUS -> { + return invalidTagOutput(resources.getText("doclet.tag.invalid_input", + ref0.toString()), + Optional.empty()); + } + + default -> throw new IllegalStateException(ref0.getKind().toString()); + } + + } + + /** + * Worker method to generate a link from the information in different kinds of tags, + * such as {@code {@link ...}} tags, {@code @see ...} tags and the {@code link} markup tag + * in a {@code {@snippet ...}} tag. + * + * @param holder the element that has the documentation comment containing the information + * @param refTree the tree node containing the information, or {@code null} if not available + * @param refSignature the normalized signature of the target of the reference + * @param ref the target of the reference + * @param tagName the name of the tag in the source, to be used in diagnostics + * @param isLinkPlain {@code true} if the link should be presented in "plain" font, + * or {@code false} for "code" font + * @param label the label for the link, + * or an empty item to use a default label derived from the signature + * @param reportWarning a function to report warnings about issues found in the reference + * + * @return the output containing the generated link, or content indicating an error + */ + private Content linkSeeReferenceOutput(Element holder, + DocTree refTree, + String refSignature, + Element ref, + String tagName, + boolean isLinkPlain, + Content label, + BiConsumer reportWarning) { + Content labelContent = plainOrCode(isLinkPlain, label); + + // The signature from the @see tag. We will output this text when a label is not specified. + Content text = plainOrCode(isLinkPlain, + Text.of(Objects.requireNonNullElse(refSignature, ""))); + + CommentHelper ch = utils.getCommentHelper(holder); + TypeElement refClass = ch.getReferencedClass(ref); + Element refMem = ch.getReferencedMember(ref); + String refMemName = ch.getReferencedMemberName(refSignature); + + if (refMemName == null && refMem != null) { + refMemName = refMem.toString(); + } + if (refClass == null) { + ModuleElement refModule = ch.getReferencedModule(ref); + if (refModule != null && utils.isIncluded(refModule)) { + return htmlWriter.getModuleLink(refModule, labelContent.isEmpty() ? text : labelContent); + } + //@see is not referencing an included class + PackageElement refPackage = ch.getReferencedPackage(ref); + if (refPackage != null && utils.isIncluded(refPackage)) { + //@see is referencing an included package + if (labelContent.isEmpty()) + labelContent = plainOrCode(isLinkPlain, + Text.of(refPackage.getQualifiedName())); + return htmlWriter.getPackageLink(refPackage, labelContent); + } else { + // @see is not referencing an included class, module or package. Check for cross links. + String refModuleName = ch.getReferencedModuleName(refSignature); + DocLink elementCrossLink = (refPackage != null) ? htmlWriter.getCrossPackageLink(refPackage) : + (configuration.extern.isModule(refModuleName)) + ? htmlWriter.getCrossModuleLink(utils.elementUtils.getModuleElement(refModuleName)) + : null; + if (elementCrossLink != null) { + // Element cross link found + return htmlWriter.links.createExternalLink(elementCrossLink, + (labelContent.isEmpty() ? text : labelContent)); + } else { + // No cross link found so print warning + if (!configuration.isDocLintReferenceGroupEnabled()) { + reportWarning.accept( + "doclet.see.class_or_package_not_found", + new Object[] { "@" + tagName, refSignature}); + } + return htmlWriter.invalidTagOutput(resources.getText("doclet.tag.invalid", tagName), + Optional.of(labelContent.isEmpty() ? text: labelContent)); + } + } + } else if (refMemName == null) { + // Must be a class reference since refClass is not null and refMemName is null. + if (labelContent.isEmpty() && refTree != null) { + TypeMirror referencedType = ch.getReferencedType(refTree); + if (utils.isGenericType(referencedType)) { + // This is a generic type link, use the TypeMirror representation. + return plainOrCode(isLinkPlain, htmlWriter.getLink( + new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, referencedType))); + } + labelContent = plainOrCode(isLinkPlain, Text.of(utils.getSimpleName(refClass))); + } + return htmlWriter.getLink(new HtmlLinkInfo(configuration, HtmlLinkInfo.Kind.DEFAULT, refClass) + .label(labelContent)); + } else if (refMem == null) { + // Must be a member reference since refClass is not null and refMemName is not null. + // However, refMem is null, so this referenced member does not exist. + return (labelContent.isEmpty() ? text: labelContent); + } else { + // Must be a member reference since refClass is not null and refMemName is not null. + // refMem is not null, so this @see tag must be referencing a valid member. + TypeElement containing = utils.getEnclosingTypeElement(refMem); + + // Find the enclosing type where the method is actually visible + // in the inheritance hierarchy. + ExecutableElement overriddenMethod = null; + if (refMem.getKind() == ElementKind.METHOD) { + VisibleMemberTable vmt = configuration.getVisibleMemberTable(containing); + overriddenMethod = vmt.getOverriddenMethod((ExecutableElement)refMem); + + if (overriddenMethod != null) + containing = utils.getEnclosingTypeElement(overriddenMethod); + } + if (refSignature.trim().startsWith("#") && + ! (utils.isPublic(containing) || utils.isLinkable(containing))) { + // Since the link is relative and the holder is not even being + // documented, this must be an inherited link. Redirect it. + // The current class either overrides the referenced member or + // inherits it automatically. + if (htmlWriter instanceof ClassWriterImpl writer) { + containing = writer.getTypeElement(); + } else if (!utils.isPublic(containing)) { + reportWarning.accept("doclet.see.class_or_package_not_accessible", + new Object[] { tagName, utils.getFullyQualifiedName(containing)}); + } else { + if (!configuration.isDocLintReferenceGroupEnabled()) { + reportWarning.accept("doclet.see.class_or_package_not_found", + new Object[] { tagName, refSignature }); + } + } + } + if (configuration.currentTypeElement != containing) { + refMemName = (utils.isConstructor(refMem)) + ? refMemName + : utils.getSimpleName(containing) + "." + refMemName; + } + if (utils.isExecutableElement(refMem)) { + if (refMemName.indexOf('(') < 0) { + refMemName += utils.makeSignature((ExecutableElement) refMem, null, true); + } + if (overriddenMethod != null) { + // The method to actually link. + refMem = overriddenMethod; + } + } + + return htmlWriter.getDocLink(HtmlLinkInfo.Kind.SEE_TAG, containing, + refMem, (labelContent.isEmpty() + ? plainOrCode(isLinkPlain, Text.of(refMemName)) + : labelContent), null, false); + } + } + + private Content plainOrCode(boolean plain, Content body) { + return (plain || body.isEmpty()) ? body : HtmlTree.CODE(body); + } + @Override public Content simpleBlockTagOutput(Element element, List simpleTags, String header) { CommentHelper ch = utils.getCommentHelper(element); @@ -434,7 +669,14 @@ public class TagletWriterImpl extends TagletWriter { //disable preview tagging inside the snippets: PreviewFlagProvider prevPreviewProvider = utils.setPreviewFlagProvider(el -> false); try { - c = new ContentBuilder(htmlWriter.linkToContent(element, e, t, sequence.toString())); + c = linkSeeReferenceOutput(element, + null, + t, + e, + "link", + false, // TODO: for now + Text.of(sequence.toString()), + (key, args) -> { /* TODO: report diagnostic */ }); } finally { utils.setPreviewFlagProvider(prevPreviewProvider); } diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/TagletWriter.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/TagletWriter.java index 5d60e83cbe6..fd8b45c4df6 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/TagletWriter.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/taglets/TagletWriter.java @@ -36,6 +36,7 @@ import javax.lang.model.type.TypeMirror; import com.sun.source.doctree.DocTree; import com.sun.source.doctree.IndexTree; +import com.sun.source.doctree.LinkTree; import com.sun.source.doctree.LiteralTree; import com.sun.source.doctree.ParamTree; import com.sun.source.doctree.ReturnTree; @@ -107,6 +108,16 @@ public abstract class TagletWriter { */ protected abstract Content deprecatedTagOutput(Element element); + /** + * Returns the output for a {@code {@link ...}} or {@code {@linkplain ...}} tag. + * + * @param element The element that owns the doc comment + * @param tag the tag + * + * @return the output + */ + protected abstract Content linkTagOutput(Element element, LinkTree tag); + /** * Returns the output for a {@code {@literal ...}} tag. * 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 d8b00bba459..860f4611813 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 @@ -156,8 +156,8 @@ public class CommentHelper { public TypeMirror getType(ReferenceTree rtree) { DocTreePath docTreePath = getDocTreePath(rtree); if (docTreePath != null) { - DocTrees doctrees = configuration.docEnv.getDocTrees(); - return doctrees.getType(docTreePath); + DocTrees docTrees = configuration.docEnv.getDocTrees(); + return docTrees.getType(docTreePath); } return null; } @@ -213,11 +213,6 @@ public class CommentHelper { return (utils.isExecutableElement(e) || utils.isVariableElement(e)) ? e : null; } - public String getReferencedMemberName(DocTree dtree) { - String s = getReferencedSignature(dtree); - return getReferencedMemberName(s); - } - public String getReferencedMemberName(String signature) { if (signature == null) { return null; @@ -259,7 +254,7 @@ public class CommentHelper { return getFirstSentenceTrees(getBody(dtree)); } - private Element getReferencedElement(DocTree dtree) { + public Element getReferencedElement(DocTree dtree) { return new ReferenceDocTreeVisitor() { @Override public Element visitReference(ReferenceTree node, Void p) { @@ -286,7 +281,10 @@ public class CommentHelper { return null; } - public String getReferencedSignature(DocTree dtree) { + /** + * {@return the normalized signature from a {@code ReferenceTree}} + */ + public String getReferencedSignature(DocTree dtree) { return new ReferenceDocTreeVisitor() { @Override public String visitReference(ReferenceTree node, Void p) { diff --git a/test/langtools/jdk/javadoc/doclet/testNewLanguageFeatures/pkg/TypeParameters.java b/test/langtools/jdk/javadoc/doclet/testNewLanguageFeatures/pkg/TypeParameters.java index db688ff682d..390c88d56ca 100644 --- a/test/langtools/jdk/javadoc/doclet/testNewLanguageFeatures/pkg/TypeParameters.java +++ b/test/langtools/jdk/javadoc/doclet/testNewLanguageFeatures/pkg/TypeParameters.java @@ -66,7 +66,7 @@ public class TypeParameters implements SubInterface { * * @param This is the first type parameter. */ - public void methodThatHasTypeParmaters(A... a) {} + public void methodThatHasTypeParameters(A... a) {} /** * This method returns a TypeParameter array and takes in a TypeParameter array