]> git.plutz.net Git - cgilite/blobdiff - markdown.awk
bugfix: endless loop in _startlist because of lacking indention removal
[cgilite] / markdown.awk
index eb40210c9bc78c8cacadda40389c73df2e8d6d6b..b5aa538055fe017b8c2b2827f23631f5d91d23b3 100755 (executable)
@@ -41,7 +41,7 @@
 # - [x] Automatic <section>-wrapping (custom)
 # -  ?  Heading identifiers (php md, pandoc)
 #   - [x] Heading attributes (custom)
-#   - [ ] <hr> ends section
+#   - [ ] <hr> terminates section
 # - [x] Automatic heading identifiers (custom)
 # - [x] Fenced code blocks (php md, pandoc)
 #   - [x] Fenced code attributes
@@ -690,39 +690,23 @@ function _block( block, LOCAL, st, len, text, title, attrib, href, guard, code,
     list = substr( block, 1, RLENGTH); block = substr( block, RLENGTH + 1);
     return "\n<dl>\n" _dlist( list ) "</dl>\n" _block( block );
 
-  # Unordered list
-  } else if ( match( block, "(^|\n) ? ? ?[-+*][ \t][^\n]+(\n|$)" \
-                 "(([ \t]*\n)* ? ? ?[-+*][ \t][^\n]+(\n|$)" \
-                 "|([ \t]*\n)*( ? ? ?\t|  +)[^\n]+(\n|$)" \
-                 "|[^\n \t][^\n]+(\n|$))*" ) ) {
-    st = RSTART; len = RLENGTH; list = substr( block, RSTART, RLENGTH);
-    sub("^\n", "", list); match(list, "^ ? ? ?[-+*]"); indent = RLENGTH;
-    gsub( "(^|\n) {0," indent - 1 "}", "\n", list); sub("^\n", "", list);
-
-    text = substr(block, 1, st - 1); block = substr(block, st + len);
-    if (match( list, "\n([0-9]+\\.|#\\.)[ \t]" )) {
-      block = substr(list, RSTART + 1) block;
-      list = substr(list, 1, RSTART);
-    }
-
-    return _block( text ) "<ul>\n" _list( list, "[-+*]" ) "</ul>\n" _block( block );
-
-  # Ordered list
-  } else if ( match( block, "(^|\n) ? ? ?([0-9]+\\.|#\\.)[ \t][^\n]+(\n|$)" \
-                 "(([ \t]*\n)* ? ? ?([0-9]+\\.|#\\.)[ \t][^\n]+(\n|$)" \
-                 "|([ \t]*\n)*( ? ? ?\t|  +)[^\n]+(\n|$)" \
-                 "|[^\n \t][^\n]+(\n|$))*" ) ) {
-    st = RSTART; len = RLENGTH; list = substr( block, RSTART, RLENGTH);
-    sub("^\n", "", list); match(list, "^ ? ? ?[0-9#]"); indent = RLENGTH;
-    gsub( "(^|\n) {0," indent - 1 "}", "\n", list); sub("^\n", "", list);
-
-    text = substr(block, 1, st - 1); block = substr(block, st + len);
-    if (match( list, "\n[-+*][ \t]" )) {
-      block = substr(list, RSTART + 1) block;
-      list = substr(list, 1, RSTART);
-    }
-
-    return _block( text ) "<ol>\n" _list( list, "([0-9]+\\.|#\\.)" ) "</ol>\n" _block( block );
+  # Unordered list types
+  } else if ( text = _startlist( block, "ul", "-",   "([+*]|[0-9]+\\.|#\\.|[0-9]+\\)|#\\))") ) {
+    return text;
+  } else if ( text = _startlist( block, "ul", "\\+", "([-*]|[0-9]+\\.|#\\.|[0-9]+\\)|#\\))") ) {
+    return text;
+  } else if ( text = _startlist( block, "ul", "\\*", "([-+]|[0-9]+\\.|#\\.|[0-9]+\\)|#\\))") ) {
+    return text;
+
+  # Ordered list types
+  } else if ( text = _startlist( block, "ol", "[0-9]+\\.", "([-+*]|#\\.|[0-9]+\\)|#\\))") ) {
+    return text;
+  } else if ( text = _startlist( block, "ol", "[0-9]+\\)", "([-+*]|[0-9]+\\.|#\\.|#\\))") ) {
+    return text;
+  } else if ( text = _startlist( block, "ol", "#\\.", "([-+*]|[0-9]+\\.|[0-9]+\\)|#\\))") ) {
+    return text;
+  } else if ( text = _startlist( block, "ol", "#\\)", "([-+*]|[0-9]+\\.|#\\.|[0-9]+\\))") ) {
+    return text;
 
   # Split paragraphs
   } else if ( match( block, /(^|\n)[[:space:]]*(\n|$)/) ) {
@@ -741,16 +725,36 @@ function _block( block, LOCAL, st, len, text, title, attrib, href, guard, code,
   }
 }
 
