/* * PDF creation tool: Tool for creating pdf content. * * Simple test bed to work with adding content and creating PDFs */ #include "mupdf/fitz.h" #include "mupdf/pdf.h" #include #include #include static void usage(void) { fprintf(stderr, "usage: mutool create [-o output.pdf] [-O options] page.txt [page2.txt ...]\n" "\t-o -\tname of PDF file to create\n" "\t-O -\tcomma separated list of output options\n" "\tpage.txt\tcontent stream with annotations for creating resources\n\n" "Content stream special commands:\n" "\t%%%%MediaBox LLX LLY URX URY\n" "\t%%%%Rotate Angle\n" "\t%%%%Font Name Filename (or base 14 font name)\n" "\t%%%%CJKFont Name Ordering WMode Style (Ordering=CNS1|GB1|Japan1|Korea1, WMode=H|V, Style=serif|sans)\n" "\t%%%%Image Name Filename\n\n" ); fputs(fz_pdf_write_options_usage, stderr); exit(1); } static fz_context *ctx = NULL; static pdf_document *doc = NULL; static void add_font_res(pdf_obj *resources, char *name, char *path, char *encname) { const unsigned char *data; int size, enc; fz_font *font; pdf_obj *subres, *ref; data = fz_lookup_base14_font(ctx, path, &size); if (data) font = fz_new_font_from_memory(ctx, path, data, size, 0, 0); else font = fz_new_font_from_file(ctx, NULL, path, 0, 0); subres = pdf_dict_get(ctx, resources, PDF_NAME(Font)); if (!subres) { subres = pdf_new_dict(ctx, doc, 10); pdf_dict_put_drop(ctx, resources, PDF_NAME(Font), subres); } enc = PDF_SIMPLE_ENCODING_LATIN; if (encname) { if (!strcmp(encname, "Latin")) enc = PDF_SIMPLE_ENCODING_LATIN; else if (!strcmp(encname, "Greek")) enc = PDF_SIMPLE_ENCODING_GREEK; else if (!strcmp(encname, "Cyrillic")) enc = PDF_SIMPLE_ENCODING_CYRILLIC; } ref = pdf_add_simple_font(ctx, doc, font, enc); pdf_dict_puts(ctx, subres, name, ref); pdf_drop_obj(ctx, ref); fz_drop_font(ctx, font); } static void add_cjkfont_res(pdf_obj *resources, char *name, char *on, char *wm, char *style) { const unsigned char *data; int size, index, ordering, wmode, serif; fz_font *font; pdf_obj *subres, *ref; if (!strcmp(on, "CNS1") || !strcmp(on, "TW") || !strcmp(on, "TC") || !strcmp(on, "Hant")) ordering = FZ_ADOBE_CNS_1; else if (!strcmp(on, "GB1") || !strcmp(on, "CN") || !strcmp(on, "SC") || !strcmp(on, "Hans")) ordering = FZ_ADOBE_GB_1; else if (!strcmp(on, "Japan1") || !strcmp(on, "JP") || !strcmp(on, "JA")) ordering = FZ_ADOBE_JAPAN_1; else if (!strcmp(on, "Korea1") || !strcmp(on, "KR") || !strcmp(on, "KO")) ordering = FZ_ADOBE_KOREA_1; else ordering = FZ_ADOBE_JAPAN_1; if (wm && !strcmp(wm, "V")) wmode = 1; else wmode = 0; if (style && (!strcmp(style, "sans") || !strcmp(style, "sans-serif"))) serif = 0; else serif = 1; data = fz_lookup_cjk_font(ctx, ordering, serif, &size, &index); font = fz_new_font_from_memory(ctx, NULL, data, size, index, 0); subres = pdf_dict_get(ctx, resources, PDF_NAME(Font)); if (!subres) { subres = pdf_new_dict(ctx, doc, 10); pdf_dict_put_drop(ctx, resources, PDF_NAME(Font), subres); } ref = pdf_add_cjk_font(ctx, doc, font, ordering, wmode, serif); pdf_dict_puts(ctx, subres, name, ref); pdf_drop_obj(ctx, ref); fz_drop_font(ctx, font); } static void add_image_res(pdf_obj *resources, char *name, char *path) { fz_image *image; pdf_obj *subres, *ref; image = fz_new_image_from_file(ctx, path); subres = pdf_dict_get(ctx, resources, PDF_NAME(XObject)); if (!subres) { subres = pdf_new_dict(ctx, doc, 10); pdf_dict_put_drop(ctx, resources, PDF_NAME(XObject), subres); } ref = pdf_add_image(ctx, doc, image, 0); pdf_dict_puts(ctx, subres, name, ref); pdf_drop_obj(ctx, ref); fz_drop_image(ctx, image); } /* The input is a raw content stream, with commands embedded in comments: %%MediaBox LLX LLY URX URY %%Rotate Angle %%Font Name Filename (or base 14 font name) [Encoding (Latin, Greek or Cyrillic)] %%CJKFont Name Ordering WMode Style (Ordering=CNS1|GB1|Japan1|Korea1, WMode=H|V, Style=serif|sans) %%Image Name Filename */ static void create_page(char *input) { fz_rect mediabox = { 0, 0, 595, 842 }; int rotate = 0; char line[4096]; char *s, *p; fz_stream *stm; fz_buffer *contents; pdf_obj *resources; pdf_obj *page; resources = pdf_new_dict(ctx, doc, 2); contents = fz_new_buffer(ctx, 1024); stm = fz_open_file(ctx, input); while (fz_read_line(ctx, stm, line, sizeof line)) { if (line[0] == '%' && line[1] == '%') { p = line; s = fz_strsep(&p, " "); if (!strcmp(s, "%%MediaBox")) { mediabox.x0 = fz_atoi(fz_strsep(&p, " ")); mediabox.y0 = fz_atoi(fz_strsep(&p, " ")); mediabox.x1 = fz_atoi(fz_strsep(&p, " ")); mediabox.y1 = fz_atoi(fz_strsep(&p, " ")); } else if (!strcmp(s, "%%Rotate")) { rotate = fz_atoi(fz_strsep(&p, " ")); } else if (!strcmp(s, "%%Font")) { char *name = fz_strsep(&p, " "); char *path = fz_strsep(&p, " "); char *enc = fz_strsep(&p, " "); if (!name || !path) fz_throw(ctx, FZ_ERROR_GENERIC, "Font directive missing arguments"); add_font_res(resources, name, path, enc); } else if (!strcmp(s, "%%CJKFont")) { char *name = fz_strsep(&p, " "); char *ordering = fz_strsep(&p, " "); char *wmode = fz_strsep(&p, " "); char *style = fz_strsep(&p, " "); if (!name || !ordering) fz_throw(ctx, FZ_ERROR_GENERIC, "CJKFont directive missing arguments"); add_cjkfont_res(resources, name, ordering, wmode, style); } else if (!strcmp(s, "%%Image")) { char *name = fz_strsep(&p, " "); char *path = fz_strsep(&p, " "); if (!name || !path) fz_throw(ctx, FZ_ERROR_GENERIC, "Image directive missing arguments"); add_image_res(resources, name, path); } } else { fz_append_string(ctx, contents, line); fz_append_byte(ctx, contents, '\n'); } } fz_drop_stream(ctx, stm); page = pdf_add_page(ctx, doc, mediabox, rotate, resources, contents); pdf_insert_page(ctx, doc, -1, page); pdf_drop_obj(ctx, page); fz_drop_buffer(ctx, contents); pdf_drop_obj(ctx, resources); } int pdfcreate_main(int argc, char **argv) { pdf_write_options opts = { 0 }; char *output = "out.pdf"; char *flags = "compress"; int i, c; while ((c = fz_getopt(argc, argv, "o:O:")) != -1) { switch (c) { case 'o': output = fz_optarg; break; case 'O': flags = fz_optarg; break; default: usage(); break; } } if (fz_optind == argc) usage(); ctx = fz_new_context(NULL, NULL, FZ_STORE_UNLIMITED); if (!ctx) { fprintf(stderr, "cannot initialise context\n"); exit(1); } pdf_parse_write_options(ctx, &opts, flags); doc = pdf_create_document(ctx); for (i = fz_optind; i < argc; ++i) create_page(argv[i]); pdf_save_document(ctx, doc, output, &opts); pdf_drop_document(ctx, doc); fz_flush_warnings(ctx); fz_drop_context(ctx); return 0; }