markly 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (82) hide show
  1. checksums.yaml +7 -0
  2. data/bin/markly +94 -0
  3. data/ext/markly/arena.c +103 -0
  4. data/ext/markly/autolink.c +425 -0
  5. data/ext/markly/autolink.h +8 -0
  6. data/ext/markly/blocks.c +1585 -0
  7. data/ext/markly/buffer.c +278 -0
  8. data/ext/markly/buffer.h +116 -0
  9. data/ext/markly/case_fold_switch.inc +4327 -0
  10. data/ext/markly/chunk.h +135 -0
  11. data/ext/markly/cmark-gfm-core-extensions.h +54 -0
  12. data/ext/markly/cmark-gfm-extension_api.h +736 -0
  13. data/ext/markly/cmark-gfm-extensions_export.h +42 -0
  14. data/ext/markly/cmark-gfm.h +817 -0
  15. data/ext/markly/cmark-gfm_export.h +42 -0
  16. data/ext/markly/cmark-gfm_version.h +7 -0
  17. data/ext/markly/cmark.c +55 -0
  18. data/ext/markly/cmark_ctype.c +44 -0
  19. data/ext/markly/cmark_ctype.h +33 -0
  20. data/ext/markly/commonmark.c +519 -0
  21. data/ext/markly/config.h +76 -0
  22. data/ext/markly/core-extensions.c +27 -0
  23. data/ext/markly/entities.inc +2138 -0
  24. data/ext/markly/ext_scanners.c +1159 -0
  25. data/ext/markly/ext_scanners.h +24 -0
  26. data/ext/markly/extconf.rb +7 -0
  27. data/ext/markly/footnotes.c +40 -0
  28. data/ext/markly/footnotes.h +25 -0
  29. data/ext/markly/houdini.h +57 -0
  30. data/ext/markly/houdini_href_e.c +100 -0
  31. data/ext/markly/houdini_html_e.c +66 -0
  32. data/ext/markly/houdini_html_u.c +149 -0
  33. data/ext/markly/html.c +465 -0
  34. data/ext/markly/html.h +27 -0
  35. data/ext/markly/inlines.c +1633 -0
  36. data/ext/markly/inlines.h +29 -0
  37. data/ext/markly/iterator.c +159 -0
  38. data/ext/markly/iterator.h +26 -0
  39. data/ext/markly/latex.c +466 -0
  40. data/ext/markly/linked_list.c +37 -0
  41. data/ext/markly/man.c +278 -0
  42. data/ext/markly/map.c +122 -0
  43. data/ext/markly/map.h +41 -0
  44. data/ext/markly/markly.c +1226 -0
  45. data/ext/markly/markly.h +16 -0
  46. data/ext/markly/node.c +979 -0
  47. data/ext/markly/node.h +118 -0
  48. data/ext/markly/parser.h +58 -0
  49. data/ext/markly/plaintext.c +235 -0
  50. data/ext/markly/plugin.c +36 -0
  51. data/ext/markly/plugin.h +34 -0
  52. data/ext/markly/references.c +42 -0
  53. data/ext/markly/references.h +26 -0
  54. data/ext/markly/registry.c +63 -0
  55. data/ext/markly/registry.h +24 -0
  56. data/ext/markly/render.c +205 -0
  57. data/ext/markly/render.h +62 -0
  58. data/ext/markly/scanners.c +20382 -0
  59. data/ext/markly/scanners.h +62 -0
  60. data/ext/markly/scanners.re +326 -0
  61. data/ext/markly/strikethrough.c +167 -0
  62. data/ext/markly/strikethrough.h +9 -0
  63. data/ext/markly/syntax_extension.c +149 -0
  64. data/ext/markly/syntax_extension.h +34 -0
  65. data/ext/markly/table.c +803 -0
  66. data/ext/markly/table.h +12 -0
  67. data/ext/markly/tagfilter.c +60 -0
  68. data/ext/markly/tagfilter.h +8 -0
  69. data/ext/markly/tasklist.c +156 -0
  70. data/ext/markly/tasklist.h +8 -0
  71. data/ext/markly/utf8.c +317 -0
  72. data/ext/markly/utf8.h +35 -0
  73. data/ext/markly/xml.c +181 -0
  74. data/lib/markly.rb +43 -0
  75. data/lib/markly/flags.rb +37 -0
  76. data/lib/markly/markly.so +0 -0
  77. data/lib/markly/node.rb +70 -0
  78. data/lib/markly/node/inspect.rb +59 -0
  79. data/lib/markly/renderer.rb +133 -0
  80. data/lib/markly/renderer/html_renderer.rb +252 -0
  81. data/lib/markly/version.rb +5 -0
  82. metadata +211 -0
