vkhater-redcarpet 2.2.3

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,77 @@
1
+ /*
2
+ * Copyright (c) 2011, Vicent Marti
3
+ *
4
+ * Permission to use, copy, modify, and distribute this software for any
5
+ * purpose with or without fee is hereby granted, provided that the above
6
+ * copyright notice and this permission notice appear in all copies.
7
+ *
8
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15
+ */
16
+
17
+ #ifndef UPSKIRT_HTML_H
18
+ #define UPSKIRT_HTML_H
19
+
20
+ #include "markdown.h"
21
+ #include "buffer.h"
22
+ #include <stdlib.h>
23
+
24
+ #ifdef __cplusplus
25
+ extern "C" {
26
+ #endif
27
+
28
+ struct html_renderopt {
29
+ struct {
30
+ int header_count;
31
+ int current_level;
32
+ int level_offset;
33
+ } toc_data;
34
+
35
+ unsigned int flags;
36
+
37
+ /* extra callbacks */
38
+ void (*link_attributes)(struct buf *ob, const struct buf *url, void *self);
39
+ };
40
+
41
+ typedef enum {
42
+ HTML_SKIP_HTML = (1 << 0),
43
+ HTML_SKIP_STYLE = (1 << 1),
44
+ HTML_SKIP_IMAGES = (1 << 2),
45
+ HTML_SKIP_LINKS = (1 << 3),
46
+ HTML_EXPAND_TABS = (1 << 4),
47
+ HTML_SAFELINK = (1 << 5),
48
+ HTML_TOC = (1 << 6),
49
+ HTML_HARD_WRAP = (1 << 7),
50
+ HTML_USE_XHTML = (1 << 8),
51
+ HTML_ESCAPE = (1 << 9),
52
+ } html_render_mode;
53
+
54
+ typedef enum {
55
+ HTML_TAG_NONE = 0,
56
+ HTML_TAG_OPEN,
57
+ HTML_TAG_CLOSE,
58
+ } html_tag;
59
+
60
+ int
61
+ sdhtml_is_tag(const uint8_t *tag_data, size_t tag_size, const char *tagname);
62
+
63
+ extern void
64
+ sdhtml_renderer(struct sd_callbacks *callbacks, struct html_renderopt *options_ptr, unsigned int render_flags);
65
+
66
+ extern void
67
+ sdhtml_toc_renderer(struct sd_callbacks *callbacks, struct html_renderopt *options_ptr);
68
+
69
+ extern void
70
+ sdhtml_smartypants(struct buf *ob, const uint8_t *text, size_t size);
71
+
72
+ #ifdef __cplusplus
73
+ }
74
+ #endif
75
+
76
+ #endif
77
+
@@ -0,0 +1,206 @@
1
+ /* C code produced by gperf version 3.0.3 */
2
+ /* Command-line: gperf -N find_block_tag -H hash_block_tag -C -c -E --ignore-case html_block_names.txt */
3
+ /* Computed positions: -k'1-2' */
4
+
5
+ #if !((' ' == 32) && ('!' == 33) && ('"' == 34) && ('#' == 35) \
6
+ && ('%' == 37) && ('&' == 38) && ('\'' == 39) && ('(' == 40) \
7
+ && (')' == 41) && ('*' == 42) && ('+' == 43) && (',' == 44) \
8
+ && ('-' == 45) && ('.' == 46) && ('/' == 47) && ('0' == 48) \
9
+ && ('1' == 49) && ('2' == 50) && ('3' == 51) && ('4' == 52) \
10
+ && ('5' == 53) && ('6' == 54) && ('7' == 55) && ('8' == 56) \
11
+ && ('9' == 57) && (':' == 58) && (';' == 59) && ('<' == 60) \
12
+ && ('=' == 61) && ('>' == 62) && ('?' == 63) && ('A' == 65) \
13
+ && ('B' == 66) && ('C' == 67) && ('D' == 68) && ('E' == 69) \
14
+ && ('F' == 70) && ('G' == 71) && ('H' == 72) && ('I' == 73) \
15
+ && ('J' == 74) && ('K' == 75) && ('L' == 76) && ('M' == 77) \
16
+ && ('N' == 78) && ('O' == 79) && ('P' == 80) && ('Q' == 81) \
17
+ && ('R' == 82) && ('S' == 83) && ('T' == 84) && ('U' == 85) \
18
+ && ('V' == 86) && ('W' == 87) && ('X' == 88) && ('Y' == 89) \
19
+ && ('Z' == 90) && ('[' == 91) && ('\\' == 92) && (']' == 93) \
20
+ && ('^' == 94) && ('_' == 95) && ('a' == 97) && ('b' == 98) \
21
+ && ('c' == 99) && ('d' == 100) && ('e' == 101) && ('f' == 102) \
22
+ && ('g' == 103) && ('h' == 104) && ('i' == 105) && ('j' == 106) \
23
+ && ('k' == 107) && ('l' == 108) && ('m' == 109) && ('n' == 110) \
24
+ && ('o' == 111) && ('p' == 112) && ('q' == 113) && ('r' == 114) \
25
+ && ('s' == 115) && ('t' == 116) && ('u' == 117) && ('v' == 118) \
26
+ && ('w' == 119) && ('x' == 120) && ('y' == 121) && ('z' == 122) \
27
+ && ('{' == 123) && ('|' == 124) && ('}' == 125) && ('~' == 126))
28
+ /* The character set is not based on ISO-646. */
29
+ error "gperf generated tables don't work with this execution character set. Please report a bug to <bug-gnu-gperf@gnu.org>."
30
+ #endif
31
+
32
+ /* maximum key range = 37, duplicates = 0 */
33
+
34
+ #ifndef GPERF_DOWNCASE
35
+ #define GPERF_DOWNCASE 1
36
+ static unsigned char gperf_downcase[256] =
37
+ {
38
+ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
39
+ 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29,
40
+ 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44,
41
+ 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59,
42
+ 60, 61, 62, 63, 64, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106,
43
+ 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121,
44
+ 122, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104,
45
+ 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119,
46
+ 120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130, 131, 132, 133, 134,
47
+ 135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 145, 146, 147, 148, 149,
48
+ 150, 151, 152, 153, 154, 155, 156, 157, 158, 159, 160, 161, 162, 163, 164,
49
+ 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176, 177, 178, 179,
50
+ 180, 181, 182, 183, 184, 185, 186, 187, 188, 189, 190, 191, 192, 193, 194,
51
+ 195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207, 208, 209,
52
+ 210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223, 224,
53
+ 225, 226, 227, 228, 229, 230, 231, 232, 233, 234, 235, 236, 237, 238, 239,
54
+ 240, 241, 242, 243, 244, 245, 246, 247, 248, 249, 250, 251, 252, 253, 254,
55
+ 255
56
+ };
57
+ #endif
58
+
59
+ #ifndef GPERF_CASE_STRNCMP
60
+ #define GPERF_CASE_STRNCMP 1
61
+ static int
62
+ gperf_case_strncmp (s1, s2, n)
63
+ register const char *s1;
64
+ register const char *s2;
65
+ register unsigned int n;
66
+ {
67
+ for (; n > 0;)
68
+ {
69
+ unsigned char c1 = gperf_downcase[(unsigned char)*s1++];
70
+ unsigned char c2 = gperf_downcase[(unsigned char)*s2++];
71
+ if (c1 != 0 && c1 == c2)
72
+ {
73
+ n--;
74
+ continue;
75
+ }
76
+ return (int)c1 - (int)c2;
77
+ }
78
+ return 0;
79
+ }
80
+ #endif
81
+
82
+ #ifdef __GNUC__
83
+ __inline
84
+ #else
85
+ #ifdef __cplusplus
86
+ inline
87
+ #endif
88
+ #endif
89
+ static unsigned int
90
+ hash_block_tag (str, len)
91
+ register const char *str;
92
+ register unsigned int len;
93
+ {
94
+ static const unsigned char asso_values[] =
95
+ {
96
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
97
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
98
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
99
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
100
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
101
+ 8, 30, 25, 20, 15, 10, 38, 38, 38, 38,
102
+ 38, 38, 38, 38, 38, 38, 0, 38, 0, 38,
103
+ 5, 5, 5, 15, 0, 38, 38, 0, 15, 10,
104
+ 0, 38, 38, 15, 0, 5, 38, 38, 38, 38,
105
+ 38, 38, 38, 38, 38, 38, 38, 38, 0, 38,
106
+ 0, 38, 5, 5, 5, 15, 0, 38, 38, 0,
107
+ 15, 10, 0, 38, 38, 15, 0, 5, 38, 38,
108
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
109
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
110
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
111
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
112
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
113
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
114
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
115
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
116
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
117
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
118
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
119
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
120
+ 38, 38, 38, 38, 38, 38, 38, 38, 38, 38,
121
+ 38, 38, 38, 38, 38, 38, 38
122
+ };
123
+ register int hval = len;
124
+
125
+ switch (hval)
126
+ {
127
+ default:
128
+ hval += asso_values[(unsigned char)str[1]+1];
129
+ /*FALLTHROUGH*/
130
+ case 1:
131
+ hval += asso_values[(unsigned char)str[0]];
132
+ break;
133
+ }
134
+ return hval;
135
+ }
136
+
137
+ #ifdef __GNUC__
138
+ __inline
139
+ #ifdef __GNUC_STDC_INLINE__
140
+ __attribute__ ((__gnu_inline__))
141
+ #endif
142
+ #endif
143
+ const char *
144
+ find_block_tag (str, len)
145
+ register const char *str;
146
+ register unsigned int len;
147
+ {
148
+ enum
149
+ {
150
+ TOTAL_KEYWORDS = 24,
151
+ MIN_WORD_LENGTH = 1,
152
+ MAX_WORD_LENGTH = 10,
153
+ MIN_HASH_VALUE = 1,
154
+ MAX_HASH_VALUE = 37
155
+ };
156
+
157
+ static const char * const wordlist[] =
158
+ {
159
+ "",
160
+ "p",
161
+ "dl",
162
+ "div",
163
+ "math",
164
+ "table",
165
+ "",
166
+ "ul",
167
+ "del",
168
+ "form",
169
+ "blockquote",
170
+ "figure",
171
+ "ol",
172
+ "fieldset",
173
+ "",
174
+ "h1",
175
+ "",
176
+ "h6",
177
+ "pre",
178
+ "", "",
179
+ "script",
180
+ "h5",
181
+ "noscript",
182
+ "",
183
+ "style",
184
+ "iframe",
185
+ "h4",
186
+ "ins",
187
+ "", "", "",
188
+ "h3",
189
+ "", "", "", "",
190
+ "h2"
191
+ };
192
+
193
+ if (len <= MAX_WORD_LENGTH && len >= MIN_WORD_LENGTH)
194
+ {
195
+ register int key = hash_block_tag (str, len);
196
+
197
+ if (key <= MAX_HASH_VALUE && key >= 0)
198
+ {
199
+ register const char *s = wordlist[key];
200
+
201
+ if ((((unsigned char)*str ^ (unsigned char)*s) & ~32) == 0 && !gperf_case_strncmp (str, s, len) && s[len] == '\0')
202
+ return s;
203
+ }
204
+ }
205
+ return 0;
206
+ }
@@ -0,0 +1,389 @@
1
+ /*
2
+ * Copyright (c) 2011, Vicent Marti
3
+ *
4
+ * Permission to use, copy, modify, and distribute this software for any
5
+ * purpose with or without fee is hereby granted, provided that the above
6
+ * copyright notice and this permission notice appear in all copies.
7
+ *
8
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15
+ */
16
+
17
+ #include "buffer.h"
18
+ #include "html.h"
19
+
20
+ #include <string.h>
21
+ #include <stdlib.h>
22
+ #include <stdio.h>
23
+ #include <ctype.h>
24
+
25
+ #if defined(_WIN32)
26
+ #define snprintf _snprintf
27
+ #endif
28
+
29
+ struct smartypants_data {
30
+ int in_squote;
31
+ int in_dquote;
32
+ };
33
+
34
+ static size_t smartypants_cb__ltag(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
35
+ static size_t smartypants_cb__dquote(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
36
+ static size_t smartypants_cb__amp(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
37
+ static size_t smartypants_cb__period(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
38
+ static size_t smartypants_cb__number(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
39
+ static size_t smartypants_cb__dash(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
40
+ static size_t smartypants_cb__parens(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
41
+ static size_t smartypants_cb__squote(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
42
+ static size_t smartypants_cb__backtick(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
43
+ static size_t smartypants_cb__escape(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size);
44
+
45
+ static size_t (*smartypants_cb_ptrs[])
46
+ (struct buf *, struct smartypants_data *, uint8_t, const uint8_t *, size_t) =
47
+ {
48
+ NULL, /* 0 */
49
+ smartypants_cb__dash, /* 1 */
50
+ smartypants_cb__parens, /* 2 */
51
+ smartypants_cb__squote, /* 3 */
52
+ smartypants_cb__dquote, /* 4 */
53
+ smartypants_cb__amp, /* 5 */
54
+ smartypants_cb__period, /* 6 */
55
+ smartypants_cb__number, /* 7 */
56
+ smartypants_cb__ltag, /* 8 */
57
+ smartypants_cb__backtick, /* 9 */
58
+ smartypants_cb__escape, /* 10 */
59
+ };
60
+
61
+ static const uint8_t smartypants_cb_chars[] = {
62
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
63
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
64
+ 0, 0, 4, 0, 0, 0, 5, 3, 2, 0, 0, 0, 0, 1, 6, 0,
65
+ 0, 7, 0, 7, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 0, 0,
66
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
67
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 10, 0, 0, 0,
68
+ 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
69
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
70
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
71
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
72
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
73
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
74
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
75
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
76
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
77
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
78
+ };
79
+
80
+ static inline int
81
+ word_boundary(uint8_t c)
82
+ {
83
+ return c == 0 || isspace(c) || ispunct(c);
84
+ }
85
+
86
+ static int
87
+ smartypants_quotes(struct buf *ob, uint8_t previous_char, uint8_t next_char, uint8_t quote, int *is_open)
88
+ {
89
+ char ent[8];
90
+
91
+ if (*is_open && !word_boundary(next_char))
92
+ return 0;
93
+
94
+ if (!(*is_open) && !word_boundary(previous_char))
95
+ return 0;
96
+
97
+ snprintf(ent, sizeof(ent), "&%c%cquo;", (*is_open) ? 'r' : 'l', quote);
98
+ *is_open = !(*is_open);
99
+ bufputs(ob, ent);
100
+ return 1;
101
+ }
102
+
103
+ static size_t
104
+ smartypants_cb__squote(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
105
+ {
106
+ if (size >= 2) {
107
+ uint8_t t1 = tolower(text[1]);
108
+
109
+ if (t1 == '\'') {
110
+ if (smartypants_quotes(ob, previous_char, size >= 3 ? text[2] : 0, 'd', &smrt->in_dquote))
111
+ return 1;
112
+ }
113
+
114
+ if ((t1 == 's' || t1 == 't' || t1 == 'm' || t1 == 'd') &&
115
+ (size == 3 || word_boundary(text[2]))) {
116
+ BUFPUTSL(ob, "&rsquo;");
117
+ return 0;
118
+ }
119
+
120
+ if (size >= 3) {
121
+ uint8_t t2 = tolower(text[2]);
122
+
123
+ if (((t1 == 'r' && t2 == 'e') ||
124
+ (t1 == 'l' && t2 == 'l') ||
125
+ (t1 == 'v' && t2 == 'e')) &&
126
+ (size == 4 || word_boundary(text[3]))) {
127
+ BUFPUTSL(ob, "&rsquo;");
128
+ return 0;
129
+ }
130
+ }
131
+ }
132
+
133
+ if (smartypants_quotes(ob, previous_char, size > 0 ? text[1] : 0, 's', &smrt->in_squote))
134
+ return 0;
135
+
136
+ bufputc(ob, text[0]);
137
+ return 0;
138
+ }
139
+
140
+ static size_t
141
+ smartypants_cb__parens(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
142
+ {
143
+ if (size >= 3) {
144
+ uint8_t t1 = tolower(text[1]);
145
+ uint8_t t2 = tolower(text[2]);
146
+
147
+ if (t1 == 'c' && t2 == ')') {
148
+ BUFPUTSL(ob, "&copy;");
149
+ return 2;
150
+ }
151
+
152
+ if (t1 == 'r' && t2 == ')') {
153
+ BUFPUTSL(ob, "&reg;");
154
+ return 2;
155
+ }
156
+
157
+ if (size >= 4 && t1 == 't' && t2 == 'm' && text[3] == ')') {
158
+ BUFPUTSL(ob, "&trade;");
159
+ return 3;
160
+ }
161
+ }
162
+
163
+ bufputc(ob, text[0]);
164
+ return 0;
165
+ }
166
+
167
+ static size_t
168
+ smartypants_cb__dash(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
169
+ {
170
+ if (size >= 3 && text[1] == '-' && text[2] == '-') {
171
+ BUFPUTSL(ob, "&mdash;");
172
+ return 2;
173
+ }
174
+
175
+ if (size >= 2 && text[1] == '-') {
176
+ BUFPUTSL(ob, "&ndash;");
177
+ return 1;
178
+ }
179
+
180
+ bufputc(ob, text[0]);
181
+ return 0;
182
+ }
183
+
184
+ static size_t
185
+ smartypants_cb__amp(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
186
+ {
187
+ if (size >= 6 && memcmp(text, "&quot;", 6) == 0) {
188
+ if (smartypants_quotes(ob, previous_char, size >= 7 ? text[6] : 0, 'd', &smrt->in_dquote))
189
+ return 5;
190
+ }
191
+
192
+ if (size >= 4 && memcmp(text, "&#0;", 4) == 0)
193
+ return 3;
194
+
195
+ bufputc(ob, '&');
196
+ return 0;
197
+ }
198
+
199
+ static size_t
200
+ smartypants_cb__period(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
201
+ {
202
+ if (size >= 3 && text[1] == '.' && text[2] == '.') {
203
+ BUFPUTSL(ob, "&hellip;");
204
+ return 2;
205
+ }
206
+
207
+ if (size >= 5 && text[1] == ' ' && text[2] == '.' && text[3] == ' ' && text[4] == '.') {
208
+ BUFPUTSL(ob, "&hellip;");
209
+ return 4;
210
+ }
211
+
212
+ bufputc(ob, text[0]);
213
+ return 0;
214
+ }
215
+
216
+ static size_t
217
+ smartypants_cb__backtick(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
218
+ {
219
+ if (size >= 2 && text[1] == '`') {
220
+ if (smartypants_quotes(ob, previous_char, size >= 3 ? text[2] : 0, 'd', &smrt->in_dquote))
221
+ return 1;
222
+ }
223
+
224
+ return 0;
225
+ }
226
+
227
+ static size_t
228
+ smartypants_cb__number(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
229
+ {
230
+ if (word_boundary(previous_char) && size >= 3) {
231
+ if (text[0] == '1' && text[1] == '/' && text[2] == '2') {
232
+ if (size == 3 || word_boundary(text[3])) {
233
+ BUFPUTSL(ob, "&frac12;");
234
+ return 2;
235
+ }
236
+ }
237
+
238
+ if (text[0] == '1' && text[1] == '/' && text[2] == '4') {
239
+ if (size == 3 || word_boundary(text[3]) ||
240
+ (size >= 5 && tolower(text[3]) == 't' && tolower(text[4]) == 'h')) {
241
+ BUFPUTSL(ob, "&frac14;");
242
+ return 2;
243
+ }
244
+ }
245
+
246
+ if (text[0] == '3' && text[1] == '/' && text[2] == '4') {
247
+ if (size == 3 || word_boundary(text[3]) ||
248
+ (size >= 6 && tolower(text[3]) == 't' && tolower(text[4]) == 'h' && tolower(text[5]) == 's')) {
249
+ BUFPUTSL(ob, "&frac34;");
250
+ return 2;
251
+ }
252
+ }
253
+ }
254
+
255
+ bufputc(ob, text[0]);
256
+ return 0;
257
+ }
258
+
259
+ static size_t
260
+ smartypants_cb__dquote(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
261
+ {
262
+ if (!smartypants_quotes(ob, previous_char, size > 0 ? text[1] : 0, 'd', &smrt->in_dquote))
263
+ BUFPUTSL(ob, "&quot;");
264
+
265
+ return 0;
266
+ }
267
+
268
+ static size_t
269
+ smartypants_cb__ltag(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
270
+ {
271
+ static const char *skip_tags[] = {
272
+ "pre", "code", "var", "samp", "kbd", "math", "script", "style"
273
+ };
274
+ static const size_t skip_tags_count = 8;
275
+
276
+ size_t tag, i = 0;
277
+
278
+ while (i < size && text[i] != '>')
279
+ i++;
280
+
281
+ for (tag = 0; tag < skip_tags_count; ++tag) {
282
+ if (sdhtml_is_tag(text, size, skip_tags[tag]) == HTML_TAG_OPEN)
283
+ break;
284
+ }
285
+
286
+ if (tag < skip_tags_count) {
287
+ for (;;) {
288
+ while (i < size && text[i] != '<')
289
+ i++;
290
+
291
+ if (i == size)
292
+ break;
293
+
294
+ if (sdhtml_is_tag(text + i, size - i, skip_tags[tag]) == HTML_TAG_CLOSE)
295
+ break;
296
+
297
+ i++;
298
+ }
299
+
300
+ while (i < size && text[i] != '>')
301
+ i++;
302
+ }
303
+
304
+ bufput(ob, text, i + 1);
305
+ return i;
306
+ }
307
+
308
+ static size_t
309
+ smartypants_cb__escape(struct buf *ob, struct smartypants_data *smrt, uint8_t previous_char, const uint8_t *text, size_t size)
310
+ {
311
+ if (size < 2)
312
+ return 0;
313
+
314
+ switch (text[1]) {
315
+ case '\\':
316
+ case '"':
317
+ case '\'':
318
+ case '.':
319
+ case '-':
320
+ case '`':
321
+ bufputc(ob, text[1]);
322
+ return 1;
323
+
324
+ default:
325
+ bufputc(ob, '\\');
326
+ return 0;
327
+ }
328
+ }
329
+
330
+ #if 0
331
+ static struct {
332
+ uint8_t c0;
333
+ const uint8_t *pattern;
334
+ const uint8_t *entity;
335
+ int skip;
336
+ } smartypants_subs[] = {
337
+ { '\'', "'s>", "&rsquo;", 0 },
338
+ { '\'', "'t>", "&rsquo;", 0 },
339
+ { '\'', "'re>", "&rsquo;", 0 },
340
+ { '\'', "'ll>", "&rsquo;", 0 },
341
+ { '\'', "'ve>", "&rsquo;", 0 },
342
+ { '\'', "'m>", "&rsquo;", 0 },
343
+ { '\'', "'d>", "&rsquo;", 0 },
344
+ { '-', "--", "&mdash;", 1 },
345
+ { '-', "<->", "&ndash;", 0 },
346
+ { '.', "...", "&hellip;", 2 },
347
+ { '.', ". . .", "&hellip;", 4 },
348
+ { '(', "(c)", "&copy;", 2 },
349
+ { '(', "(r)", "&reg;", 2 },
350
+ { '(', "(tm)", "&trade;", 3 },
351
+ { '3', "<3/4>", "&frac34;", 2 },
352
+ { '3', "<3/4ths>", "&frac34;", 2 },
353
+ { '1', "<1/2>", "&frac12;", 2 },
354
+ { '1', "<1/4>", "&frac14;", 2 },
355
+ { '1', "<1/4th>", "&frac14;", 2 },
356
+ { '&', "&#0;", 0, 3 },
357
+ };
358
+ #endif
359
+
360
+ void
361
+ sdhtml_smartypants(struct buf *ob, const uint8_t *text, size_t size)
362
+ {
363
+ size_t i;
364
+ struct smartypants_data smrt = {0, 0};
365
+
366
+ if (!text)
367
+ return;
368
+
369
+ bufgrow(ob, size);
370
+
371
+ for (i = 0; i < size; ++i) {
372
+ size_t org;
373
+ uint8_t action = 0;
374
+
375
+ org = i;
376
+ while (i < size && (action = smartypants_cb_chars[text[i]]) == 0)
377
+ i++;
378
+
379
+ if (i > org)
380
+ bufput(ob, text + org, i - org);
381
+
382
+ if (i < size) {
383
+ i += smartypants_cb_ptrs[(int)action]
384
+ (ob, &smrt, i ? text[i - 1] : 0, text + i, size - i);
385
+ }
386
+ }
387
+ }
388
+
389
+