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 e4422a9948e..45de5e4a9ab 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 @@ -27,6 +27,7 @@ package jdk.javadoc.internal.doclets.formats.html; import java.util.ArrayList; import java.util.Collections; +import java.util.HashMap; import java.util.LinkedList; import java.util.List; import java.util.ListIterator; @@ -196,6 +197,14 @@ public class HtmlDocletWriter { protected Script mainBodyScript; + /** + * A table of the anchors used for at-index and related tags, + * so that they can be made unique by appending a suitable suffix. + * (Ideally, javadoc should be tracking all id's generated in a file + * to avoid generating duplicates.) + */ + Map indexAnchorTable = new HashMap<>(); + /** * Constructor to construct the HtmlStandardWriter object. * 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 481631f54f7..25697432145 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 @@ -409,11 +409,16 @@ public class TagletWriterImpl extends TagletWriter { } private Content createAnchorAndSearchIndex(Element element, String tagText, String desc){ - String anchorName = htmlWriter.links.getName(tagText); Content result = null; if (isFirstSentence && inSummary) { result = new StringContent(tagText); } else { + String anchorName = htmlWriter.links.getName(tagText); + int count = htmlWriter.indexAnchorTable.computeIfAbsent(anchorName, s -> 0); + htmlWriter.indexAnchorTable.put(anchorName, count + 1); + if (count > 0) { + anchorName += "-" + count; + } result = HtmlTree.A_ID(HtmlStyle.searchTagResult, anchorName, new StringContent(tagText)); if (configuration.createindex && !tagText.isEmpty()) { SearchIndexItem si = new SearchIndexItem(); diff --git a/test/langtools/jdk/javadoc/doclet/testIndexTaglet/TestIndexTaglet.java b/test/langtools/jdk/javadoc/doclet/testIndexTaglet/TestIndexTaglet.java index e5befb297f2..e37f09668a5 100644 --- a/test/langtools/jdk/javadoc/doclet/testIndexTaglet/TestIndexTaglet.java +++ b/test/langtools/jdk/javadoc/doclet/testIndexTaglet/TestIndexTaglet.java @@ -47,7 +47,7 @@ public class TestIndexTaglet extends JavadocTester { public static void main(String... args) throws Exception { TestIndexTaglet tester = new TestIndexTaglet(); - tester.runTests(m -> new Object[]{Paths.get(m.getName())}); + tester.runTests(m -> new Object[] { Paths.get(m.getName()) }); } TestIndexTaglet() { @@ -104,4 +104,27 @@ public class TestIndexTaglet extends JavadocTester { checkOutput(Output.OUT, true, "warning: {@index} tag, which expands to , within "); } + + @Test + public void testDuplicateReferences(Path base) throws Exception { + Path srcDir = base.resolve("src"); + Path outDir = base.resolve("out"); + + new ClassBuilder(tb, "pkg.A") + .setModifiers("public", "class") + .setComments("This is a class. Here is {@index foo first}.") + .addMembers(MethodBuilder.parse("public void m() {}") + .setComments("This is a method. Here is {@index foo second}.")) + .write(srcDir); + + javadoc("-d", outDir.toString(), + "-sourcepath", srcDir.toString(), + "pkg"); + + checkExit(Exit.OK); + + checkOutput("pkg/A.html", true, + "This is a class. Here is foo.", + "This is a method. Here is foo."); + } } diff --git a/test/langtools/jdk/javadoc/doclet/testSystemPropertyTaglet/TestSystemPropertyTaglet.java b/test/langtools/jdk/javadoc/doclet/testSystemPropertyTaglet/TestSystemPropertyTaglet.java index b3597bc3188..f87a3ccaebe 100644 --- a/test/langtools/jdk/javadoc/doclet/testSystemPropertyTaglet/TestSystemPropertyTaglet.java +++ b/test/langtools/jdk/javadoc/doclet/testSystemPropertyTaglet/TestSystemPropertyTaglet.java @@ -47,7 +47,7 @@ public class TestSystemPropertyTaglet extends JavadocTester { public static void main(String... args) throws Exception { TestSystemPropertyTaglet tester = new TestSystemPropertyTaglet(); - tester.runTests(m -> new Object[]{Paths.get(m.getName())}); + tester.runTests(m -> new Object[] { Paths.get(m.getName()) }); } TestSystemPropertyTaglet() { @@ -118,4 +118,27 @@ public class TestSystemPropertyTaglet extends JavadocTester { checkOutput(Output.OUT, true, "warning: {@systemProperty} tag, which expands to , within "); } + + @Test + public void testDuplicateReferences(Path base) throws Exception { + Path srcDir = base.resolve("src"); + Path outDir = base.resolve("out"); + + new ClassBuilder(tb, "pkg.A") + .setModifiers("public", "class") + .setComments("This is a class. Here is {@systemProperty foo}.") + .addMembers(MethodBuilder.parse("public void m() {}") + .setComments("This is a method. Here is {@systemProperty foo}.")) + .write(srcDir); + + javadoc("-d", outDir.toString(), + "-sourcepath", srcDir.toString(), + "pkg"); + + checkExit(Exit.OK); + + checkOutput("pkg/A.html", true, + "This is a class. Here is foo.", + "This is a method. Here is foo."); + } }