123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341 |
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #include <assert.h>
- #include "cmark_ctype.h"
- #include "config.h"
- #include "cmark.h"
- #include "node.h"
- #include "buffer.h"
- #include "houdini.h"
- #include "scanners.h"
- #define BUFFER_SIZE 100
- // Functions to convert cmark_nodes to HTML strings.
- static void escape_html(cmark_strbuf *dest, const unsigned char *source,
- bufsize_t length) {
- houdini_escape_html0(dest, source, length, 0);
- }
- static CMARK_INLINE void cr(cmark_strbuf *html) {
- if (html->size && html->ptr[html->size - 1] != '\n')
- cmark_strbuf_putc(html, '\n');
- }
- struct render_state {
- cmark_strbuf *html;
- cmark_node *plain;
- };
- static void S_render_sourcepos(cmark_node *node, cmark_strbuf *html,
- int options) {
- char buffer[BUFFER_SIZE];
- if (CMARK_OPT_SOURCEPOS & options) {
- snprintf(buffer, BUFFER_SIZE, " data-sourcepos=\"%d:%d-%d:%d\"",
- cmark_node_get_start_line(node), cmark_node_get_start_column(node),
- cmark_node_get_end_line(node), cmark_node_get_end_column(node));
- cmark_strbuf_puts(html, buffer);
- }
- }
- static int S_render_node(cmark_node *node, cmark_event_type ev_type,
- struct render_state *state, int options) {
- cmark_node *parent;
- cmark_node *grandparent;
- cmark_strbuf *html = state->html;
- char start_heading[] = "<h0";
- char end_heading[] = "</h0";
- bool tight;
- char buffer[BUFFER_SIZE];
- bool entering = (ev_type == CMARK_EVENT_ENTER);
- if (state->plain == node) { // back at original node
- state->plain = NULL;
- }
- if (state->plain != NULL) {
- switch (node->type) {
- case CMARK_NODE_TEXT:
- case CMARK_NODE_CODE:
- case CMARK_NODE_HTML_INLINE:
- escape_html(html, node->as.literal.data, node->as.literal.len);
- break;
- case CMARK_NODE_LINEBREAK:
- case CMARK_NODE_SOFTBREAK:
- cmark_strbuf_putc(html, ' ');
- break;
- default:
- break;
- }
- return 1;
- }
- switch (node->type) {
- case CMARK_NODE_DOCUMENT:
- break;
- case CMARK_NODE_BLOCK_QUOTE:
- if (entering) {
- cr(html);
- cmark_strbuf_puts(html, "<blockquote");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, ">\n");
- } else {
- cr(html);
- cmark_strbuf_puts(html, "</blockquote>\n");
- }
- break;
- case CMARK_NODE_LIST: {
- cmark_list_type list_type = node->as.list.list_type;
- int start = node->as.list.start;
- if (entering) {
- cr(html);
- if (list_type == CMARK_BULLET_LIST) {
- cmark_strbuf_puts(html, "<ul");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, ">\n");
- } else if (start == 1) {
- cmark_strbuf_puts(html, "<ol");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, ">\n");
- } else {
- snprintf(buffer, BUFFER_SIZE, "<ol start=\"%d\"", start);
- cmark_strbuf_puts(html, buffer);
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, ">\n");
- }
- } else {
- cmark_strbuf_puts(html,
- list_type == CMARK_BULLET_LIST ? "</ul>\n" : "</ol>\n");
- }
- break;
- }
- case CMARK_NODE_ITEM:
- if (entering) {
- cr(html);
- cmark_strbuf_puts(html, "<li");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_putc(html, '>');
- } else {
- cmark_strbuf_puts(html, "</li>\n");
- }
- break;
- case CMARK_NODE_HEADING:
- if (entering) {
- cr(html);
- start_heading[2] = (char)('0' + node->as.heading.level);
- cmark_strbuf_puts(html, start_heading);
- S_render_sourcepos(node, html, options);
- cmark_strbuf_putc(html, '>');
- } else {
- end_heading[3] = (char)('0' + node->as.heading.level);
- cmark_strbuf_puts(html, end_heading);
- cmark_strbuf_puts(html, ">\n");
- }
- break;
- case CMARK_NODE_CODE_BLOCK:
- cr(html);
- if (node->as.code.info.len == 0) {
- cmark_strbuf_puts(html, "<pre");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, "><code>");
- } else {
- bufsize_t first_tag = 0;
- while (first_tag < node->as.code.info.len &&
- !cmark_isspace(node->as.code.info.data[first_tag])) {
- first_tag += 1;
- }
- cmark_strbuf_puts(html, "<pre");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, "><code class=\"language-");
- escape_html(html, node->as.code.info.data, first_tag);
- cmark_strbuf_puts(html, "\">");
- }
- escape_html(html, node->as.code.literal.data, node->as.code.literal.len);
- cmark_strbuf_puts(html, "</code></pre>\n");
- break;
- case CMARK_NODE_HTML_BLOCK:
- cr(html);
- if (!(options & CMARK_OPT_UNSAFE)) {
- cmark_strbuf_puts(html, "<!-- raw HTML omitted -->");
- } else {
- cmark_strbuf_put(html, node->as.literal.data, node->as.literal.len);
- }
- cr(html);
- break;
- case CMARK_NODE_CUSTOM_BLOCK:
- cr(html);
- if (entering) {
- cmark_strbuf_put(html, node->as.custom.on_enter.data,
- node->as.custom.on_enter.len);
- } else {
- cmark_strbuf_put(html, node->as.custom.on_exit.data,
- node->as.custom.on_exit.len);
- }
- cr(html);
- break;
- case CMARK_NODE_THEMATIC_BREAK:
- cr(html);
- cmark_strbuf_puts(html, "<hr");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_puts(html, " />\n");
- break;
- case CMARK_NODE_PARAGRAPH:
- parent = cmark_node_parent(node);
- grandparent = cmark_node_parent(parent);
- if (grandparent != NULL && grandparent->type == CMARK_NODE_LIST) {
- tight = grandparent->as.list.tight;
- } else {
- tight = false;
- }
- if (!tight) {
- if (entering) {
- cr(html);
- cmark_strbuf_puts(html, "<p");
- S_render_sourcepos(node, html, options);
- cmark_strbuf_putc(html, '>');
- } else {
- cmark_strbuf_puts(html, "</p>\n");
- }
- }
- break;
- case CMARK_NODE_TEXT:
- escape_html(html, node->as.literal.data, node->as.literal.len);
- break;
- case CMARK_NODE_LINEBREAK:
- cmark_strbuf_puts(html, "<br />\n");
- break;
- case CMARK_NODE_SOFTBREAK:
- if (options & CMARK_OPT_HARDBREAKS) {
- cmark_strbuf_puts(html, "<br />\n");
- } else if (options & CMARK_OPT_NOBREAKS) {
- cmark_strbuf_putc(html, ' ');
- } else {
- cmark_strbuf_putc(html, '\n');
- }
- break;
- case CMARK_NODE_CODE:
- cmark_strbuf_puts(html, "<code>");
- escape_html(html, node->as.literal.data, node->as.literal.len);
- cmark_strbuf_puts(html, "</code>");
- break;
- case CMARK_NODE_HTML_INLINE:
- if (!(options & CMARK_OPT_UNSAFE)) {
- cmark_strbuf_puts(html, "<!-- raw HTML omitted -->");
- } else {
- cmark_strbuf_put(html, node->as.literal.data, node->as.literal.len);
- }
- break;
- case CMARK_NODE_CUSTOM_INLINE:
- if (entering) {
- cmark_strbuf_put(html, node->as.custom.on_enter.data,
- node->as.custom.on_enter.len);
- } else {
- cmark_strbuf_put(html, node->as.custom.on_exit.data,
- node->as.custom.on_exit.len);
- }
- break;
- case CMARK_NODE_STRONG:
- if (entering) {
- cmark_strbuf_puts(html, "<strong>");
- } else {
- cmark_strbuf_puts(html, "</strong>");
- }
- break;
- case CMARK_NODE_EMPH:
- if (entering) {
- cmark_strbuf_puts(html, "<em>");
- } else {
- cmark_strbuf_puts(html, "</em>");
- }
- break;
- case CMARK_NODE_LINK:
- if (entering) {
- cmark_strbuf_puts(html, "<a href=\"");
- if ((options & CMARK_OPT_UNSAFE) ||
- !(scan_dangerous_url(&node->as.link.url, 0))) {
- houdini_escape_href(html, node->as.link.url.data,
- node->as.link.url.len);
- }
- if (node->as.link.title.len) {
- cmark_strbuf_puts(html, "\" title=\"");
- escape_html(html, node->as.link.title.data, node->as.link.title.len);
- }
- cmark_strbuf_puts(html, "\">");
- } else {
- cmark_strbuf_puts(html, "</a>");
- }
- break;
- case CMARK_NODE_IMAGE:
- if (entering) {
- cmark_strbuf_puts(html, "<img src=\"");
- if ((options & CMARK_OPT_UNSAFE) ||
- !(scan_dangerous_url(&node->as.link.url, 0))) {
- houdini_escape_href(html, node->as.link.url.data,
- node->as.link.url.len);
- }
- cmark_strbuf_puts(html, "\" alt=\"");
- state->plain = node;
- } else {
- if (node->as.link.title.len) {
- cmark_strbuf_puts(html, "\" title=\"");
- escape_html(html, node->as.link.title.data, node->as.link.title.len);
- }
- cmark_strbuf_puts(html, "\" />");
- }
- break;
- default:
- assert(false);
- break;
- }
- // cmark_strbuf_putc(html, 'x');
- return 1;
- }
- char *cmark_render_html(cmark_node *root, int options) {
- char *result;
- cmark_strbuf html = CMARK_BUF_INIT(cmark_node_mem(root));
- cmark_event_type ev_type;
- cmark_node *cur;
- struct render_state state = {&html, NULL};
- cmark_iter *iter = cmark_iter_new(root);
- while ((ev_type = cmark_iter_next(iter)) != CMARK_EVENT_DONE) {
- cur = cmark_iter_get_node(iter);
- S_render_node(cur, ev_type, &state, options);
- }
- result = (char *)cmark_strbuf_detach(&html);
- cmark_iter_free(iter);
- return result;
- }
|