@@ -0,0 +1,29 @@
1
+ #ifndef CMARK_INLINES_H
2
+ #define CMARK_INLINES_H
3
+
4
+ #ifdef __cplusplus
5
+ extern "C" {
6
+ #endif
7
+
8
+ #include "references.h"
9
+
10
+ cmark_chunk cmark_clean_url(cmark_mem *mem, cmark_chunk *url);
11
+ cmark_chunk cmark_clean_title(cmark_mem *mem, cmark_chunk *title);
12
+
13
+ CMARK_GFM_EXPORT
14
+ void cmark_parse_inlines(cmark_parser *parser,
15
+ cmark_node *parent,
16
+ cmark_map *refmap,
17
+ int options);
18
+
19
+ bufsize_t cmark_parse_reference_inline(cmark_mem *mem, cmark_chunk *input,
20
+ cmark_map *refmap);
21
+
22
+ void cmark_inlines_add_special_character(unsigned char c, bool emphasis);
23
+ void cmark_inlines_remove_special_character(unsigned char c, bool emphasis);
24
+
25
+ #ifdef __cplusplus
26
+ }
27
+ #endif
28
+
29
+ #endif
@@ -0,0 +1,159 @@
1
+ #include <assert.h>
2
+ #include <stdlib.h>
3
+
4
+ #include "config.h"
5
+ #include "node.h"
6
+ #include "cmark-gfm.h"
7
+ #include "iterator.h"
8
+
9
+ cmark_iter *cmark_iter_new(cmark_node *root) {
10
+ if (root == NULL) {
11
+ return NULL;
12
+ }
13
+ cmark_mem *mem = root->content.mem;
14
+ cmark_iter *iter = (cmark_iter *)mem->calloc(1, sizeof(cmark_iter));
15
+ iter->mem = mem;
16
+ iter->root = root;
17
+ iter->cur.ev_type = CMARK_EVENT_NONE;
18
+ iter->cur.node = NULL;
19
+ iter->next.ev_type = CMARK_EVENT_ENTER;
20
+ iter->next.node = root;
21
+ return iter;
22
+ }
23
+
24
+ void cmark_iter_free(cmark_iter *iter) { iter->mem->free(iter); }
25
+
26
+ static bool S_is_leaf(cmark_node *node) {
27
+ switch (node->type) {
28
+ case CMARK_NODE_HTML_BLOCK:
29
+ case CMARK_NODE_THEMATIC_BREAK:
30
+ case CMARK_NODE_CODE_BLOCK:
31
+ case CMARK_NODE_TEXT:
32
+ case CMARK_NODE_SOFTBREAK:
33
+ case CMARK_NODE_LINEBREAK:
34
+ case CMARK_NODE_CODE:
35
+ case CMARK_NODE_HTML_INLINE:
36
+ return 1;
37
+ }
38
+ return 0;
39
+ }
40
+
41
+ cmark_event_type cmark_iter_next(cmark_iter *iter) {
42
+ cmark_event_type ev_type = iter->next.ev_type;
43
+ cmark_node *node = iter->next.node;
44
+
45
+ iter->cur.ev_type = ev_type;
46
+ iter->cur.node = node;
47
+
48
+ if (ev_type == CMARK_EVENT_DONE) {
49
+ return ev_type;
50
+ }
51
+
52
+ /* roll forward to next item, setting both fields */
53
+ if (ev_type == CMARK_EVENT_ENTER && !S_is_leaf(node)) {
54
+ if (node->first_child == NULL) {
55
+ /* stay on this node but exit */
56
+ iter->next.ev_type = CMARK_EVENT_EXIT;
57
+ } else {
58
+ iter->next.ev_type = CMARK_EVENT_ENTER;
59
+ iter->next.node = node->first_child;
60
+ }
61
+ } else if (node == iter->root) {
62
+ /* don't move past root */
63
+ iter->next.ev_type = CMARK_EVENT_DONE;
64
+ iter->next.node = NULL;
65
+ } else if (node->next) {
66
+ iter->next.ev_type = CMARK_EVENT_ENTER;
67
+ iter->next.node = node->next;
68
+ } else if (node->parent) {
69
+ iter->next.ev_type = CMARK_EVENT_EXIT;
70
+ iter->next.node = node->parent;
71
+ } else {
72
+ assert(false);
73
+ iter->next.ev_type = CMARK_EVENT_DONE;
74
+ iter->next.node = NULL;
75
+ }
76
+
77
+ return ev_type;
78
+ }
79
+
80
+ void cmark_iter_reset(cmark_iter *iter, cmark_node *current,
81
+ cmark_event_type event_type) {
82
+ iter->next.ev_type = event_type;
83
+ iter->next.node = current;
84
+ cmark_iter_next(iter);
85
+ }
86
+
87
+ cmark_node *cmark_iter_get_node(cmark_iter *iter) { return iter->cur.node; }
88
+
89
+ cmark_event_type cmark_iter_get_event_type(cmark_iter *iter) {
90
+ return iter->cur.ev_type;
91
+ }
92
+
93
+ cmark_node *cmark_iter_get_root(cmark_iter *iter) { return iter->root; }
94
+
95
+ void cmark_consolidate_text_nodes(cmark_node *root) {
96
+ if (root == NULL) {
97
+ return;
98
+ }
99
+ cmark_iter *iter = cmark_iter_new(root);
100
+ cmark_strbuf buf = CMARK_BUF_INIT(iter->mem);
101
+ cmark_event_type ev_type;
102
+ cmark_node *cur, *tmp, *next;
103
+
104
+ while ((ev_type = cmark_iter_next(iter)) != CMARK_EVENT_DONE) {
105
+ cur = cmark_iter_get_node(iter);
106
+ if (ev_type == CMARK_EVENT_ENTER && cur->type == CMARK_NODE_TEXT &&
107
+ cur->next && cur->next->type == CMARK_NODE_TEXT) {
108
+ cmark_strbuf_clear(&buf);
109
+ cmark_strbuf_put(&buf, cur->as.literal.data, cur->as.literal.len);
110
+ tmp = cur->next;
111
+ while (tmp && tmp->type == CMARK_NODE_TEXT) {
112
+ cmark_iter_next(iter); // advance pointer
113
+ cmark_strbuf_put(&buf, tmp->as.literal.data, tmp->as.literal.len);
114
+ cur->end_column = tmp->end_column;
115
+ next = tmp->next;
116
+ cmark_node_free(tmp);
117
+ tmp = next;
118
+ }
119
+ cmark_chunk_free(iter->mem, &cur->as.literal);
120
+ cur->as.literal = cmark_chunk_buf_detach(&buf);
121
+ }
122
+ }
123
+
124
+ cmark_strbuf_free(&buf);
125
+ cmark_iter_free(iter);
126
+ }
127
+
128
+ void cmark_node_own(cmark_node *root) {
129
+ if (root == NULL) {
130
+ return;
131
+ }
132
+ cmark_iter *iter = cmark_iter_new(root);
133
+ cmark_event_type ev_type;
134
+ cmark_node *cur;
135
+
136
+ while ((ev_type = cmark_iter_next(iter)) != CMARK_EVENT_DONE) {
137
+ cur = cmark_iter_get_node(iter);
138
+ if (ev_type == CMARK_EVENT_ENTER) {
139
+ switch (cur->type) {
140
+ case CMARK_NODE_TEXT:
141
+ case CMARK_NODE_HTML_INLINE:
142
+ case CMARK_NODE_CODE:
143
+ case CMARK_NODE_HTML_BLOCK:
144
+ cmark_chunk_to_cstr(iter->mem, &cur->as.literal);
145
+ break;
146
+ case CMARK_NODE_LINK:
147
+ cmark_chunk_to_cstr(iter->mem, &cur->as.link.url);
148
+ cmark_chunk_to_cstr(iter->mem, &cur->as.link.title);
149
+ break;
150
+ case CMARK_NODE_CUSTOM_INLINE:
151
+ cmark_chunk_to_cstr(iter->mem, &cur->as.custom.on_enter);
152
+ cmark_chunk_to_cstr(iter->mem, &cur->as.custom.on_exit);
153
+ break;
154
+ }
155
+ }
156
+ }
157
+
158
+ cmark_iter_free(iter);
159
+ }
@@ -0,0 +1,26 @@
1
+ #ifndef CMARK_ITERATOR_H
2
+ #define CMARK_ITERATOR_H
3
+
4
+ #ifdef __cplusplus
5
+ extern "C" {
6
+ #endif
7
+
8
+ #include "cmark-gfm.h"
9
+
10
+ typedef struct {
11
+ cmark_event_type ev_type;
12
+ cmark_node *node;
13
+ } cmark_iter_state;
14
+
15
+ struct cmark_iter {
16
+ cmark_mem *mem;
17
+ cmark_node *root;
18
+ cmark_iter_state cur;
19
+ cmark_iter_state next;
20
+ };
21
+
22
+ #ifdef __cplusplus
23
+ }
24
+ #endif
25
+
26
+ #endif
@@ -0,0 +1,466 @@
1
+ #include <stdlib.h>
2
+ #include <stdio.h>
3
+ #include <string.h>
4
+ #include <assert.h>
5
+
6
+ #include "config.h"
7
+ #include "cmark-gfm.h"
8
+ #include "node.h"
9
+ #include "buffer.h"
10
+ #include "utf8.h"
11
+ #include "scanners.h"
12
+ #include "render.h"
13
+ #include "syntax_extension.h"
14
+
15
+ #define OUT(s, wrap, escaping) renderer->out(renderer, node, s, wrap, escaping)
16
+ #define LIT(s) renderer->out(renderer, node, s, false, LITERAL)
17
+ #define CR() renderer->cr(renderer)
18
+ #define BLANKLINE() renderer->blankline(renderer)
19
+ #define LIST_NUMBER_STRING_SIZE 20
20
+
21
+ static CMARK_INLINE void outc(cmark_renderer *renderer, cmark_node *node,
22
+ cmark_escaping escape,
23
+ int32_t c, unsigned char nextc) {
24
+ if (escape == LITERAL) {
25
+ cmark_render_code_point(renderer, c);
26
+ return;
27
+ }
28
+
29
+ switch (c) {
30
+ case 123: // '{'
31
+ case 125: // '}'
32
+ case 35: // '#'
33
+ case 37: // '%'
34
+ case 38: // '&'
35
+ cmark_render_ascii(renderer, "\\");
36
+ cmark_render_code_point(renderer, c);
37
+ break;
38
+ case 36: // '$'
39
+ case 95: // '_'
40
+ if (escape == NORMAL) {
41
+ cmark_render_ascii(renderer, "\\");
42
+ }
43
+ cmark_render_code_point(renderer, c);
44
+ break;
45
+ case 45: // '-'
46
+ if (nextc == 45) { // prevent ligature
47
+ cmark_render_ascii(renderer, "-{}");
48
+ } else {
49
+ cmark_render_ascii(renderer, "-");
50
+ }
51
+ break;
52
+ case 126: // '~'
53
+ if (escape == NORMAL) {
54
+ cmark_render_ascii(renderer, "\\textasciitilde{}");
55
+ } else {
56
+ cmark_render_code_point(renderer, c);
57
+ }
58
+ break;
59
+ case 94: // '^'
60
+ cmark_render_ascii(renderer, "\\^{}");
61
+ break;
62
+ case 92: // '\\'
63
+ if (escape == URL) {
64
+ // / acts as path sep even on windows:
65
+ cmark_render_ascii(renderer, "/");
66
+ } else {
67
+ cmark_render_ascii(renderer, "\\textbackslash{}");
68
+ }
69
+ break;
70
+ case 124: // '|'
71
+ cmark_render_ascii(renderer, "\\textbar{}");
72
+ break;
73
+ case 60: // '<'
74
+ cmark_render_ascii(renderer, "\\textless{}");
75
+ break;
76
+ case 62: // '>'
77
+ cmark_render_ascii(renderer, "\\textgreater{}");
78
+ break;
79
+ case 91: // '['
80
+ case 93: // ']'
81
+ cmark_render_ascii(renderer, "{");
82
+ cmark_render_code_point(renderer, c);
83
+ cmark_render_ascii(renderer, "}");
84
+ break;
85
+ case 34: // '"'
86
+ cmark_render_ascii(renderer, "\\textquotedbl{}");
87
+ // requires \usepackage[T1]{fontenc}
88
+ break;
89
+ case 39: // '\''
90
+ cmark_render_ascii(renderer, "\\textquotesingle{}");
91
+ // requires \usepackage{textcomp}
92
+ break;
93
+ case 160: // nbsp
94
+ cmark_render_ascii(renderer, "~");
95
+ break;
96
+ case 8230: // hellip
97
+ cmark_render_ascii(renderer, "\\ldots{}");
98
+ break;
99
+ case 8216: // lsquo
100
+ if (escape == NORMAL) {
101
+ cmark_render_ascii(renderer, "`");
102
+ } else {
103
+ cmark_render_code_point(renderer, c);
104
+ }
105
+ break;
106
+ case 8217: // rsquo
107
+ if (escape == NORMAL) {
108
+ cmark_render_ascii(renderer, "\'");
109
+ } else {
110
+ cmark_render_code_point(renderer, c);
111
+ }
112
+ break;
113
+ case 8220: // ldquo
114
+ if (escape == NORMAL) {
115
+ cmark_render_ascii(renderer, "``");
116
+ } else {
117
+ cmark_render_code_point(renderer, c);
118
+ }
119
+ break;
120
+ case 8221: // rdquo
121
+ if (escape == NORMAL) {
122
+ cmark_render_ascii(renderer, "''");
123
+ } else {
124
+ cmark_render_code_point(renderer, c);
125
+ }
126
+ break;
127
+ case 8212: // emdash
128
+ if (escape == NORMAL) {
129
+ cmark_render_ascii(renderer, "---");
130
+ } else {
131
+ cmark_render_code_point(renderer, c);
132
+ }
133
+ break;
134
+ case 8211: // endash
135
+ if (escape == NORMAL) {
136
+ cmark_render_ascii(renderer, "--");
137
+ } else {
138
+ cmark_render_code_point(renderer, c);
139
+ }
140
+ break;
141
+ default:
142
+ cmark_render_code_point(renderer, c);
143
+ }
144
+ }
145
+
146
+ typedef enum {
147
+ NO_LINK,
148
+ URL_AUTOLINK,
149
+ EMAIL_AUTOLINK,
150
+ NORMAL_LINK,
151
+ INTERNAL_LINK
152
+ } link_type;
153
+
154
+ static link_type get_link_type(cmark_node *node) {
155
+ size_t title_len, url_len;
156
+ cmark_node *link_text;
157
+ char *realurl;
158
+ int realurllen;
159
+ bool isemail = false;
160
+
161
+ if (node->type != CMARK_NODE_LINK) {
162
+ return NO_LINK;
163
+ }
164
+
165
+ const char *url = cmark_node_get_url(node);
166
+ cmark_chunk url_chunk = cmark_chunk_literal(url);
167
+
168
+ if (url && *url == '#') {
169
+ return INTERNAL_LINK;
170
+ }
171
+
172
+ url_len = strlen(url);
173
+ if (url_len == 0 || scan_scheme(&url_chunk, 0) == 0) {
174
+ return NO_LINK;
175
+ }
176
+
177
+ const char *title = cmark_node_get_title(node);
178
+ title_len = strlen(title);
179
+ // if it has a title, we can't treat it as an autolink:
180
+ if (title_len == 0) {
181
+
182
+ link_text = node->first_child;
183
+ cmark_consolidate_text_nodes(link_text);
184
+
185
+ if (!link_text)
186
+ return NO_LINK;
187
+
188
+ realurl = (char *)url;
189
+ realurllen = (int)url_len;
190
+ if (strncmp(realurl, "mailto:", 7) == 0) {
191
+ realurl += 7;
192
+ realurllen -= 7;
193
+ isemail = true;
194
+ }
195
+ if (realurllen == link_text->as.literal.len &&
196
+ strncmp(realurl, (char *)link_text->as.literal.data,
197
+ link_text->as.literal.len) == 0) {
198
+ if (isemail) {
199
+ return EMAIL_AUTOLINK;
200
+ } else {
201
+ return URL_AUTOLINK;
202
+ }
203
+ }
204
+ }
205
+
206
+ return NORMAL_LINK;
207
+ }
208
+
209
+ static int S_get_enumlevel(cmark_node *node) {
210
+ int enumlevel = 0;
211
+ cmark_node *tmp = node;
212
+ while (tmp) {
213
+ if (tmp->type == CMARK_NODE_LIST &&
214
+ cmark_node_get_list_type(node) == CMARK_ORDERED_LIST) {
215
+ enumlevel++;
216
+ }
217
+ tmp = tmp->parent;
218
+ }
219
+ return enumlevel;
220
+ }
221
+
222
+ static int S_render_node(cmark_renderer *renderer, cmark_node *node,
223
+ cmark_event_type ev_type, int options) {
224
+ int list_number;
225
+ int enumlevel;
226
+ char list_number_string[LIST_NUMBER_STRING_SIZE];
227
+ bool entering = (ev_type == CMARK_EVENT_ENTER);
228
+ cmark_list_type list_type;
229
+ bool allow_wrap = renderer->width > 0 && !(CMARK_OPT_NOBREAKS & options);
230
+
231
+ if (node->extension && node->extension->latex_render_func) {
232
+ node->extension->latex_render_func(node->extension, renderer, node, ev_type, options);
233
+ return 1;
234
+ }
235
+
236
+ switch (node->type) {
237
+ case CMARK_NODE_DOCUMENT:
238
+ break;
239
+
240
+ case CMARK_NODE_BLOCK_QUOTE:
241
+ if (entering) {
242
+ LIT("\\begin{quote}");
243
+ CR();
244
+ } else {
245
+ LIT("\\end{quote}");
246
+ BLANKLINE();
247
+ }
248
+ break;
249
+
250
+ case CMARK_NODE_LIST:
251
+ list_type = cmark_node_get_list_type(node);
252
+ if (entering) {
253
+ LIT("\\begin{");
254
+ LIT(list_type == CMARK_ORDERED_LIST ? "enumerate" : "itemize");
255
+ LIT("}");
256
+ CR();
257
+ list_number = cmark_node_get_list_start(node);
258
+ if (list_number > 1) {
259
+ enumlevel = S_get_enumlevel(node);
260
+ // latex normally supports only five levels
261
+ if (enumlevel >= 1 && enumlevel <= 5) {
262
+ snprintf(list_number_string, LIST_NUMBER_STRING_SIZE, "%d",
263
+ list_number);
264
+ LIT("\\setcounter{enum");
265
+ switch (enumlevel) {
266
+ case 1: LIT("i"); break;
267
+ case 2: LIT("ii"); break;
268
+ case 3: LIT("iii"); break;
269
+ case 4: LIT("iv"); break;
270
+ case 5: LIT("v"); break;
271
+ default: LIT("i"); break;
272
+ }
273
+ LIT("}{");
274
+ OUT(list_number_string, false, NORMAL);
275
+ LIT("}");
276
+ }
277
+ CR();
278
+ }
279
+ } else {
280
+ LIT("\\end{");
281
+ LIT(list_type == CMARK_ORDERED_LIST ? "enumerate" : "itemize");
282
+ LIT("}");
283
+ BLANKLINE();
284
+ }
285
+ break;
286
+
287
+ case CMARK_NODE_ITEM:
288
+ if (entering) {
289
+ LIT("\\item ");
290
+ } else {
291
+ CR();
292
+ }
293
+ break;
294
+
295
+ case CMARK_NODE_HEADING:
296
+ if (entering) {
297
+ switch (cmark_node_get_heading_level(node)) {
298
+ case 1:
299
+ LIT("\\section");
300
+ break;
301
+ case 2:
302
+ LIT("\\subsection");
303
+ break;
304
+ case 3:
305
+ LIT("\\subsubsection");
306
+ break;
307
+ case 4:
308
+ LIT("\\paragraph");
309
+ break;
310
+ case 5:
311
+ LIT("\\subparagraph");
312
+ break;
313
+ }
314
+ LIT("{");
315
+ } else {
316
+ LIT("}");
317
+ BLANKLINE();
318
+ }
319
+ break;
320
+
321
+ case CMARK_NODE_CODE_BLOCK:
322
+ CR();
323
+ LIT("\\begin{verbatim}");
324
+ CR();
325
+ OUT(cmark_node_get_literal(node), false, LITERAL);
326
+ CR();
327
+ LIT("\\end{verbatim}");
328
+ BLANKLINE();
329
+ break;
330
+
331
+ case CMARK_NODE_HTML_BLOCK:
332
+ break;
333
+
334
+ case CMARK_NODE_CUSTOM_BLOCK:
335
+ CR();
336
+ OUT(entering ? cmark_node_get_on_enter(node) : cmark_node_get_on_exit(node),
337
+ false, LITERAL);
338
+ CR();
339
+ break;
340
+
341
+ case CMARK_NODE_THEMATIC_BREAK:
342
+ BLANKLINE();
343
+ LIT("\\begin{center}\\rule{0.5\\linewidth}{\\linethickness}\\end{center}");
344
+ BLANKLINE();
345
+ break;
346
+
347
+ case CMARK_NODE_PARAGRAPH:
348
+ if (!entering) {
349
+ BLANKLINE();
350
+ }
351
+ break;
352
+
353
+ case CMARK_NODE_TEXT:
354
+ OUT(cmark_node_get_literal(node), allow_wrap, NORMAL);
355
+ break;
356
+
357
+ case CMARK_NODE_LINEBREAK:
358
+ LIT("\\\\");
359
+ CR();
360
+ break;
361
+
362
+ case CMARK_NODE_SOFTBREAK:
363
+ if (options & CMARK_OPT_HARDBREAKS) {
364
+ LIT("\\\\");
365
+ CR();
366
+ } else if (renderer->width == 0 && !(CMARK_OPT_NOBREAKS & options)) {
367
+ CR();
368
+ } else {
369
+ OUT(" ", allow_wrap, NORMAL);
370
+ }
371
+ break;
372
+
373
+ case CMARK_NODE_CODE:
374
+ LIT("\\texttt{");
375
+ OUT(cmark_node_get_literal(node), false, NORMAL);
376
+ LIT("}");
377
+ break;
378
+
379
+ case CMARK_NODE_HTML_INLINE:
380
+ break;
381
+
382
+ case CMARK_NODE_CUSTOM_INLINE:
383
+ OUT(entering ? cmark_node_get_on_enter(node) : cmark_node_get_on_exit(node),
384
+ false, LITERAL);
385
+ break;
386
+
387
+ case CMARK_NODE_STRONG:
388
+ if (entering) {
389
+ LIT("\\textbf{");
390
+ } else {
391
+ LIT("}");
392
+ }
393
+ break;
394
+
395
+ case CMARK_NODE_EMPH:
396
+ if (entering) {
397
+ LIT("\\emph{");
398
+ } else {
399
+ LIT("}");
400
+ }
401
+ break;
402
+
403
+ case CMARK_NODE_LINK:
404
+ if (entering) {
405
+ const char *url = cmark_node_get_url(node);
406
+ // requires \usepackage{hyperref}
407
+ switch (get_link_type(node)) {
408
+ case URL_AUTOLINK:
409
+ LIT("\\url{");
410
+ OUT(url, false, URL);
411
+ LIT("}");
412
+ return 0; // Don't process further nodes to avoid double-rendering artefacts
413
+ case EMAIL_AUTOLINK:
414
+ LIT("\\href{");
415
+ OUT(url, false, URL);
416
+ LIT("}\\nolinkurl{");
417
+ break;
418
+ case NORMAL_LINK:
419
+ LIT("\\href{");
420
+ OUT(url, false, URL);
421
+ LIT("}{");
422
+ break;
423
+ case INTERNAL_LINK:
424
+ LIT("\\protect\\hyperlink{");
425
+ OUT(url + 1, false, URL);
426
+ LIT("}{");
427
+ break;
428
+ case NO_LINK:
429
+ LIT("{"); // error?
430
+ }
431
+ } else {
432
+ LIT("}");
433
+ }
434
+
435
+ break;
436
+
437
+ case CMARK_NODE_IMAGE:
438
+ if (entering) {
439
+ LIT("\\protect\\includegraphics{");
440
+ // requires \include{graphicx}
441
+ OUT(cmark_node_get_url(node), false, URL);
442
+ LIT("}");
443
+ return 0;
444
+ }
445
+ break;
446
+
447
+ case CMARK_NODE_FOOTNOTE_DEFINITION:
448
+ case CMARK_NODE_FOOTNOTE_REFERENCE:
449
+ // TODO
450
+ break;
451
+
452
+ default:
453
+ assert(false);
454
+ break;
455
+ }
456
+
457
+ return 1;
458
+ }
459
+
460
+ char *cmark_render_latex(cmark_node *root, int options, int width) {
461
+ return cmark_render_latex_with_mem(root, options, width, cmark_node_mem(root));
462
+ }
463
+
464
+ char *cmark_render_latex_with_mem(cmark_node *root, int options, int width, cmark_mem *mem) {
465
+ return cmark_render(mem, root, options, width, outc, S_render_node);
466
+ }