aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-markdown
diff options
context:
space:
mode:
Diffstat (limited to 'sonar-markdown')
-rw-r--r--sonar-markdown/src/main/java/org/sonar/markdown/HtmlListChannel.java83
-rw-r--r--sonar-markdown/src/test/java/org/sonar/markdown/MarkdownTest.java17
2 files changed, 87 insertions, 13 deletions
diff --git a/sonar-markdown/src/main/java/org/sonar/markdown/HtmlListChannel.java b/sonar-markdown/src/main/java/org/sonar/markdown/HtmlListChannel.java
index 7267db63b26..c8842f0a1c8 100644
--- a/sonar-markdown/src/main/java/org/sonar/markdown/HtmlListChannel.java
+++ b/sonar-markdown/src/main/java/org/sonar/markdown/HtmlListChannel.java
@@ -23,21 +23,61 @@ import org.sonar.channel.Channel;
import org.sonar.channel.CodeReader;
import org.sonar.channel.RegexChannel;
+/**
+ * Lists come in two flavors:
+ * <ul>
+ * <li>Unordered lists, triggered by lines that start with a <code>*</code></li>
+ * <li>Ordered lists (added in 4.4), triggered by lines that start with a digit followed by a <code>.</code></li>
+ * </ul>
+
+ * E.g., the input:
+ * <pre>
+ * * One
+ * * Two
+ * * Three
+ * </pre>
+ * will produce:
+ * {@literal<ul>}{@literal<li>}One{@literal</li>}
+ * {@literal<li>}Two{@literal</li>}
+ * {@literal<li>}Three{@literal</li>}{@literal</ul>}
+ *
+ * Whereas the input:
+ * <pre>
+ * 1. One
+ * 1. Two
+ * 1. Three
+ * </pre>
+ * will produce:
+ * {@literal<ol>}{@literal<li>}One{@literal</li>}
+ * {@literal<li>}Two{@literal</li>}
+ * {@literal<li>}Three{@literal</li>}{@literal</ol>}
+ *
+ * @since 2.10.1
+ */
class HtmlListChannel extends Channel<MarkdownOutput> {
- private ListElementChannel listElement = new ListElementChannel();
+ private OrderedListElementChannel orderedListElement = new OrderedListElementChannel();
+ private UnorderedListElementChannel unorderedListElement = new UnorderedListElementChannel();
private EndOfLine endOfLine = new EndOfLine();
private boolean pendingListConstruction;
@Override
public boolean consume(CodeReader code, MarkdownOutput output) {
try {
- if (code.getColumnPosition() == 0 && listElement.consume(code, output)) {
- while (endOfLine.consume(code, output) && listElement.consume(code, output)) {
- // consume input
+ ListElementChannel currentChannel = null;
+ if (code.getColumnPosition() == 0) {
+ if (orderedListElement.consume(code, output)) {
+ currentChannel = orderedListElement;
+ } else if (unorderedListElement.consume(code, output)) {
+ currentChannel = unorderedListElement;
+ }
+ if (currentChannel != null) {
+ while (endOfLine.consume(code, output) && currentChannel.consume(code, output)) {
+ // consume input
+ }
+ output.append("</" + currentChannel.listElement + ">");
+ return true;
}
- output.append("</ul>");
- return true;
}
return false;
} finally {
@@ -45,16 +85,31 @@ class HtmlListChannel extends Channel<MarkdownOutput> {
}
}
- private class ListElementChannel extends RegexChannel<MarkdownOutput> {
+ private class OrderedListElementChannel extends ListElementChannel {
+ public OrderedListElementChannel() {
+ super("\\d\\.", "ol");
+ }
+ }
+
+ private class UnorderedListElementChannel extends ListElementChannel {
+ public UnorderedListElementChannel() {
+ super("\\*", "ul");
+ }
+ }
- public ListElementChannel() {
- super("\\s*+\\*\\s[^\r\n]*+");
+ private abstract class ListElementChannel extends RegexChannel<MarkdownOutput> {
+
+ private String listElement;
+
+ protected ListElementChannel(String markerRegexp, String listElement) {
+ super("\\s*+" + markerRegexp + "\\s[^\r\n]*+");
+ this.listElement = listElement;
}
@Override
protected void consume(CharSequence token, MarkdownOutput output) {
if (!pendingListConstruction) {
- output.append("<ul>");
+ output.append("<" + listElement + ">");
pendingListConstruction = true;
}
output.append("<li>");
@@ -64,8 +119,12 @@ class HtmlListChannel extends Channel<MarkdownOutput> {
private int searchIndexOfFirstCharacter(CharSequence token) {
for (int index = 0; index < token.length(); index++) {
- if (token.charAt(index) == '*') {
- while (++index<token.length()) {
+ if (token.charAt(index) == '*'
+ || Character.isDigit(token.charAt(index))) {
+ if (token.charAt(index + 1) == '.') {
+ index ++;
+ }
+ while (++ index < token.length()) {
if (token.charAt(index) != ' ') {
return index;
}
diff --git a/sonar-markdown/src/test/java/org/sonar/markdown/MarkdownTest.java b/sonar-markdown/src/test/java/org/sonar/markdown/MarkdownTest.java
index 62048e6e5a6..13b0099976a 100644
--- a/sonar-markdown/src/test/java/org/sonar/markdown/MarkdownTest.java
+++ b/sonar-markdown/src/test/java/org/sonar/markdown/MarkdownTest.java
@@ -37,10 +37,25 @@ public class MarkdownTest {
}
@Test
- public void shouldDecorateList() {
+ public void shouldDecorateUnorderedList() {
assertThat(Markdown.convertToHtml(" * one\r* two\r\n* three\n * \n *five"))
.isEqualTo("<ul><li>one</li>\r<li>two</li>\r\n<li>three</li>\n<li> </li>\n</ul> *five");
assertThat(Markdown.convertToHtml(" * one\r* two")).isEqualTo("<ul><li>one</li>\r<li>two</li></ul>");
+ assertThat(Markdown.convertToHtml("* \r*")).isEqualTo("<ul><li> </li>\r</ul>*");
+ }
+
+ @Test
+ public void shouldDecorateOrderedList() {
+ assertThat(Markdown.convertToHtml(" 1. one\r1. two\r\n1. three\n 1. \n 1.five"))
+ .isEqualTo("<ol><li>one</li>\r<li>two</li>\r\n<li>three</li>\n<li> </li>\n</ol> 1.five");
+ assertThat(Markdown.convertToHtml(" 1. one\r1. two")).isEqualTo("<ol><li>one</li>\r<li>two</li></ol>");
+ assertThat(Markdown.convertToHtml("1. \r1.")).isEqualTo("<ol><li> </li>\r</ol>1.");
+ }
+
+ @Test
+ public void shouldDecorateMixedOrderedAndUnorderedList() {
+ assertThat(Markdown.convertToHtml(" 1. one\r* two\r\n1. three\n * \n 1.five"))
+ .isEqualTo("<ol><li>one</li>\r</ol><ul><li>two</li>\r\n</ul><ol><li>three</li>\n</ol><ul><li> </li>\n</ul> 1.five");
}
@Test