-function _list (block, mark, LOCAL, len, st, text, indent, task) {
+function _startlist(block, type, mark, exclude, LOCAL, st, len, list, indent, text) {
+  if (match( block, "(^|\n) ? ? ?" mark "[ \t][^\n]+(\n|$)" \
+         "(([ \t]*\n)* ? ? ?" mark "[ \t][^\n]+(\n|$)" \
+         "|([ \t]*\n)*( ? ? ?\t|  +)[^\n]+(\n|$)" \
+         "|[^\n \t][^\n]+(\n|$))*" ) ) {
+    st = RSTART; len = RLENGTH; list = substr( block, RSTART, RLENGTH);
+
+    sub("^\n", "", list); match(list, "^ ? ? ?"); indent = RLENGTH;
+    gsub( "(^|\n) {0," indent "}", "\n", list); sub("^\n", "", list);
+
+    text = substr(block, 1, st - 1); block = substr(block, st + len);
+    if (match( list, "\n" exclude "[ \t]" )) {
+      block = substr(list, RSTART + 1) block;
+      list = substr(list, 1, RSTART);
+    }
+
+    return _block( text ) "<" type ">\n" _list( list, mark ) "</" type ">\n" _block( block );
+  } else return 0;
+}
+
+function _list (block, mark, p, LOCAL, len, st, text, indent, task) {
   if ( match(block, "^([ \t]*\n)*$")) return;
+
   match(block, "^" mark "[ \t]"); indent = RLENGTH;
   sub("^" mark "[ \t]", "", block);
 
-  match( block, "\n" mark "[ \t][^\n]+(\n|$)" \
-      "(([ \t]*\n)* ? ? ?" mark "[ \t][^\n]+(\n|$)" \
-      "|([ \t]*\n)*( ? ? ?\t|  +)[^\n]+(\n|$)" \
-      "|[^\n \t][^\n]+(\n|$))*");
-  (RLENGTH == -1) ? st = length(block) + 1 : st = RSTART;
+  if (match(block, /\n[ \t]*\n/)) p = 1;
+
+  match( block, "\n" mark "[ \t][^\n]+(\n|$)" );
+  st = (RLENGTH == -1) ? length(block) + 1 : RSTART;
   text = substr(block, 1, st); block = substr(block, st + 1);
 
   gsub("\n {0," indent "}", "\n", text);
@@ -760,13 +764,13 @@ function _list (block, mark, LOCAL, len, st, text, indent, task) {
          match( text, /^\[\/\]/  ) ? "<li class=\"task partial\"><input type=checkbox disabled>"      : \
          match( text, /^\[\?\]/  ) ? "<li class=\"task unsure\"><input type=checkbox disabled>"       : \
          match( text, /^\[[xX]\]/) ? "<li class=\"task done\"><input type=checkbox disabled checked>" : "<li>";
-  sub(/^\[[-? /xX]\]/, "", text);
+  sub(/^\[[-? \/xX]\]/, "", text);
 
   text = _nblock( text );
-  if (match( text, "^<p>(</p[^>]|</[^p]|<[^/]|[^<])*</p>\n$" ))
+  if ( ! p && match( text, "^<p>(</p[^>]|</[^p]|<[^/]|[^<])*</p>\n$" ))
      gsub( "(^<p>|</p>\n$)", "", text);
 
-  return task text "</li>\n" _list(block, mark);
+  return task text "</li>\n" _list(block, mark, p);
 }
 
 function _dlist (block, LOCAL, len, st, text, indent, p) {