/*
 * Copyright (c) 2003, 2020, 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.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * 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 jdk.javadoc.internal.doclets.formats.html;

import java.util.List;

import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.ModuleElement;
import javax.lang.model.element.PackageElement;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.SimpleElementVisitor14;

import com.sun.source.doctree.DocTree;
import com.sun.source.doctree.IndexTree;
import com.sun.source.doctree.ParamTree;
import com.sun.source.doctree.SystemPropertyTree;
import jdk.javadoc.internal.doclets.formats.html.SearchIndexItem.Category;
import jdk.javadoc.internal.doclets.formats.html.markup.ContentBuilder;
import jdk.javadoc.internal.doclets.formats.html.markup.HtmlStyle;
import jdk.javadoc.internal.doclets.formats.html.markup.HtmlTree;
import jdk.javadoc.internal.doclets.formats.html.markup.RawHtml;
import jdk.javadoc.internal.doclets.formats.html.markup.StringContent;
import jdk.javadoc.internal.doclets.toolkit.BaseConfiguration;
import jdk.javadoc.internal.doclets.toolkit.Content;
import jdk.javadoc.internal.doclets.toolkit.DocFileElement;
import jdk.javadoc.internal.doclets.toolkit.DocletElement;
import jdk.javadoc.internal.doclets.toolkit.Resources;
import jdk.javadoc.internal.doclets.toolkit.builders.SerializedFormBuilder;
import jdk.javadoc.internal.doclets.toolkit.taglets.ParamTaglet;
import jdk.javadoc.internal.doclets.toolkit.taglets.TagletWriter;
import jdk.javadoc.internal.doclets.toolkit.util.CommentHelper;
import jdk.javadoc.internal.doclets.toolkit.util.DocLink;
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
import jdk.javadoc.internal.doclets.toolkit.util.DocPaths;
import jdk.javadoc.internal.doclets.toolkit.util.DocletConstants;
import jdk.javadoc.internal.doclets.toolkit.util.Utils;

The taglet writer that writes HTML.

This is NOT part of any supported API. If you write code that depends on this, you do so at your own risk. This code and its internal interfaces are subject to change or deletion without notice.

/** * The taglet writer that writes HTML. * * <p><b>This is NOT part of any supported API. * If you write code that depends on this, you do so at your own risk. * This code and its internal interfaces are subject to change or * deletion without notice.</b> */
public class TagletWriterImpl extends TagletWriter { private final HtmlDocletWriter htmlWriter; private final HtmlConfiguration configuration; private final HtmlOptions options; private final Utils utils; private final boolean inSummary; private final Resources resources; private final Contents contents; public TagletWriterImpl(HtmlDocletWriter htmlWriter, boolean isFirstSentence) { this(htmlWriter, isFirstSentence, false); } public TagletWriterImpl(HtmlDocletWriter htmlWriter, boolean isFirstSentence, boolean inSummary) { super(isFirstSentence); this.htmlWriter = htmlWriter; this.inSummary = inSummary; configuration = htmlWriter.configuration; options = configuration.getOptions(); utils = configuration.utils; resources = configuration.getDocResources(); contents = configuration.getContents(); } @Override public Content getOutputInstance() { return new ContentBuilder(); } @Override protected Content codeTagOutput(Element element, DocTree tag) { CommentHelper ch = utils.getCommentHelper(element); StringContent content = new StringContent(utils.normalizeNewlines(ch.getText(tag))); Content result = HtmlTree.CODE(content); return result; } @Override protected Content indexTagOutput(Element element, DocTree tag) { CommentHelper ch = utils.getCommentHelper(element); IndexTree itt = (IndexTree)tag; String tagText = ch.getText(itt.getSearchTerm()); if (tagText.charAt(0) == '"' && tagText.charAt(tagText.length() - 1) == '"') { tagText = tagText.substring(1, tagText.length() - 1) .replaceAll("\\s+", " "); } String desc = ch.getText(itt.getDescription()); return createAnchorAndSearchIndex(element, tagText, desc, false); } @Override public Content getDocRootOutput() { String path; if (htmlWriter.pathToRoot.isEmpty()) path = "."; else path = htmlWriter.pathToRoot.getPath(); return new StringContent(path); } @Override public Content deprecatedTagOutput(Element element) { ContentBuilder result = new ContentBuilder(); CommentHelper ch = utils.getCommentHelper(element); List<? extends DocTree> deprs = utils.getBlockTags(element, DocTree.Kind.DEPRECATED); if (utils.isTypeElement(element)) { if (utils.isDeprecated(element)) { result.add(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, htmlWriter.getDeprecatedPhrase(element))); if (!deprs.isEmpty()) { List<? extends DocTree> commentTags = ch.getDescription(deprs.get(0)); if (!commentTags.isEmpty()) { result.add(commentTagsToOutput(null, element, commentTags, false)); } } } } else { if (utils.isDeprecated(element)) { result.add(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, htmlWriter.getDeprecatedPhrase(element))); if (!deprs.isEmpty()) { List<? extends DocTree> bodyTags = ch.getBody(deprs.get(0)); Content body = commentTagsToOutput(null, element, bodyTags, false); if (!body.isEmpty()) result.add(HtmlTree.DIV(HtmlStyle.deprecationComment, body)); } } else { Element ee = utils.getEnclosingTypeElement(element); if (utils.isDeprecated(ee)) { result.add(HtmlTree.SPAN(HtmlStyle.deprecatedLabel, htmlWriter.getDeprecatedPhrase(ee))); } } } return result; } @Override protected Content literalTagOutput(Element element, DocTree tag) { CommentHelper ch = utils.getCommentHelper(element); Content result = new StringContent(utils.normalizeNewlines(ch.getText(tag))); return result; } @Override public Content getParamHeader(ParamTaglet.ParamKind kind) { Content header; switch (kind) { case PARAMETER: header = contents.parameters ; break; case TYPE_PARAMETER: header = contents.typeParameters ; break; case RECORD_COMPONENT: header = contents.recordComponents ; break; default: throw new IllegalArgumentException(kind.toString()); } return HtmlTree.DT(header); } @Override @SuppressWarnings("preview") public Content paramTagOutput(Element element, DocTree paramTag, String paramName) { ContentBuilder body = new ContentBuilder(); CommentHelper ch = utils.getCommentHelper(element); // define id attributes for state components so that generated descriptions may refer to them boolean defineID = (element.getKind() == ElementKind.RECORD) && (paramTag instanceof ParamTree) && !((ParamTree) paramTag).isTypeParameter(); Content nameTree = new StringContent(paramName); body.add(HtmlTree.CODE(defineID ? HtmlTree.SPAN_ID("param-" + paramName, nameTree) : nameTree)); body.add(" - "); List<? extends DocTree> description = ch.getDescription(paramTag); body.add(htmlWriter.commentTagsToContent(paramTag, element, description, false, inSummary)); return HtmlTree.DD(body); } @Override public Content propertyTagOutput(Element element, DocTree tag, String prefix) { Content body = new ContentBuilder(); CommentHelper ch = utils.getCommentHelper(element); body.add(new RawHtml(prefix)); body.add(" "); body.add(HtmlTree.CODE(new RawHtml(ch.getText(tag)))); body.add("."); Content result = HtmlTree.P(body); return result; } @Override public Content returnTagOutput(Element element, DocTree returnTag) { CommentHelper ch = utils.getCommentHelper(element); return new ContentBuilder( HtmlTree.DT(contents.returns), HtmlTree.DD(htmlWriter.commentTagsToContent( returnTag, element, ch.getDescription(returnTag), false, inSummary))); } @Override public Content seeTagOutput(Element holder, List<? extends DocTree> seeTags) { ContentBuilder body = new ContentBuilder(); for (DocTree dt : seeTags) { appendSeparatorIfNotEmpty(body); body.add(htmlWriter.seeTagToContent(holder, dt)); } if (utils.isVariableElement(holder) && ((VariableElement)holder).getConstantValue() != null && htmlWriter instanceof ClassWriterImpl) { //Automatically add link to constant values page for constant fields. appendSeparatorIfNotEmpty(body); DocPath constantsPath = htmlWriter.pathToRoot.resolve(DocPaths.CONSTANT_VALUES); String whichConstant = ((ClassWriterImpl) htmlWriter).getTypeElement().getQualifiedName() + "." + utils.getSimpleName(holder); DocLink link = constantsPath.fragment(whichConstant); body.add(htmlWriter.links.createLink(link, new StringContent(resources.getText("doclet.Constants_Summary")))); } if (utils.isClass(holder) && utils.isSerializable((TypeElement)holder)) { //Automatically add link to serialized form page for serializable classes. if (SerializedFormBuilder.serialInclude(utils, holder) && SerializedFormBuilder.serialInclude(utils, utils.containingPackage(holder))) { appendSeparatorIfNotEmpty(body); DocPath serialPath = htmlWriter.pathToRoot.resolve(DocPaths.SERIALIZED_FORM); DocLink link = serialPath.fragment(utils.getFullyQualifiedName(holder)); body.add(htmlWriter.links.createLink(link, new StringContent(resources.getText("doclet.Serialized_Form")))); } } if (body.isEmpty()) return body; return new ContentBuilder( HtmlTree.DT(contents.seeAlso), HtmlTree.DD(body)); } private void appendSeparatorIfNotEmpty(ContentBuilder body) { if (!body.isEmpty()) { body.add(", "); body.add(DocletConstants.NL); } } @Override public Content simpleTagOutput(Element element, List<? extends DocTree> simpleTags, String header) { CommentHelper ch = utils.getCommentHelper(element); ContentBuilder body = new ContentBuilder(); boolean many = false; for (DocTree simpleTag : simpleTags) { if (many) { body.add(", "); } List<? extends DocTree> bodyTags = ch.getBody(simpleTag); body.add(htmlWriter.commentTagsToContent(simpleTag, element, bodyTags, false, inSummary)); many = true; } return new ContentBuilder( HtmlTree.DT(new RawHtml(header)), HtmlTree.DD(body)); } @Override public Content simpleTagOutput(Element element, DocTree simpleTag, String header) { CommentHelper ch = utils.getCommentHelper(element); List<? extends DocTree> description = ch.getDescription(simpleTag); Content body = htmlWriter.commentTagsToContent(simpleTag, element, description, false, inSummary); return new ContentBuilder( HtmlTree.DT(new RawHtml(header)), HtmlTree.DD(body)); } @Override protected Content systemPropertyTagOutput(Element element, DocTree tag) { SystemPropertyTree itt = (SystemPropertyTree) tag; String tagText = itt.getPropertyName().toString(); return HtmlTree.CODE(createAnchorAndSearchIndex(element, tagText, resources.getText("doclet.System_Property"), true)); } @Override public Content getThrowsHeader() { return HtmlTree.DT(contents.throws_); } @Override public Content throwsTagOutput(Element element, DocTree throwsTag, TypeMirror substituteType) { ContentBuilder body = new ContentBuilder(); CommentHelper ch = utils.getCommentHelper(element); Element exception = ch.getException(throwsTag); Content excName; if (substituteType != null) { excName = htmlWriter.getLink(new LinkInfoImpl(configuration, LinkInfoImpl.Kind.MEMBER, substituteType)); } else if (exception == null) { excName = new RawHtml(ch.getExceptionName(throwsTag).toString()); } else if (exception.asType() == null) { excName = new RawHtml(utils.getFullyQualifiedName(exception)); } else { LinkInfoImpl link = new LinkInfoImpl(configuration, LinkInfoImpl.Kind.MEMBER, exception.asType()); link.excludeTypeBounds = true; excName = htmlWriter.getLink(link); } body.add(HtmlTree.CODE(excName)); List<? extends DocTree> description = ch.getDescription(throwsTag); Content desc = htmlWriter.commentTagsToContent(throwsTag, element, description, false, inSummary); if (desc != null && !desc.isEmpty()) { body.add(" - "); body.add(desc); } HtmlTree result = HtmlTree.DD(body); return result; } @Override public Content throwsTagOutput(TypeMirror throwsType) { HtmlTree result = HtmlTree.DD(HtmlTree.CODE(htmlWriter.getLink( new LinkInfoImpl(configuration, LinkInfoImpl.Kind.MEMBER, throwsType)))); return result; } @Override public Content valueTagOutput(VariableElement field, String constantVal, boolean includeLink) { return includeLink ? htmlWriter.getDocLink(LinkInfoImpl.Kind.VALUE_TAG, field, constantVal, false) : new StringContent(constantVal); } @Override public Content commentTagsToOutput(DocTree holderTag, List<? extends DocTree> tags) { return commentTagsToOutput(holderTag, null, tags, false); } @Override public Content commentTagsToOutput(Element holder, List<? extends DocTree> tags) { return commentTagsToOutput(null, holder, tags, false); } @Override public Content commentTagsToOutput(DocTree holderTag, Element holder, List<? extends DocTree> tags, boolean isFirstSentence) { return htmlWriter.commentTagsToContent(holderTag, holder, tags, isFirstSentence, inSummary); } @Override public BaseConfiguration configuration() { return configuration; } @Override protected TypeElement getCurrentPageElement() { return htmlWriter.getCurrentPageElement(); } @SuppressWarnings("preview") private Content createAnchorAndSearchIndex(Element element, String tagText, String desc, boolean isSystemProperty) { Content result = null; if (isFirstSentence && inSummary) { result = new StringContent(tagText); } else { String anchorName = htmlWriter.links.getName(tagText); int count = htmlWriter.indexAnchorTable .compute(anchorName, (k, v) -> v == null ? 0 : v + 1); if (count > 0) { anchorName += "-" + count; } result = HtmlTree.SPAN(anchorName, HtmlStyle.searchTagResult, new StringContent(tagText)); if (options.createIndex() && !tagText.isEmpty()) { SearchIndexItem si = new SearchIndexItem(); si.setLabel(tagText); si.setDescription(desc); si.setUrl(htmlWriter.path.getPath() + "#" + anchorName); new SimpleElementVisitor14<Void, Void>() { @Override public Void visitModule(ModuleElement e, Void p) { si.setHolder(resources.getText("doclet.module") + " " + utils.getFullyQualifiedName(e)); return null; } @Override public Void visitPackage(PackageElement e, Void p) { si.setHolder(resources.getText("doclet.package") + " " + utils.getFullyQualifiedName(e)); return null; } @Override public Void visitType(TypeElement e, Void p) { si.setHolder(utils.getTypeElementName(e, true) + " " + utils.getFullyQualifiedName(e)); return null; } @Override public Void visitExecutable(ExecutableElement e, Void p) { si.setHolder(utils.getFullyQualifiedName(utils.getEnclosingTypeElement(e)) + "." + utils.getSimpleName(e) + utils.flatSignature(e, htmlWriter.getCurrentPageElement())); return null; } @Override public Void visitVariable(VariableElement e, Void p) { TypeElement te = utils.getEnclosingTypeElement(e); si.setHolder(utils.getFullyQualifiedName(te) + "." + utils.getSimpleName(e)); return null; } @Override public Void visitUnknown(Element e, Void p) { if (e instanceof DocletElement) { DocletElement de = (DocletElement) e; si.setElement(de); switch (de.getSubKind()) { case OVERVIEW: si.setHolder(resources.getText("doclet.Overview")); break; case DOCFILE: si.setHolder(getHolderName(de)); break; default: throw new IllegalStateException(); } return null; } else { return super.visitUnknown(e, p); } } @Override protected Void defaultAction(Element e, Void p) { si.setHolder(utils.getFullyQualifiedName(e)); return null; } }.visit(element); si.setCategory(isSystemProperty ? Category.SYSTEM_PROPERTY : Category.INDEX); configuration.searchItems.add(si); } } return result; } private String getHolderName(DocletElement de) { PackageElement pe = de.getPackageElement(); if (pe.isUnnamed()) { // if package is unnamed use enclosing module only if it is named Element ee = pe.getEnclosingElement(); if (ee instanceof ModuleElement && !((ModuleElement)ee).isUnnamed()) { return resources.getText("doclet.module") + " " + utils.getFullyQualifiedName(ee); } return pe.toString(); // "Unnamed package" or similar } return resources.getText("doclet.package") + " " + utils.getFullyQualifiedName(pe); } }