/* * storycode.c * * Copyright © 2019 Thomas White * * This file is part of Colloquium. * * Colloquium is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * 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. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include "narrative.h" #include "slide.h" #include "stylesheet.h" #include "narrative_priv.h" #include "slide_priv.h" #include "storycode_parse.h" #include "storycode_lex.h" extern int scdebug; Narrative *storycode_parse_presentation(const char *sc) { YY_BUFFER_STATE b; Narrative *n; b = sc_scan_string(sc); //scdebug = 1; n = narrative_new(); scparse(n); sc_delete_buffer(b); //narrative_debug(n); return n; } static int write_string(GOutputStream *fh, char *str) { gssize r; GError *error = NULL; r = g_output_stream_write(fh, str, strlen(str), NULL, &error); if ( r == -1 ) { fprintf(stderr, "Write failed: %s\n", error->message); return 1; } return 0; } char unitc(enum length_unit unit) { if ( unit == LENGTH_FRAC ) return 'f'; if ( unit == LENGTH_UNIT ) return 'u'; return '?'; } const char *bgcolc(enum gradient bggrad) { if ( bggrad == GRAD_NONE ) return ""; if ( bggrad == GRAD_HORIZ ) return "HORIZONTAL "; if ( bggrad == GRAD_VERT ) return "VERTICAL "; return "?"; } const char *alignc(enum alignment ali) { if ( ali == ALIGN_LEFT ) return "left"; if ( ali == ALIGN_CENTER ) return "center"; if ( ali == ALIGN_RIGHT ) return "right"; return "?"; } static const char *maybe_alignment(enum alignment ali) { if ( ali == ALIGN_INHERIT ) return ""; if ( ali == ALIGN_LEFT ) return "[left]"; if ( ali == ALIGN_CENTER ) return "[center]"; if ( ali == ALIGN_RIGHT ) return "[right]"; return "[?]"; } static void write_run_border(GOutputStream *fh, enum text_run_type t) { if ( t == TEXT_RUN_BOLD ) write_string(fh, "*"); if ( t == TEXT_RUN_ITALIC ) write_string(fh, "/"); if ( t == TEXT_RUN_UNDERLINE ) write_string(fh, "_"); } static char *escape_text(const char *in) { int i, j; size_t len; size_t nl = 0; size_t np = 0; const char *esc = "*/_"; size_t n_esc = 3; char *out; len = strlen(in); for ( i=0; igeom.w.len, unitc(item->geom.w.unit), item->geom.h.len, unitc(item->geom.h.unit), item->geom.x.len, unitc(item->geom.x.unit), item->geom.y.len, unitc(item->geom.y.unit), maybe_alignment(item->align)); } else { snprintf(tmp, 255, " %s%s", t, maybe_alignment(item->align)); } indent = strlen(tmp); write_string(fh, tmp); write_string(fh, ": "); write_para(fh, item->paras[0].runs, item->paras[0].n_runs); write_string(fh, "\n"); for ( i=0; in_paras; i++ ) { write_string(fh, tmp); write_string(fh, ": "); write_para(fh, item->paras[i].runs, item->paras[i].n_runs); write_string(fh, "\n"); } } static void write_image(GOutputStream *fh, SlideItem *item) { char tmp[256]; snprintf(tmp, 255, " IMAGE[%.4g%cx%.4g%c+%.4g%c+%.4g%c]", item->geom.w.len, unitc(item->geom.w.unit), item->geom.h.len, unitc(item->geom.h.unit), item->geom.x.len, unitc(item->geom.x.unit), item->geom.y.len, unitc(item->geom.y.unit)); write_string(fh, tmp); write_string(fh, ": "); write_string(fh, item->filename); write_string(fh, "\n"); } static int write_slide(GOutputStream *fh, Slide *s) { int i; for ( i=0; in_items; i++ ) { switch ( s->items[i].type ) { case SLIDE_ITEM_TEXT: write_text(fh, &s->items[i], 1, "TEXT"); break; case SLIDE_ITEM_PRESTITLE: write_text(fh, &s->items[i], 0, "PRESTITLE"); break; case SLIDE_ITEM_SLIDETITLE: write_text(fh, &s->items[i], 0, "SLIDETITLE"); break; case SLIDE_ITEM_FOOTER: write_string(fh, " FOOTER\n"); break; case SLIDE_ITEM_IMAGE: write_image(fh, &s->items[i]); break; } } return 0; } static int write_item(GOutputStream *fh, struct narrative_item *item) { switch ( item->type ) { case NARRATIVE_ITEM_TEXT: /* FIXME: separate alignment */ if ( write_string(fh, ": ") ) return 1; write_para(fh, item->runs, item->n_runs); if ( write_string(fh, "\n") ) return 1; break; case NARRATIVE_ITEM_PRESTITLE: /* FIXME: separate alignment */ if ( write_string(fh, "PRESTITLE: ") ) return 1; write_para(fh, item->runs, item->n_runs); if ( write_string(fh, "\n") ) return 1; break; case NARRATIVE_ITEM_BP: /* FIXME: separate alignment */ if ( write_string(fh, "BP: ") ) return 1; write_para(fh, item->runs, item->n_runs); if ( write_string(fh, "\n") ) return 1; break; case NARRATIVE_ITEM_SLIDE: /* FIXME: separate slide size */ if ( write_string(fh, "SLIDE {\n") ) return 1; if ( write_slide(fh, item->slide) ) return 1; if ( write_string(fh, "}\n") ) return 1; break; case NARRATIVE_ITEM_EOP: if ( write_string(fh, "ENDOFPRESENTATION\n") ) return 1; break; } return 0; } int storycode_write_presentation(Narrative *n, GOutputStream *fh) { int i; char *ss_text; /* Stylesheet */ ss_text = stylesheet_serialise(n->stylesheet); if ( ss_text == NULL ) return 1; if ( write_string(fh, ss_text) ) return 1; if ( write_string(fh, "\n") ) return 1; for ( i=0; in_items; i++ ) { if ( write_item(fh, &n->items[i]) ) return 1; } return 0; }