123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- /* copyright (c) 2022 - 2024 grunfink et al. / MIT license */
- #ifndef _XS_HTML_H
- #define _XS_HTML_H
- typedef struct xs_html xs_html;
- xs_str *xs_html_encode(const char *str);
- xs_html *xs_html_attr(const char *key, const char *value);
- xs_html *xs_html_text(const char *content);
- xs_html *xs_html_raw(const char *content);
- xs_html *_xs_html_add(xs_html *tag, xs_html *var[]);
- #define xs_html_add(tag, ...) _xs_html_add(tag, (xs_html *[]) { __VA_ARGS__, NULL })
- xs_html *_xs_html_tag(const char *tag, xs_html *var[]);
- #define xs_html_tag(tag, ...) _xs_html_tag(tag, (xs_html *[]) { __VA_ARGS__, NULL })
- xs_html *_xs_html_sctag(const char *tag, xs_html *var[]);
- #define xs_html_sctag(tag, ...) _xs_html_sctag(tag, (xs_html *[]) { __VA_ARGS__, NULL })
- xs_html *_xs_html_container(xs_html *var[]);
- #define xs_html_container(...) _xs_html_container((xs_html *[]) { __VA_ARGS__, NULL })
- void xs_html_render_f(xs_html *h, FILE *f);
- xs_str *xs_html_render_s(xs_html *tag, const char *prefix);
- #define xs_html_render(tag) xs_html_render_s(tag, NULL)
- #ifdef XS_IMPLEMENTATION
- typedef enum {
- XS_HTML_TAG,
- XS_HTML_SCTAG,
- XS_HTML_CONTAINER,
- XS_HTML_ATTR,
- XS_HTML_TEXT
- } xs_html_type;
- struct xs_html {
- xs_html_type type;
- xs_str *content;
- xs_html *attrs;
- xs_html *tags;
- xs_html *next;
- };
- xs_str *xs_html_encode(const char *str)
- /* encodes str using HTML entities */
- {
- xs_str *s = xs_str_new(NULL);
- int o = 0;
- const char *e = str + strlen(str);
- for (;;) {
- char *ec = "<>\"'&"; /* characters to escape */
- const char *q = e;
- int z;
- /* find the nearest happening of a char */
- while (*ec) {
- char *m = memchr(str, *ec++, q - str);
- if (m)
- q = m;
- }
- /* copy string to here */
- z = q - str;
- s = xs_insert_m(s, o, str, z);
- o += z;
- /* if q points to the end, nothing more to do */
- if (q == e)
- break;
- /* insert the escaped char */
- char tmp[8];
- z = snprintf(tmp, sizeof(tmp), "&#%d;", *q);
- s = xs_insert_m(s, o, tmp, z);
- o += z;
- str = q + 1;
- }
- return s;
- }
- #define XS_HTML_NEW() memset(xs_realloc(NULL, sizeof(xs_html)), '\0', sizeof(xs_html))
- xs_html *xs_html_attr(const char *key, const char *value)
- /* creates an HTML block with an attribute */
- {
- xs_html *a = XS_HTML_NEW();
- a->type = XS_HTML_ATTR;
- if (value) {
- xs *ev = xs_html_encode(value);
- a->content = xs_fmt("%s=\"%s\"", key, ev);
- }
- else
- a->content = xs_dup(key);
- return a;
- }
- xs_html *xs_html_text(const char *content)
- /* creates an HTML block of text, escaping it previously */
- {
- xs_html *a = XS_HTML_NEW();
- a->type = XS_HTML_TEXT;
- a->content = xs_html_encode(content);
- return a;
- }
- xs_html *xs_html_raw(const char *content)
- /* creates an HTML block without escaping (for pre-formatted HTML, comments, etc) */
- {
- xs_html *a = XS_HTML_NEW();
- a->type = XS_HTML_TEXT;
- a->content = xs_dup(content);
- return a;
- }
- xs_html *_xs_html_add(xs_html *tag, xs_html *var[])
- /* add data (attrs, tags or text) to a tag */
- {
- while (*var) {
- xs_html *data = *var++;
- if (data->type == XS_HTML_ATTR) {
- data->next = tag->attrs;
- tag->attrs = data;
- }
- else {
- data->next = tag->tags;
- tag->tags = data;
- }
- }
- return tag;
- }
- static xs_html *_xs_html_tag_t(xs_html_type type, const char *tag, xs_html *var[])
- /* creates a tag with a variable list of attributes and subtags */
- {
- xs_html *a = XS_HTML_NEW();
- a->type = type;
- /* a tag can be NULL, to be a kind of 'wrapper' */
- if (tag)
- a->content = xs_dup(tag);
- _xs_html_add(a, var);
- return a;
- }
- xs_html *_xs_html_tag(const char *tag, xs_html *var[])
- {
- return _xs_html_tag_t(XS_HTML_TAG, tag, var);
- }
- xs_html *_xs_html_sctag(const char *tag, xs_html *var[])
- {
- return _xs_html_tag_t(XS_HTML_SCTAG, tag, var);
- }
- xs_html *_xs_html_container(xs_html *var[])
- {
- return _xs_html_tag_t(XS_HTML_CONTAINER, NULL, var);
- }
- void xs_html_render_f(xs_html *h, FILE *f)
- /* renders the tag and its subtags into a file */
- {
- if (h == NULL)
- return;
- /* follow the chain */
- xs_html_render_f(h->next, f);
- switch (h->type) {
- case XS_HTML_TAG:
- fprintf(f, "<%s", h->content);
- /* attributes */
- xs_html_render_f(h->attrs, f);
- fprintf(f, ">");
- /* sub-tags */
- xs_html_render_f(h->tags, f);
- fprintf(f, "</%s>", h->content);
- break;
- case XS_HTML_SCTAG:
- fprintf(f, "<%s", h->content);
- /* attributes */
- xs_html_render_f(h->attrs, f);
- fprintf(f, "/>");
- break;
- case XS_HTML_CONTAINER:
- /* sub-tags */
- xs_html_render_f(h->tags, f);
- break;
- case XS_HTML_ATTR:
- fprintf(f, " ");
- /* fallthrough */
- case XS_HTML_TEXT:
- fprintf(f, "%s", h->content);
- break;
- }
- xs_free(h->content);
- xs_free(h);
- }
- xs_str *xs_html_render_s(xs_html *tag, const char *prefix)
- /* renders to a string */
- {
- xs_str *s = NULL;
- size_t sz;
- FILE *f;
- if ((f = open_memstream(&s, &sz)) != NULL) {
- if (prefix)
- fprintf(f, "%s", prefix);
- xs_html_render_f(tag, f);
- fclose(f);
- }
- return s;
- }
- #endif /* XS_IMPLEMENTATION */
- #endif /* _XS_HTML_H */
|