diff --git a/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java b/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java index 6e35d40ed2440aa27c9e8f5ddabb995d416416a7..85b5d22755e5cda29de92f4d1cf313c0bff670b7 100644 --- a/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java +++ b/src/share/classes/com/sun/tools/doclets/formats/html/HtmlDocletWriter.java @@ -1620,14 +1620,8 @@ public class HtmlDocletWriter extends HtmlDocWriter { if (isFirstSentence) { text = removeNonInlineHtmlTags(text); } - StringTokenizer lines = new StringTokenizer(text, "\r\n", true); - StringBuilder textBuff = new StringBuilder(); - while (lines.hasMoreTokens()) { - StringBuilder line = new StringBuilder(lines.nextToken()); - Util.replaceTabs(configuration, line); - textBuff.append(line.toString()); - } - result.addContent(new RawHtml(textBuff.toString())); + text = Util.replaceTabs(configuration, text); + result.addContent(new RawHtml(text)); } } return result; @@ -1732,60 +1726,55 @@ public class HtmlDocletWriter extends HtmlDocWriter { return text; } - public String removeNonInlineHtmlTags(String text) { - if (text.indexOf('<') < 0) { - return text; + static Set blockTags = new HashSet<>(); + static { + for (HtmlTag t: HtmlTag.values()) { + if (t.blockType == HtmlTag.BlockType.BLOCK) + blockTags.add(t.value); } - String noninlinetags[] = { "", "
    ", "
", - "
", "
", "", "
", - "", "", "", "", - "", "", "

", "

", - "
  • ", "
  • ", "
    ", "
    ", - "", "", "
    ", "
    ", - "

    ", "

    ", "

    ", "

    ", - "

    ", "

    ", "

    ", "

    ", - "
    ", "
    ", "
    ", "
    ", - "
    ", "
    ", "", "", - "", "", "
    ", - "
    ", "
    ", - "
    ", "
    ", - "", "
      ", "
    ", - "
    ", "
    ", "", "
    ", - "", "", "", "", - "", "", "

    ", "

    ", - "
  • ", "
  • ", "
    ", "
    ", - "", "", "
    ", "
    ", - "

    ", "

    ", "

    ", "

    ", - "

    ", "

    ", "

    ", "

    ", - "
    ", "
    ", "
    ", "
    ", - "
    ", "
    ", "", "", - "", "", "
    ", - "
    ", "
    ", - "
    ", "
    " - }; - for (int i = 0; i < noninlinetags.length; i++) { - text = replace(text, noninlinetags[i], ""); - } - return text; } - public String replace(String text, String tobe, String by) { - while (true) { - int startindex = text.indexOf(tobe); - if (startindex < 0) { - return text; + public static String removeNonInlineHtmlTags(String text) { + final int len = text.length(); + + int startPos = 0; // start of text to copy + int lessThanPos = text.indexOf('<'); // position of latest '<' + if (lessThanPos < 0) { + return text; + } + + StringBuilder result = new StringBuilder(); + main: while (lessThanPos != -1) { + int currPos = lessThanPos + 1; + if (currPos == len) + break; + char ch = text.charAt(currPos); + if (ch == '/') { + if (++currPos == len) + break; + ch = text.charAt(currPos); } - int endindex = startindex + tobe.length(); - StringBuilder replaced = new StringBuilder(); - if (startindex > 0) { - replaced.append(text.substring(0, startindex)); + int tagPos = currPos; + while (isHtmlTagLetterOrDigit(ch)) { + if (++currPos == len) + break main; + ch = text.charAt(currPos); } - replaced.append(by); - if (text.length() > endindex) { - replaced.append(text.substring(endindex)); + if (ch == '>' && blockTags.contains(text.substring(tagPos, currPos).toLowerCase())) { + result.append(text, startPos, lessThanPos); + startPos = currPos + 1; } - text = replaced.toString(); + lessThanPos = text.indexOf('<', currPos); } + result.append(text.substring(startPos)); + + return result.toString(); + } + + private static boolean isHtmlTagLetterOrDigit(char ch) { + return ('a' <= ch && ch <= 'z') || + ('A' <= ch && ch <= 'Z') || + ('1' <= ch && ch <= '6'); } /** diff --git a/src/share/classes/com/sun/tools/doclets/formats/html/SourceToHTMLConverter.java b/src/share/classes/com/sun/tools/doclets/formats/html/SourceToHTMLConverter.java index 41ca9ea7192d2eacb1de0486cf1217427033810e..9d528a9352187cca44ff1ef40585ada471334a73 100644 --- a/src/share/classes/com/sun/tools/doclets/formats/html/SourceToHTMLConverter.java +++ b/src/share/classes/com/sun/tools/doclets/formats/html/SourceToHTMLConverter.java @@ -269,9 +269,7 @@ public class SourceToHTMLConverter { */ private void addLine(Content pre, String line, int currentLineNo) { if (line != null) { - StringBuilder lineBuffer = new StringBuilder(line); - Util.replaceTabs(configuration, lineBuffer); - pre.addContent(lineBuffer.toString()); + pre.addContent(Util.replaceTabs(configuration, line)); Content anchor = HtmlTree.A_NAME("line." + Integer.toString(currentLineNo)); pre.addContent(anchor); pre.addContent(NEW_LINE); diff --git a/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlTag.java b/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlTag.java index 08b6123580796caa28c25b97e58b813dceb33958..fe9f5545937b554c6db093ce6ebeb074a09c961e 100644 --- a/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlTag.java +++ b/src/share/classes/com/sun/tools/doclets/formats/html/markup/HtmlTag.java @@ -44,6 +44,7 @@ public enum HtmlTag { CENTER, CODE(BlockType.INLINE, EndTag.END), DD, + DIR, DIV, DL, DT, @@ -63,6 +64,7 @@ public enum HtmlTag { I(BlockType.INLINE, EndTag.END), IMG(BlockType.INLINE, EndTag.NOEND), LI, + LISTING, LINK(BlockType.OTHER, EndTag.NOEND), MENU, META(BlockType.OTHER, EndTag.NOEND), @@ -85,14 +87,14 @@ public enum HtmlTag { TT(BlockType.INLINE, EndTag.END), UL; - protected final BlockType blockType; - protected final EndTag endTag; - private final String value; + public final BlockType blockType; + public final EndTag endTag; + public final String value; /** * Enum representing the type of HTML element. */ - protected static enum BlockType { + public static enum BlockType { BLOCK, INLINE, OTHER; @@ -101,7 +103,7 @@ public enum HtmlTag { /** * Enum representing HTML end tag requirement. */ - protected static enum EndTag { + public static enum EndTag { END, NOEND; } diff --git a/src/share/classes/com/sun/tools/doclets/internal/toolkit/util/Util.java b/src/share/classes/com/sun/tools/doclets/internal/toolkit/util/Util.java index 120bbb48fd07f69ac765301eb6053fa2968e0286..acdae4545a723898a2dfc13a04be580a0bfbe5f7 100644 --- a/src/share/classes/com/sun/tools/doclets/internal/toolkit/util/Util.java +++ b/src/share/classes/com/sun/tools/doclets/internal/toolkit/util/Util.java @@ -593,20 +593,42 @@ public class Util { } /** - * Replace all tabs with the appropriate number of spaces. + * Replace all tabs in a string with the appropriate number of spaces. + * The string may be a multi-line string. * @param configuration the doclet configuration defining the setting for the * tab length. - * @param sb the StringBuilder in which to replace the tabs + * @param text the text for which the tabs should be expanded + * @return the text with all tabs expanded */ - public static void replaceTabs(Configuration configuration, StringBuilder sb) { - int tabLength = configuration.sourcetab; - String whitespace = configuration.tabSpaces; - int index = 0; - while ((index = sb.indexOf("\t", index)) != -1) { - int spaceCount = tabLength - index % tabLength; - sb.replace(index, index+1, whitespace.substring(0, spaceCount)); - index += spaceCount; + public static String replaceTabs(Configuration configuration, String text) { + if (text.indexOf("\t") == -1) + return text; + + final int tabLength = configuration.sourcetab; + final String whitespace = configuration.tabSpaces; + final int textLength = text.length(); + StringBuilder result = new StringBuilder(textLength); + int pos = 0; + int lineLength = 0; + for (int i = 0; i < textLength; i++) { + char ch = text.charAt(i); + switch (ch) { + case '\n': case '\r': + lineLength = 0; + break; + case '\t': + result.append(text, pos, i); + int spaceCount = tabLength - lineLength % tabLength; + result.append(whitespace, 0, spaceCount); + lineLength += spaceCount; + pos = i + 1; + break; + default: + lineLength++; + } } + result.append(text, pos, textLength); + return result.toString(); } /**