mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-18 02:47:12 -05:00
1122 lines
37 KiB
C
1122 lines
37 KiB
C
|
/**********************************************************************
|
||
|
|
||
|
markdown_output.c - functions for printing Elements parsed by
|
||
|
markdown_peg.
|
||
|
(c) 2008 John MacFarlane (jgm at berkeley dot edu).
|
||
|
|
||
|
This program is free software; you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License or the MIT
|
||
|
license. See LICENSE for details.
|
||
|
|
||
|
This program is distributed in the hope that it will be useful,
|
||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
GNU General Public License for more details.
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
#include <stdbool.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <assert.h>
|
||
|
#include <glib.h>
|
||
|
#include "markdown_peg.h"
|
||
|
#include "odf.h"
|
||
|
|
||
|
static int extensions;
|
||
|
static int odf_type = 0;
|
||
|
|
||
|
static void print_html_string(GString *out, char *str, bool obfuscate);
|
||
|
static void print_html_element_list(GString *out, element *list, bool obfuscate);
|
||
|
static void print_html_element(GString *out, element *elt, bool obfuscate);
|
||
|
static void print_latex_string(GString *out, char *str);
|
||
|
static void print_latex_element_list(GString *out, element *list);
|
||
|
static void print_latex_element(GString *out, element *elt);
|
||
|
static void print_groff_string(GString *out, char *str);
|
||
|
static void print_groff_mm_element_list(GString *out, element *list);
|
||
|
static void print_groff_mm_element(GString *out, element *elt, int count);
|
||
|
static void print_odf_code_string(GString *out, char *str);
|
||
|
static void print_odf_string(GString *out, char *str);
|
||
|
static void print_odf_element_list(GString *out, element *list);
|
||
|
static void print_odf_element(GString *out, element *elt);
|
||
|
static bool list_contains_key(element *list, int key);
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Utility functions for printing
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
static int padded = 2; /* Number of newlines after last output.
|
||
|
Starts at 2 so no newlines are needed at start.
|
||
|
*/
|
||
|
|
||
|
static GSList *endnotes = NULL; /* List of endnotes to print after main content. */
|
||
|
static int notenumber = 0; /* Number of footnote. */
|
||
|
|
||
|
/* pad - add newlines if needed */
|
||
|
static void pad(GString *out, int num) {
|
||
|
while (num-- > padded)
|
||
|
g_string_append_printf(out, "\n");;
|
||
|
padded = num;
|
||
|
}
|
||
|
|
||
|
/* determine whether a certain element is contained within a given list */
|
||
|
static bool list_contains_key(element *list, int key) {
|
||
|
element *step = NULL;
|
||
|
|
||
|
step = list;
|
||
|
while ( step != NULL ) {
|
||
|
if (step->key == key) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
if (step->children != NULL) {
|
||
|
if (list_contains_key(step->children, key)) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
}
|
||
|
step = step->next;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Functions for printing Elements as HTML
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
/* print_html_string - print string, escaping for HTML
|
||
|
* If obfuscate selected, convert characters to hex or decimal entities at random */
|
||
|
static void print_html_string(GString *out, char *str, bool obfuscate) {
|
||
|
while (*str != '\0') {
|
||
|
switch (*str) {
|
||
|
case '&':
|
||
|
g_string_append_printf(out, "&");
|
||
|
break;
|
||
|
case '<':
|
||
|
g_string_append_printf(out, "<");
|
||
|
break;
|
||
|
case '>':
|
||
|
g_string_append_printf(out, ">");
|
||
|
break;
|
||
|
case '"':
|
||
|
g_string_append_printf(out, """);
|
||
|
break;
|
||
|
default:
|
||
|
if (obfuscate) {
|
||
|
if (rand() % 2 == 0)
|
||
|
g_string_append_printf(out, "&#%d;", (int) *str);
|
||
|
else
|
||
|
g_string_append_printf(out, "&#x%x;", (unsigned int) *str);
|
||
|
}
|
||
|
else
|
||
|
g_string_append_c(out, *str);
|
||
|
}
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_html_element_list - print a list of elements as HTML */
|
||
|
static void print_html_element_list(GString *out, element *list, bool obfuscate) {
|
||
|
while (list != NULL) {
|
||
|
print_html_element(out, list, obfuscate);
|
||
|
list = list->next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* add_endnote - add an endnote to global endnotes list. */
|
||
|
static void add_endnote(element *elt) {
|
||
|
endnotes = g_slist_prepend(endnotes, elt);
|
||
|
}
|
||
|
|
||
|
/* print_html_element - print an element as HTML */
|
||
|
static void print_html_element(GString *out, element *elt, bool obfuscate) {
|
||
|
int lev;
|
||
|
switch (elt->key) {
|
||
|
case SPACE:
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
break;
|
||
|
case LINEBREAK:
|
||
|
g_string_append_printf(out, "<br/>\n");
|
||
|
break;
|
||
|
case STR:
|
||
|
print_html_string(out, elt->contents.str, obfuscate);
|
||
|
break;
|
||
|
case ELLIPSIS:
|
||
|
g_string_append_printf(out, "…");
|
||
|
break;
|
||
|
case EMDASH:
|
||
|
g_string_append_printf(out, "—");
|
||
|
break;
|
||
|
case ENDASH:
|
||
|
g_string_append_printf(out, "–");
|
||
|
break;
|
||
|
case APOSTROPHE:
|
||
|
g_string_append_printf(out, "’");
|
||
|
break;
|
||
|
case SINGLEQUOTED:
|
||
|
g_string_append_printf(out, "‘");
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "’");
|
||
|
break;
|
||
|
case DOUBLEQUOTED:
|
||
|
g_string_append_printf(out, "“");
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "”");
|
||
|
break;
|
||
|
case CODE:
|
||
|
g_string_append_printf(out, "<code>");
|
||
|
print_html_string(out, elt->contents.str, obfuscate);
|
||
|
g_string_append_printf(out, "</code>");
|
||
|
break;
|
||
|
case HTML:
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
break;
|
||
|
case LINK:
|
||
|
if (strstr(elt->contents.link->url, "mailto:") == elt->contents.link->url)
|
||
|
obfuscate = true; /* obfuscate mailto: links */
|
||
|
g_string_append_printf(out, "<a href=\"");
|
||
|
print_html_string(out, elt->contents.link->url, obfuscate);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
if (strlen(elt->contents.link->title) > 0) {
|
||
|
g_string_append_printf(out, " title=\"");
|
||
|
print_html_string(out, elt->contents.link->title, obfuscate);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
}
|
||
|
g_string_append_printf(out, ">");
|
||
|
print_html_element_list(out, elt->contents.link->label, obfuscate);
|
||
|
g_string_append_printf(out, "</a>");
|
||
|
break;
|
||
|
case IMAGE:
|
||
|
g_string_append_printf(out, "<img src=\"");
|
||
|
print_html_string(out, elt->contents.link->url, obfuscate);
|
||
|
g_string_append_printf(out, "\" alt=\"");
|
||
|
print_html_element_list(out, elt->contents.link->label, obfuscate);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
if (strlen(elt->contents.link->title) > 0) {
|
||
|
g_string_append_printf(out, " title=\"");
|
||
|
print_html_string(out, elt->contents.link->title, obfuscate);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
}
|
||
|
g_string_append_printf(out, " />");
|
||
|
break;
|
||
|
case EMPH:
|
||
|
g_string_append_printf(out, "<em>");
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "</em>");
|
||
|
break;
|
||
|
case STRONG:
|
||
|
g_string_append_printf(out, "<strong>");
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "</strong>");
|
||
|
break;
|
||
|
case LIST:
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
break;
|
||
|
case RAW:
|
||
|
/* Shouldn't occur - these are handled by process_raw_blocks() */
|
||
|
assert(elt->key != RAW);
|
||
|
break;
|
||
|
case H1: case H2: case H3: case H4: case H5: case H6:
|
||
|
lev = elt->key - H1 + 1; /* assumes H1 ... H6 are in order */
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "<h%1d>", lev);
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "</h%1d>", lev);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PLAIN:
|
||
|
pad(out, 1);
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PARA:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "<p>");
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "</p>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HRULE:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "<hr />");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HTMLBLOCK:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case VERBATIM:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "%s", "<pre><code>");
|
||
|
print_html_string(out, elt->contents.str, obfuscate);
|
||
|
g_string_append_printf(out, "%s", "</code></pre>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case BULLETLIST:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "%s", "<ul>");
|
||
|
padded = 0;
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "%s", "</ul>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case ORDEREDLIST:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "%s", "<ol>");
|
||
|
padded = 0;
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "</ol>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case LISTITEM:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "<li>");
|
||
|
padded = 2;
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
g_string_append_printf(out, "</li>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case BLOCKQUOTE:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "<blockquote>\n");
|
||
|
padded = 2;
|
||
|
print_html_element_list(out, elt->children, obfuscate);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "</blockquote>");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case REFERENCE:
|
||
|
/* Nonprinting */
|
||
|
break;
|
||
|
case NOTE:
|
||
|
/* if contents.str == 0, then print note; else ignore, since this
|
||
|
* is a note block that has been incorporated into the notes list */
|
||
|
if (elt->contents.str == 0) {
|
||
|
add_endnote(elt);
|
||
|
++notenumber;
|
||
|
g_string_append_printf(out, "<a class=\"noteref\" id=\"fnref%d\" href=\"#fn%d\" title=\"Jump to note %d\">[%d]</a>",
|
||
|
notenumber, notenumber, notenumber, notenumber);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
fprintf(stderr, "print_html_element encountered unknown element key = %d\n", elt->key);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void print_html_endnotes(GString *out) {
|
||
|
int counter = 0;
|
||
|
GSList *note;
|
||
|
element *note_elt;
|
||
|
if (endnotes == NULL)
|
||
|
return;
|
||
|
note = g_slist_reverse(endnotes);
|
||
|
g_string_append_printf(out, "<hr/>\n<ol id=\"notes\">");
|
||
|
while (note != NULL) {
|
||
|
note_elt = note->data;
|
||
|
counter++;
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "<li id=\"fn%d\">\n", counter);
|
||
|
padded = 2;
|
||
|
print_html_element_list(out, note_elt->children, false);
|
||
|
g_string_append_printf(out, " <a href=\"#fnref%d\" title=\"Jump back to reference\">[back]</a>", counter);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "</li>");
|
||
|
note = note->next;
|
||
|
}
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "</ol>");
|
||
|
g_slist_free(endnotes);
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Functions for printing Elements as LaTeX
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
/* print_latex_string - print string, escaping for LaTeX */
|
||
|
static void print_latex_string(GString *out, char *str) {
|
||
|
while (*str != '\0') {
|
||
|
switch (*str) {
|
||
|
case '{': case '}': case '$': case '%':
|
||
|
case '&': case '_': case '#':
|
||
|
g_string_append_printf(out, "\\%c", *str);
|
||
|
break;
|
||
|
case '^':
|
||
|
g_string_append_printf(out, "\\^{}");
|
||
|
break;
|
||
|
case '\\':
|
||
|
g_string_append_printf(out, "\\textbackslash{}");
|
||
|
break;
|
||
|
case '~':
|
||
|
g_string_append_printf(out, "\\ensuremath{\\sim}");
|
||
|
break;
|
||
|
case '|':
|
||
|
g_string_append_printf(out, "\\textbar{}");
|
||
|
break;
|
||
|
case '<':
|
||
|
g_string_append_printf(out, "\\textless{}");
|
||
|
break;
|
||
|
case '>':
|
||
|
g_string_append_printf(out, "\\textgreater{}");
|
||
|
break;
|
||
|
default:
|
||
|
g_string_append_c(out, *str);
|
||
|
}
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_latex_element_list - print a list of elements as LaTeX */
|
||
|
static void print_latex_element_list(GString *out, element *list) {
|
||
|
while (list != NULL) {
|
||
|
print_latex_element(out, list);
|
||
|
list = list->next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_latex_element - print an element as LaTeX */
|
||
|
static void print_latex_element(GString *out, element *elt) {
|
||
|
int lev;
|
||
|
int i;
|
||
|
switch (elt->key) {
|
||
|
case SPACE:
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
break;
|
||
|
case LINEBREAK:
|
||
|
g_string_append_printf(out, "\\\\\n");
|
||
|
break;
|
||
|
case STR:
|
||
|
print_latex_string(out, elt->contents.str);
|
||
|
break;
|
||
|
case ELLIPSIS:
|
||
|
g_string_append_printf(out, "\\ldots{}");
|
||
|
break;
|
||
|
case EMDASH:
|
||
|
g_string_append_printf(out, "---");
|
||
|
break;
|
||
|
case ENDASH:
|
||
|
g_string_append_printf(out, "--");
|
||
|
break;
|
||
|
case APOSTROPHE:
|
||
|
g_string_append_printf(out, "'");
|
||
|
break;
|
||
|
case SINGLEQUOTED:
|
||
|
g_string_append_printf(out, "`");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "'");
|
||
|
break;
|
||
|
case DOUBLEQUOTED:
|
||
|
g_string_append_printf(out, "``");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "''");
|
||
|
break;
|
||
|
case CODE:
|
||
|
g_string_append_printf(out, "\\texttt{");
|
||
|
print_latex_string(out, elt->contents.str);
|
||
|
g_string_append_printf(out, "}");
|
||
|
break;
|
||
|
case HTML:
|
||
|
/* don't print HTML */
|
||
|
break;
|
||
|
case LINK:
|
||
|
g_string_append_printf(out, "\\href{%s}{", elt->contents.link->url);
|
||
|
print_latex_element_list(out, elt->contents.link->label);
|
||
|
g_string_append_printf(out, "}");
|
||
|
break;
|
||
|
case IMAGE:
|
||
|
g_string_append_printf(out, "\\includegraphics{%s}", elt->contents.link->url);
|
||
|
break;
|
||
|
case EMPH:
|
||
|
g_string_append_printf(out, "\\emph{");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "}");
|
||
|
break;
|
||
|
case STRONG:
|
||
|
g_string_append_printf(out, "\\textbf{");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "}");
|
||
|
break;
|
||
|
case LIST:
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
break;
|
||
|
case RAW:
|
||
|
/* Shouldn't occur - these are handled by process_raw_blocks() */
|
||
|
assert(elt->key != RAW);
|
||
|
break;
|
||
|
case H1: case H2: case H3:
|
||
|
pad(out, 2);
|
||
|
lev = elt->key - H1 + 1; /* assumes H1 ... H6 are in order */
|
||
|
g_string_append_printf(out, "\\");
|
||
|
for (i = elt->key; i > H1; i--)
|
||
|
g_string_append_printf(out, "sub");
|
||
|
g_string_append_printf(out, "section{");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case H4: case H5: case H6:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "\\noindent\\textbf{");
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PLAIN:
|
||
|
pad(out, 1);
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PARA:
|
||
|
pad(out, 2);
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HRULE:
|
||
|
pad(out, 2);
|
||
|
g_string_append_printf(out, "\\begin{center}\\rule{3in}{0.4pt}\\end{center}\n");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HTMLBLOCK:
|
||
|
/* don't print HTML block */
|
||
|
break;
|
||
|
case VERBATIM:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\begin{verbatim}\n");
|
||
|
print_latex_string(out, elt->contents.str);
|
||
|
g_string_append_printf(out, "\n\\end{verbatim}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case BULLETLIST:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\begin{itemize}");
|
||
|
padded = 0;
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\end{itemize}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case ORDEREDLIST:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\begin{enumerate}");
|
||
|
padded = 0;
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\end{enumerate}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case LISTITEM:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\item ");
|
||
|
padded = 2;
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "\n");
|
||
|
break;
|
||
|
case BLOCKQUOTE:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\begin{quote}");
|
||
|
padded = 0;
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\end{quote}");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case NOTE:
|
||
|
/* if contents.str == 0, then print note; else ignore, since this
|
||
|
* is a note block that has been incorporated into the notes list */
|
||
|
if (elt->contents.str == 0) {
|
||
|
g_string_append_printf(out, "\\footnote{");
|
||
|
padded = 2;
|
||
|
print_latex_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "}");
|
||
|
padded = 0;
|
||
|
}
|
||
|
break;
|
||
|
case REFERENCE:
|
||
|
/* Nonprinting */
|
||
|
break;
|
||
|
default:
|
||
|
fprintf(stderr, "print_latex_element encountered unknown element key = %d\n", elt->key);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Functions for printing Elements as groff (mm macros)
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
static bool in_list_item = false; /* True if we're parsing contents of a list item. */
|
||
|
|
||
|
/* print_groff_string - print string, escaping for groff */
|
||
|
static void print_groff_string(GString *out, char *str) {
|
||
|
while (*str != '\0') {
|
||
|
switch (*str) {
|
||
|
case '\\':
|
||
|
g_string_append_printf(out, "\\e");
|
||
|
break;
|
||
|
default:
|
||
|
g_string_append_c(out, *str);
|
||
|
}
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_groff_mm_element_list - print a list of elements as groff ms */
|
||
|
static void print_groff_mm_element_list(GString *out, element *list) {
|
||
|
int count = 1;
|
||
|
while (list != NULL) {
|
||
|
print_groff_mm_element(out, list, count);
|
||
|
list = list->next;
|
||
|
count++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_groff_mm_element - print an element as groff ms */
|
||
|
static void print_groff_mm_element(GString *out, element *elt, int count) {
|
||
|
int lev;
|
||
|
switch (elt->key) {
|
||
|
case SPACE:
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case LINEBREAK:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".br\n");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case STR:
|
||
|
print_groff_string(out, elt->contents.str);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case ELLIPSIS:
|
||
|
g_string_append_printf(out, "...");
|
||
|
break;
|
||
|
case EMDASH:
|
||
|
g_string_append_printf(out, "\\[em]");
|
||
|
break;
|
||
|
case ENDASH:
|
||
|
g_string_append_printf(out, "\\[en]");
|
||
|
break;
|
||
|
case APOSTROPHE:
|
||
|
g_string_append_printf(out, "'");
|
||
|
break;
|
||
|
case SINGLEQUOTED:
|
||
|
g_string_append_printf(out, "`");
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "'");
|
||
|
break;
|
||
|
case DOUBLEQUOTED:
|
||
|
g_string_append_printf(out, "\\[lq]");
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "\\[rq]");
|
||
|
break;
|
||
|
case CODE:
|
||
|
g_string_append_printf(out, "\\fC");
|
||
|
print_groff_string(out, elt->contents.str);
|
||
|
g_string_append_printf(out, "\\fR");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HTML:
|
||
|
/* don't print HTML */
|
||
|
break;
|
||
|
case LINK:
|
||
|
print_groff_mm_element_list(out, elt->contents.link->label);
|
||
|
g_string_append_printf(out, " (%s)", elt->contents.link->url);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case IMAGE:
|
||
|
g_string_append_printf(out, "[IMAGE: ");
|
||
|
print_groff_mm_element_list(out, elt->contents.link->label);
|
||
|
g_string_append_printf(out, "]");
|
||
|
padded = 0;
|
||
|
/* not supported */
|
||
|
break;
|
||
|
case EMPH:
|
||
|
g_string_append_printf(out, "\\fI");
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "\\fR");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case STRONG:
|
||
|
g_string_append_printf(out, "\\fB");
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "\\fR");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case LIST:
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case RAW:
|
||
|
/* Shouldn't occur - these are handled by process_raw_blocks() */
|
||
|
assert(elt->key != RAW);
|
||
|
break;
|
||
|
case H1: case H2: case H3: case H4: case H5: case H6:
|
||
|
lev = elt->key - H1 + 1;
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".H %d \"", lev);
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PLAIN:
|
||
|
pad(out, 1);
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PARA:
|
||
|
pad(out, 1);
|
||
|
if (!in_list_item || count != 1)
|
||
|
g_string_append_printf(out, ".P\n");
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HRULE:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, "\\l'\\n(.lu*8u/10u'");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case HTMLBLOCK:
|
||
|
/* don't print HTML block */
|
||
|
break;
|
||
|
case VERBATIM:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".VERBON 2\n");
|
||
|
print_groff_string(out, elt->contents.str);
|
||
|
g_string_append_printf(out, ".VERBOFF");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case BULLETLIST:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".BL");
|
||
|
padded = 0;
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".LE 1");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case ORDEREDLIST:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".AL");
|
||
|
padded = 0;
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".LE 1");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case LISTITEM:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".LI\n");
|
||
|
in_list_item = true;
|
||
|
padded = 2;
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
in_list_item = false;
|
||
|
break;
|
||
|
case BLOCKQUOTE:
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".DS I\n");
|
||
|
padded = 2;
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".DE");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case NOTE:
|
||
|
/* if contents.str == 0, then print note; else ignore, since this
|
||
|
* is a note block that has been incorporated into the notes list */
|
||
|
if (elt->contents.str == 0) {
|
||
|
g_string_append_printf(out, "\\*F\n");
|
||
|
g_string_append_printf(out, ".FS\n");
|
||
|
padded = 2;
|
||
|
print_groff_mm_element_list(out, elt->children);
|
||
|
pad(out, 1);
|
||
|
g_string_append_printf(out, ".FE\n");
|
||
|
padded = 1;
|
||
|
}
|
||
|
break;
|
||
|
case REFERENCE:
|
||
|
/* Nonprinting */
|
||
|
break;
|
||
|
default:
|
||
|
fprintf(stderr, "print_groff_mm_element encountered unknown element key = %d\n", elt->key);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Functions for printing Elements as ODF
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
/* print_odf_code_string - print string, escaping for HTML and saving newlines
|
||
|
*/
|
||
|
static void print_odf_code_string(GString *out, char *str) {
|
||
|
char *tmp;
|
||
|
while (*str != '\0') {
|
||
|
switch (*str) {
|
||
|
case '&':
|
||
|
g_string_append_printf(out, "&");
|
||
|
break;
|
||
|
case '<':
|
||
|
g_string_append_printf(out, "<");
|
||
|
break;
|
||
|
case '>':
|
||
|
g_string_append_printf(out, ">");
|
||
|
break;
|
||
|
case '"':
|
||
|
g_string_append_printf(out, """);
|
||
|
break;
|
||
|
case '\n':
|
||
|
g_string_append_printf(out, "<text:line-break/>");
|
||
|
break;
|
||
|
case ' ':
|
||
|
tmp = str;
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
g_string_append_printf(out, "<text:tab/>");
|
||
|
str = tmp;
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
g_string_append_c(out, *str);
|
||
|
}
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_odf_string - print string, escaping for HTML and saving newlines */
|
||
|
static void print_odf_string(GString *out, char *str) {
|
||
|
char *tmp;
|
||
|
while (*str != '\0') {
|
||
|
switch (*str) {
|
||
|
case '&':
|
||
|
g_string_append_printf(out, "&");
|
||
|
break;
|
||
|
case '<':
|
||
|
g_string_append_printf(out, "<");
|
||
|
break;
|
||
|
case '>':
|
||
|
g_string_append_printf(out, ">");
|
||
|
break;
|
||
|
case '"':
|
||
|
g_string_append_printf(out, """);
|
||
|
break;
|
||
|
case '\n':
|
||
|
tmp = str;
|
||
|
tmp--;
|
||
|
if (*tmp == ' ') {
|
||
|
tmp--;
|
||
|
if (*tmp == ' ') {
|
||
|
g_string_append_printf(out, "<text:line-break/>");
|
||
|
} else {
|
||
|
g_string_append_printf(out, "\n");
|
||
|
}
|
||
|
} else {
|
||
|
g_string_append_printf(out, "\n");
|
||
|
}
|
||
|
break;
|
||
|
case ' ':
|
||
|
tmp = str;
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
tmp++;
|
||
|
if (*tmp == ' ') {
|
||
|
g_string_append_printf(out, "<text:tab/>");
|
||
|
str = tmp;
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
} else {
|
||
|
g_string_append_printf(out, " ");
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
g_string_append_c(out, *str);
|
||
|
}
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_odf_element_list - print an element list as ODF */
|
||
|
static void print_odf_element_list(GString *out, element *list) {
|
||
|
while (list != NULL) {
|
||
|
print_odf_element(out, list);
|
||
|
list = list->next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* print_odf_element - print an element as ODF */
|
||
|
static void print_odf_element(GString *out, element *elt) {
|
||
|
int lev;
|
||
|
int old_type = 0;
|
||
|
switch (elt->key) {
|
||
|
case SPACE:
|
||
|
g_string_append_printf(out, "%s", elt->contents.str);
|
||
|
break;
|
||
|
case LINEBREAK:
|
||
|
g_string_append_printf(out, "<text:line-break/>");
|
||
|
break;
|
||
|
case STR:
|
||
|
print_html_string(out, elt->contents.str, 0);
|
||
|
break;
|
||
|
case ELLIPSIS:
|
||
|
g_string_append_printf(out, "…");
|
||
|
break;
|
||
|
case EMDASH:
|
||
|
g_string_append_printf(out, "—");
|
||
|
break;
|
||
|
case ENDASH:
|
||
|
g_string_append_printf(out, "–");
|
||
|
break;
|
||
|
case APOSTROPHE:
|
||
|
g_string_append_printf(out, "’");
|
||
|
break;
|
||
|
case SINGLEQUOTED:
|
||
|
g_string_append_printf(out, "‘");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "’");
|
||
|
break;
|
||
|
case DOUBLEQUOTED:
|
||
|
g_string_append_printf(out, "“");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "”");
|
||
|
break;
|
||
|
case CODE:
|
||
|
g_string_append_printf(out, "<text:span text:style-name=\"Source_20_Text\">");
|
||
|
print_html_string(out, elt->contents.str, 0);
|
||
|
g_string_append_printf(out, "</text:span>");
|
||
|
break;
|
||
|
case HTML:
|
||
|
break;
|
||
|
case LINK:
|
||
|
g_string_append_printf(out, "<text:a xlink:type=\"simple\" xlink:href=\"");
|
||
|
print_html_string(out, elt->contents.link->url, 0);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
if (strlen(elt->contents.link->title) > 0) {
|
||
|
g_string_append_printf(out, " office:name=\"");
|
||
|
print_html_string(out, elt->contents.link->title, 0);
|
||
|
g_string_append_printf(out, "\"");
|
||
|
}
|
||
|
g_string_append_printf(out, ">");
|
||
|
print_odf_element_list(out, elt->contents.link->label);
|
||
|
g_string_append_printf(out, "</text:a>");
|
||
|
break;
|
||
|
case IMAGE:
|
||
|
g_string_append_printf(out, "<draw:frame text:anchor-type=\"as-char\"\ndraw:z-index=\"0\" draw:style-name=\"fr1\" svg:width=\"95%%\"");
|
||
|
g_string_append_printf(out, ">\n<draw:text-box><text:p><draw:frame text:anchor-type=\"as-char\" draw:z-index=\"1\" ");
|
||
|
g_string_append_printf(out, "><draw:image xlink:href=\"");
|
||
|
print_odf_string(out, elt->contents.link->url);
|
||
|
g_string_append_printf(out,"\" xlink:type=\"simple\" xlink:show=\"embed\" xlink:actuate=\"onLoad\" draw:filter-name=\"<All formats>\"/>\n</draw:frame></text:p>");
|
||
|
g_string_append_printf(out, "</draw:text-box></draw:frame>\n");
|
||
|
break;
|
||
|
case EMPH:
|
||
|
g_string_append_printf(out,
|
||
|
"<text:span text:style-name=\"MMD-Italic\">");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "</text:span>");
|
||
|
break;
|
||
|
case STRONG:
|
||
|
g_string_append_printf(out,
|
||
|
"<text:span text:style-name=\"MMD-Bold\">");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "</text:span>");
|
||
|
break;
|
||
|
case LIST:
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
break;
|
||
|
case RAW:
|
||
|
/* Shouldn't occur - these are handled by process_raw_blocks() */
|
||
|
assert(elt->key != RAW);
|
||
|
break;
|
||
|
case H1: case H2: case H3: case H4: case H5: case H6:
|
||
|
lev = elt->key - H1 + 1; /* assumes H1 ... H6 are in order */
|
||
|
g_string_append_printf(out, "<text:h text:outline-level=\"%d\">", lev);
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "</text:h>\n");
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PLAIN:
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
padded = 0;
|
||
|
break;
|
||
|
case PARA:
|
||
|
g_string_append_printf(out, "<text:p");
|
||
|
switch (odf_type) {
|
||
|
case BLOCKQUOTE:
|
||
|
g_string_append_printf(out," text:style-name=\"Quotations\"");
|
||
|
break;
|
||
|
case CODE:
|
||
|
g_string_append_printf(out," text:style-name=\"Preformatted Text\"");
|
||
|
break;
|
||
|
case VERBATIM:
|
||
|
g_string_append_printf(out," text:style-name=\"Preformatted Text\"");
|
||
|
break;
|
||
|
case ORDEREDLIST:
|
||
|
case BULLETLIST:
|
||
|
g_string_append_printf(out," text:style-name=\"P2\"");
|
||
|
break;
|
||
|
case NOTE:
|
||
|
g_string_append_printf(out," text:style-name=\"Footnote\"");
|
||
|
break;
|
||
|
default:
|
||
|
g_string_append_printf(out," text:style-name=\"Standard\"");
|
||
|
break;
|
||
|
}
|
||
|
g_string_append_printf(out, ">");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "</text:p>\n");
|
||
|
break;
|
||
|
case HRULE:
|
||
|
g_string_append_printf(out,"<text:p text:style-name=\"Horizontal_20_Line\"/>\n");
|
||
|
break;
|
||
|
case HTMLBLOCK:
|
||
|
/* don't print HTML block */
|
||
|
/* but do print HTML comments for raw ODF */
|
||
|
if (strncmp(elt->contents.str,"<!--",4) == 0) {
|
||
|
/* trim "-->" from end */
|
||
|
elt->contents.str[strlen(elt->contents.str)-3] = '\0';
|
||
|
g_string_append_printf(out, "%s", &elt->contents.str[4]);
|
||
|
}
|
||
|
break;
|
||
|
case VERBATIM:
|
||
|
old_type = odf_type;
|
||
|
odf_type = VERBATIM;
|
||
|
g_string_append_printf(out, "<text:p text:style-name=\"Preformatted Text\">");
|
||
|
print_odf_code_string(out, elt->contents.str);
|
||
|
g_string_append_printf(out, "</text:p>\n");
|
||
|
odf_type = old_type;
|
||
|
break;
|
||
|
case BULLETLIST:
|
||
|
if ((odf_type == BULLETLIST) ||
|
||
|
(odf_type == ORDEREDLIST)) {
|
||
|
/* I think this was made unnecessary by another change.
|
||
|
Same for ORDEREDLIST below */
|
||
|
/* g_string_append_printf(out, "</text:p>"); */
|
||
|
}
|
||
|
old_type = odf_type;
|
||
|
odf_type = BULLETLIST;
|
||
|
g_string_append_printf(out, "%s", "<text:list>");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "%s", "</text:list>");
|
||
|
odf_type = old_type;
|
||
|
break;
|
||
|
case ORDEREDLIST:
|
||
|
if ((odf_type == BULLETLIST) ||
|
||
|
(odf_type == ORDEREDLIST)) {
|
||
|
/* g_string_append_printf(out, "</text:p>"); */
|
||
|
}
|
||
|
old_type = odf_type;
|
||
|
odf_type = ORDEREDLIST;
|
||
|
g_string_append_printf(out, "%s", "<text:list>\n");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "%s", "</text:list>\n");
|
||
|
odf_type = old_type;
|
||
|
break;
|
||
|
case LISTITEM:
|
||
|
g_string_append_printf(out, "<text:list-item>\n");
|
||
|
if (elt->children->children->key != PARA) {
|
||
|
g_string_append_printf(out, "<text:p text:style-name=\"P2\">");
|
||
|
}
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
|
||
|
if ((list_contains_key(elt->children,BULLETLIST) ||
|
||
|
(list_contains_key(elt->children,ORDEREDLIST)))) {
|
||
|
} else {
|
||
|
if (elt->children->children->key != PARA) {
|
||
|
g_string_append_printf(out, "</text:p>");
|
||
|
}
|
||
|
}
|
||
|
g_string_append_printf(out, "</text:list-item>\n");
|
||
|
break;
|
||
|
case BLOCKQUOTE:
|
||
|
old_type = odf_type;
|
||
|
odf_type = BLOCKQUOTE;
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
odf_type = old_type;
|
||
|
break;
|
||
|
case REFERENCE:
|
||
|
break;
|
||
|
case NOTE:
|
||
|
old_type = odf_type;
|
||
|
odf_type = NOTE;
|
||
|
/* if contents.str == 0 then print; else ignore - like above */
|
||
|
if (elt->contents.str == 0) {
|
||
|
g_string_append_printf(out, "<text:note text:id=\"\" text:note-class=\"footnote\"><text:note-body>\n");
|
||
|
print_odf_element_list(out, elt->children);
|
||
|
g_string_append_printf(out, "</text:note-body>\n</text:note>\n");
|
||
|
}
|
||
|
elt->children = NULL;
|
||
|
odf_type = old_type;
|
||
|
break;
|
||
|
break; default:
|
||
|
fprintf(stderr, "print_odf_element encountered unknown element key = %d\n", elt->key);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
|
||
|
Parameterized function for printing an Element.
|
||
|
|
||
|
***********************************************************************/
|
||
|
|
||
|
void print_element_list(GString *out, element *elt, int format, int exts) {
|
||
|
/* Initialize globals */
|
||
|
endnotes = NULL;
|
||
|
notenumber = 0;
|
||
|
|
||
|
extensions = exts;
|
||
|
padded = 2; /* set padding to 2, so no extra blank lines at beginning */
|
||
|
switch (format) {
|
||
|
case HTML_FORMAT:
|
||
|
print_html_element_list(out, elt, false);
|
||
|
if (endnotes != NULL) {
|
||
|
pad(out, 2);
|
||
|
print_html_endnotes(out);
|
||
|
}
|
||
|
break;
|
||
|
case LATEX_FORMAT:
|
||
|
print_latex_element_list(out, elt);
|
||
|
break;
|
||
|
case GROFF_MM_FORMAT:
|
||
|
print_groff_mm_element_list(out, elt);
|
||
|
break;
|
||
|
case ODF_FORMAT:
|
||
|
print_odf_header(out);
|
||
|
g_string_append_printf(out, "<office:body>\n<office:text>\n");
|
||
|
if (elt != NULL) print_odf_element_list(out,elt);
|
||
|
print_odf_footer(out);
|
||
|
break;
|
||
|
default:
|
||
|
fprintf(stderr, "print_element - unknown format = %d\n", format);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
}
|