2024-12-10 22:35:12 +01:00
|
|
|
|
//
|
|
|
|
|
// Simple markdown to PDF converter example for PDFio.
|
|
|
|
|
//
|
|
|
|
|
// Copyright © 2024 by Michael R Sweet.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under Apache License v2.0. See the file "LICENSE" for more
|
|
|
|
|
// information.
|
|
|
|
|
//
|
|
|
|
|
// Usage:
|
|
|
|
|
//
|
|
|
|
|
// ./md2pdf FILENAME.md >FILENAME.pdf
|
|
|
|
|
//
|
|
|
|
|
// The generated PDF file is formatted for a "universal" paper size (8.27x11",
|
|
|
|
|
// the intersection of US Letter and ISO A4) with 1" top and bottom margins and
|
|
|
|
|
// 0.5" side margins. The document title (if present) is centered at the top
|
|
|
|
|
// of the second and subsequent pages while the current heading and page number
|
|
|
|
|
// are provided at the bottom of each page.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
2024-12-12 02:14:32 +01:00
|
|
|
|
#include <math.h>
|
2024-12-11 21:37:03 +01:00
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
# include <io.h>
|
|
|
|
|
#else
|
|
|
|
|
# include <unistd.h>
|
|
|
|
|
#endif // _WIN32
|
2024-12-10 22:35:12 +01:00
|
|
|
|
#include "mmd.h"
|
|
|
|
|
#include <pdfio.h>
|
|
|
|
|
#include <pdfio-content.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Types...
|
|
|
|
|
//
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
typedef enum doccolor_e // Document color enumeration
|
|
|
|
|
{
|
|
|
|
|
DOCCOLOR_BLACK, // #000
|
|
|
|
|
DOCCOLOR_RED, // #900
|
|
|
|
|
DOCCOLOR_GREEN, // #090
|
|
|
|
|
DOCCOLOR_BLUE, // #00C
|
|
|
|
|
DOCCOLOR_GRAY // #555
|
|
|
|
|
} doccolor_t;
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
typedef enum docfont_e // Document font enumeration
|
|
|
|
|
{
|
|
|
|
|
DOCFONT_REGULAR, // Roboto-Regular
|
|
|
|
|
DOCFONT_BOLD, // Roboto-Bold
|
|
|
|
|
DOCFONT_ITALIC, // Roboto-Italic
|
|
|
|
|
DOCFONT_MONOSPACE, // RobotoMono-Regular
|
|
|
|
|
DOCFONT_MAX // Number of fonts
|
|
|
|
|
} docfont_t;
|
|
|
|
|
|
|
|
|
|
typedef struct docimage_s // Document image info
|
|
|
|
|
{
|
|
|
|
|
const char *url; // Reference URL
|
|
|
|
|
pdfio_obj_t *obj; // Image object
|
|
|
|
|
} docimage_t;
|
|
|
|
|
|
|
|
|
|
#define DOCIMAGE_MAX 1000 // Maximum number of images
|
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
typedef struct doclink_s // Document link info
|
|
|
|
|
{
|
|
|
|
|
const char *url; // Reference URL
|
|
|
|
|
pdfio_rect_t box; // Link box
|
|
|
|
|
} doclink_t;
|
|
|
|
|
|
|
|
|
|
#define DOCLINK_MAX 1000 // Maximum number of links/page
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
typedef struct docdata_s // Document formatting data
|
|
|
|
|
{
|
|
|
|
|
pdfio_file_t *pdf; // PDF file
|
|
|
|
|
pdfio_rect_t media_box; // Media (page) box
|
|
|
|
|
pdfio_rect_t crop_box; // Crop box (for margins)
|
|
|
|
|
pdfio_rect_t art_box; // Art box (for markdown content)
|
|
|
|
|
pdfio_obj_t *fonts[DOCFONT_MAX]; // Embedded fonts
|
|
|
|
|
size_t num_images; // Number of embedded images
|
|
|
|
|
docimage_t images[DOCIMAGE_MAX]; // Embedded images
|
|
|
|
|
const char *title; // Document title
|
|
|
|
|
char *heading; // Current document heading
|
|
|
|
|
pdfio_stream_t *st; // Current page stream
|
|
|
|
|
double y; // Current position on page
|
2024-12-12 02:14:32 +01:00
|
|
|
|
docfont_t font; // Current font
|
|
|
|
|
double fsize; // Current font size
|
2024-12-11 00:41:23 +01:00
|
|
|
|
doccolor_t color; // Current color
|
2024-12-11 21:37:03 +01:00
|
|
|
|
pdfio_obj_t *annots; // Annotations object (for links)
|
|
|
|
|
size_t num_links; // Number of links for this page
|
|
|
|
|
doclink_t links[DOCLINK_MAX]; // Links for this page
|
2024-12-10 22:35:12 +01:00
|
|
|
|
} docdata_t;
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
typedef struct linefrag_s // Line fragment
|
|
|
|
|
{
|
|
|
|
|
double x, // X position of item
|
|
|
|
|
width, // Width of item
|
|
|
|
|
height; // Height of item
|
|
|
|
|
size_t imagenum; // Image number
|
|
|
|
|
const char *text; // Text string
|
|
|
|
|
bool ws; // Whitespace before text?
|
|
|
|
|
docfont_t font; // Text font
|
|
|
|
|
doccolor_t color; // Text color
|
|
|
|
|
} linefrag_t;
|
|
|
|
|
|
|
|
|
|
#define LINEFRAG_MAX 200 // Maximum number of fragments on a line
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Macros...
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#define in2pt(in) (in * 72.0)
|
|
|
|
|
#define mm2pt(mm) (mm * 72.0 / 25.4)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Constants...
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static const char * const docfont_filenames[] =
|
|
|
|
|
{
|
|
|
|
|
"Roboto-Regular.ttf",
|
|
|
|
|
"Roboto-Bold.ttf",
|
|
|
|
|
"Roboto-Italic.ttf",
|
|
|
|
|
"RobotoMono-Regular.ttf"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const char * const docfont_names[] =
|
|
|
|
|
{
|
|
|
|
|
"FR",
|
|
|
|
|
"FB",
|
|
|
|
|
"FI",
|
|
|
|
|
"FM"
|
|
|
|
|
};
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
#define LINE_HEIGHT 1.4 // Multiplier for line height
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
#define SIZE_BODY 11.0 // Size of body text (points)
|
2024-12-11 00:41:23 +01:00
|
|
|
|
#define SIZE_CODEBLOCK 10.0 // Size of code block text (points)
|
|
|
|
|
#define SIZE_HEADFOOT 9.0 // Size of header/footer text (points)
|
|
|
|
|
#define SIZE_HEADING_1 18.0 // Size of first level heading (points)
|
|
|
|
|
#define SIZE_HEADING_2 16.0 // Size of top level heading (points)
|
|
|
|
|
#define SIZE_HEADING_3 15.0 // Size of top level heading (points)
|
|
|
|
|
#define SIZE_HEADING_4 14.0 // Size of top level heading (points)
|
|
|
|
|
#define SIZE_HEADING_5 13.0 // Size of top level heading (points)
|
|
|
|
|
#define SIZE_HEADING_6 12.0 // Size of top level heading (points)
|
|
|
|
|
#define SIZE_TABLE 10.0 // Size of table text (points)
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
#define PAGE_WIDTH mm2pt(210) // Page width in points
|
|
|
|
|
#define PAGE_LENGTH in2pt(11) // Page length in points
|
|
|
|
|
#define PAGE_LEFT in2pt(0.5) // Left margin in points
|
|
|
|
|
#define PAGE_RIGHT (PAGE_WIDTH - in2pt(0.5))
|
|
|
|
|
// Right margin in points
|
|
|
|
|
#define PAGE_BOTTOM in2pt(1.0) // Bottom margin in points
|
|
|
|
|
#define PAGE_TOP (PAGE_LENGTH - in2pt(1.0))
|
|
|
|
|
// Top margin in points
|
|
|
|
|
#define PAGE_HEADER (PAGE_LENGTH - in2pt(0.5))
|
|
|
|
|
// Vertical position of header
|
|
|
|
|
#define PAGE_FOOTER in2pt(0.5) // Vertical position of footer
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
#define UNICODE_VALUE true // `true` for Unicode text, `false` for ISO-8859-1
|
|
|
|
|
|
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
//
|
|
|
|
|
// 'mmd_walk_next()' - Find the next markdown node.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static mmd_t * // O - Next node or `NULL` at end
|
|
|
|
|
mmd_walk_next(mmd_t *top, // I - Top node
|
|
|
|
|
mmd_t *node) // I - Current node
|
|
|
|
|
{
|
|
|
|
|
mmd_t *next, // Next node
|
|
|
|
|
*parent; // Parent node
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Figure out the next node under "top"...
|
|
|
|
|
if ((next = mmdGetFirstChild(node)) == NULL)
|
|
|
|
|
{
|
|
|
|
|
if ((next = mmdGetNextSibling(node)) == NULL)
|
|
|
|
|
{
|
|
|
|
|
if ((parent = mmdGetParent(node)) != top)
|
|
|
|
|
{
|
|
|
|
|
while ((next = mmdGetNextSibling(parent)) == NULL)
|
|
|
|
|
{
|
|
|
|
|
if ((parent = mmdGetParent(parent)) == top)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (next);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'add_images()' - Scan the markdown document for images.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
add_images(docdata_t *dd, // I - Document data
|
|
|
|
|
mmd_t *doc) // I - Markdown document
|
|
|
|
|
{
|
|
|
|
|
mmd_t *current, // Current node
|
|
|
|
|
*next; // Next node
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Scan the entire document for images...
|
|
|
|
|
for (current = mmdGetFirstChild(doc); current; current = next)
|
|
|
|
|
{
|
|
|
|
|
// Get next node
|
|
|
|
|
next = mmd_walk_next(doc, current);
|
|
|
|
|
|
|
|
|
|
// Look for image nodes...
|
|
|
|
|
if (mmdGetType(current) == MMD_TYPE_IMAGE)
|
|
|
|
|
{
|
|
|
|
|
const char *url, // URL for image
|
|
|
|
|
*ext; // Extension
|
|
|
|
|
|
|
|
|
|
url = mmdGetURL(current);
|
|
|
|
|
ext = strrchr(url, '.');
|
|
|
|
|
|
|
|
|
|
fprintf(stderr, "IMAGE(%s), ext=\"%s\"\n", url, ext);
|
|
|
|
|
|
|
|
|
|
if (!access(url, 0) && ext && (!strcmp(ext, ".png") || !strcmp(ext, ".jpg") || !strcmp(ext, ".jpeg")))
|
|
|
|
|
{
|
|
|
|
|
// Local JPEG or PNG file, so add it if we haven't already...
|
|
|
|
|
size_t i; // Looping var
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < dd->num_images; i ++)
|
|
|
|
|
{
|
|
|
|
|
if (!strcmp(dd->images[i].url, url))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (i >= dd->num_images && dd->num_images < DOCIMAGE_MAX)
|
|
|
|
|
{
|
|
|
|
|
dd->images[i].url = url;
|
|
|
|
|
if ((dd->images[i].obj = pdfioFileCreateImageObjFromFile(dd->pdf, url, false)) != NULL)
|
|
|
|
|
dd->num_images ++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
//
|
2024-12-12 02:14:32 +01:00
|
|
|
|
// 'set_color()' - Set the stroke and fill color as needed.
|
2024-12-11 00:41:23 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
set_color(docdata_t *dd, // I - Document data
|
|
|
|
|
doccolor_t color) // I - Document color
|
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (color == dd->color)
|
|
|
|
|
return;
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
switch (color)
|
|
|
|
|
{
|
|
|
|
|
case DOCCOLOR_BLACK :
|
|
|
|
|
pdfioContentSetFillColorDeviceGray(dd->st, 0.0);
|
|
|
|
|
pdfioContentSetStrokeColorDeviceGray(dd->st, 0.0);
|
|
|
|
|
break;
|
|
|
|
|
case DOCCOLOR_RED :
|
|
|
|
|
pdfioContentSetFillColorDeviceRGB(dd->st, 0.6, 0.0, 0.0);
|
|
|
|
|
pdfioContentSetStrokeColorDeviceRGB(dd->st, 0.6, 0.0, 0.0);
|
|
|
|
|
break;
|
|
|
|
|
case DOCCOLOR_GREEN :
|
|
|
|
|
pdfioContentSetFillColorDeviceRGB(dd->st, 0.0, 0.6, 0.0);
|
|
|
|
|
pdfioContentSetStrokeColorDeviceRGB(dd->st, 0.0, 0.6, 0.0);
|
|
|
|
|
break;
|
|
|
|
|
case DOCCOLOR_BLUE :
|
|
|
|
|
pdfioContentSetFillColorDeviceRGB(dd->st, 0.0, 0.0, 0.8);
|
|
|
|
|
pdfioContentSetStrokeColorDeviceRGB(dd->st, 0.0, 0.0, 0.8);
|
|
|
|
|
break;
|
|
|
|
|
case DOCCOLOR_GRAY :
|
|
|
|
|
pdfioContentSetFillColorDeviceGray(dd->st, 0.333);
|
|
|
|
|
pdfioContentSetStrokeColorDeviceGray(dd->st, 0.333);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dd->color = color;
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
//
|
|
|
|
|
// 'set_font()' - Set the font typeface and size as needed.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
set_font(docdata_t *dd, // I - Document data
|
|
|
|
|
docfont_t font, // I - Font
|
|
|
|
|
double fsize) // I - Font size
|
|
|
|
|
{
|
|
|
|
|
if (font == dd->font && fabs(fsize - dd->fsize) < 0.1)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (font == DOCFONT_MAX)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
pdfioContentSetTextFont(dd->st, docfont_names[font], fsize);
|
|
|
|
|
|
|
|
|
|
dd->font = font;
|
|
|
|
|
dd->fsize = fsize;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
//
|
|
|
|
|
// 'new_page()' - Start a new page.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
new_page(docdata_t *dd) // I - Document data
|
|
|
|
|
{
|
|
|
|
|
pdfio_dict_t *page_dict; // Page dictionary
|
|
|
|
|
docfont_t fontface; // Current font face
|
|
|
|
|
size_t i; // Looping var
|
|
|
|
|
char temp[32]; // Temporary string
|
|
|
|
|
double width; // Width of fragment
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Close the current page...
|
|
|
|
|
pdfioStreamClose(dd->st);
|
|
|
|
|
|
|
|
|
|
// Prep the new page...
|
|
|
|
|
page_dict = pdfioDictCreate(dd->pdf);
|
|
|
|
|
|
|
|
|
|
pdfioDictSetRect(page_dict, "MediaBox", &dd->media_box);
|
|
|
|
|
// pdfioDictSetRect(page_dict, "CropBox", &dd->crop_box);
|
|
|
|
|
pdfioDictSetRect(page_dict, "ArtBox", &dd->art_box);
|
|
|
|
|
|
|
|
|
|
for (fontface = DOCFONT_REGULAR; fontface < DOCFONT_MAX; fontface ++)
|
|
|
|
|
pdfioPageDictAddFont(page_dict, docfont_names[fontface], dd->fonts[fontface]);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < dd->num_images; i ++)
|
|
|
|
|
{
|
|
|
|
|
snprintf(temp, sizeof(temp), "I%u", (unsigned)i);
|
|
|
|
|
pdfioPageDictAddImage(page_dict, temp, dd->images[i].obj);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
dd->st = pdfioFileCreatePage(dd->pdf, page_dict);
|
|
|
|
|
dd->color = DOCCOLOR_BLACK;
|
|
|
|
|
dd->font = DOCFONT_MAX;
|
|
|
|
|
dd->fsize = 0.0;
|
|
|
|
|
dd->y = dd->art_box.y2;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
// Add header/footer text
|
2024-12-11 00:41:23 +01:00
|
|
|
|
set_color(dd, DOCCOLOR_GRAY);
|
2024-12-12 02:14:32 +01:00
|
|
|
|
set_font(dd, DOCFONT_REGULAR, SIZE_HEADFOOT);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
if (pdfioFileGetNumPages(dd->pdf) > 1 && dd->title)
|
|
|
|
|
{
|
|
|
|
|
// Show title in header...
|
|
|
|
|
width = pdfioContentTextMeasure(dd->fonts[DOCFONT_REGULAR], dd->title, SIZE_HEADFOOT);
|
|
|
|
|
|
|
|
|
|
pdfioContentTextBegin(dd->st);
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, dd->crop_box.x1 + 0.5 * (dd->crop_box.x2 - dd->crop_box.x1 - width), dd->crop_box.y2 - SIZE_HEADFOOT);
|
2024-12-11 00:41:23 +01:00
|
|
|
|
pdfioContentTextShow(dd->st, UNICODE_VALUE, dd->title);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
pdfioContentTextEnd(dd->st);
|
|
|
|
|
|
|
|
|
|
pdfioContentPathMoveTo(dd->st, dd->crop_box.x1, dd->crop_box.y2 - 2 * SIZE_HEADFOOT * LINE_HEIGHT + SIZE_HEADFOOT);
|
|
|
|
|
pdfioContentPathLineTo(dd->st, dd->crop_box.x2, dd->crop_box.y2 - 2 * SIZE_HEADFOOT * LINE_HEIGHT + SIZE_HEADFOOT);
|
|
|
|
|
pdfioContentStroke(dd->st);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Show page number and current heading...
|
|
|
|
|
pdfioContentPathMoveTo(dd->st, dd->crop_box.x1, dd->crop_box.y1 + SIZE_HEADFOOT * LINE_HEIGHT);
|
|
|
|
|
pdfioContentPathLineTo(dd->st, dd->crop_box.x2, dd->crop_box.y1 + SIZE_HEADFOOT * LINE_HEIGHT);
|
|
|
|
|
pdfioContentStroke(dd->st);
|
|
|
|
|
|
|
|
|
|
pdfioContentTextBegin(dd->st);
|
|
|
|
|
snprintf(temp, sizeof(temp), "%u", (unsigned)pdfioFileGetNumPages(dd->pdf));
|
|
|
|
|
if (pdfioFileGetNumPages(dd->pdf) & 1)
|
|
|
|
|
{
|
|
|
|
|
// Page number on right...
|
|
|
|
|
width = pdfioContentTextMeasure(dd->fonts[DOCFONT_REGULAR], temp, SIZE_HEADFOOT);
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, dd->crop_box.x2 - width, dd->crop_box.y1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Page number on left...
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, dd->crop_box.x1, dd->crop_box.y1);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
pdfioContentTextShow(dd->st, UNICODE_VALUE, temp);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
pdfioContentTextEnd(dd->st);
|
|
|
|
|
|
|
|
|
|
if (dd->heading)
|
|
|
|
|
{
|
|
|
|
|
pdfioContentTextBegin(dd->st);
|
2024-12-11 00:41:23 +01:00
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
if (pdfioFileGetNumPages(dd->pdf) & 1)
|
|
|
|
|
{
|
|
|
|
|
// Current heading on left...
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, dd->crop_box.x1, dd->crop_box.y1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
width = pdfioContentTextMeasure(dd->fonts[DOCFONT_REGULAR], dd->heading, SIZE_HEADFOOT);
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, dd->crop_box.x2 - width, dd->crop_box.y1);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-11 00:41:23 +01:00
|
|
|
|
pdfioContentTextShow(dd->st, UNICODE_VALUE, dd->heading);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
pdfioContentTextEnd(dd->st);
|
|
|
|
|
}
|
2024-12-12 02:14:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'render_line()' - Render a line of text/graphics.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
render_line(docdata_t *dd, // I - Document data
|
|
|
|
|
double margin_top, // I - Top margin
|
|
|
|
|
double lineheight, // I - Height of line
|
|
|
|
|
size_t num_frags, // I - Number of line fragments
|
|
|
|
|
linefrag_t *frags) // I - Line fragments
|
|
|
|
|
{
|
|
|
|
|
size_t i; // Looping var
|
|
|
|
|
linefrag_t *frag; // Current line fragment
|
|
|
|
|
bool in_text = false; // Are we in a text block?
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!dd->st)
|
|
|
|
|
new_page(dd);
|
|
|
|
|
|
|
|
|
|
dd->y -= margin_top + lineheight;
|
|
|
|
|
if (dd->y < dd->art_box.y1)
|
|
|
|
|
{
|
|
|
|
|
new_page(dd);
|
|
|
|
|
|
|
|
|
|
dd->y -= lineheight;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(stderr, "num_frags=%u, y=%g\n", (unsigned)num_frags, dd->y);
|
|
|
|
|
|
|
|
|
|
for (i = 0, frag = frags; i < num_frags; i ++, frag ++)
|
|
|
|
|
{
|
|
|
|
|
if (frag->text)
|
|
|
|
|
{
|
|
|
|
|
// Draw text
|
|
|
|
|
fprintf(stderr, " text=\"%s\", font=%d, color=%d, x=%g\n", frag->text, frag->font, frag->color, frag->x);
|
|
|
|
|
|
|
|
|
|
set_color(dd, frag->color);
|
|
|
|
|
set_font(dd, frag->font, frag->height);
|
|
|
|
|
|
|
|
|
|
if (!in_text)
|
|
|
|
|
{
|
|
|
|
|
pdfioContentTextBegin(dd->st);
|
|
|
|
|
pdfioContentTextMoveTo(dd->st, frag->x, dd->y);
|
|
|
|
|
|
|
|
|
|
in_text = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (frag->ws)
|
|
|
|
|
pdfioContentTextShowf(dd->st, UNICODE_VALUE, " %s", frag->text);
|
|
|
|
|
else
|
|
|
|
|
pdfioContentTextShow(dd->st, UNICODE_VALUE, frag->text);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Draw image
|
|
|
|
|
char imagename[32]; // Current image name
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
fprintf(stderr, " imagenum=%u, x=%g, width=%g, height=%g\n", (unsigned)frag->imagenum, frag->x, frag->width, frag->height);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (in_text)
|
|
|
|
|
{
|
|
|
|
|
pdfioContentTextEnd(dd->st);
|
|
|
|
|
in_text = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
snprintf(imagename, sizeof(imagename), "I%u", (unsigned)frag->imagenum);
|
|
|
|
|
pdfioContentDrawImage(dd->st, imagename, frag->x, dd->y, frag->width, frag->height);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (in_text)
|
|
|
|
|
pdfioContentTextEnd(dd->st);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'format_block()' - Format a block of text
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
format_block(docdata_t *dd, // I - Document data
|
|
|
|
|
mmd_t *block, // I - Block to format
|
2024-12-12 02:14:32 +01:00
|
|
|
|
docfont_t deffont, // I - Default font
|
|
|
|
|
double fsize, // I - Size of font
|
2024-12-10 22:35:12 +01:00
|
|
|
|
double left, // I - Left margin
|
|
|
|
|
double right, // I - Right margin
|
|
|
|
|
const char *leader) // I - Leader text on the first line
|
|
|
|
|
{
|
|
|
|
|
mmd_type_t blocktype; // Block type
|
|
|
|
|
mmd_t *current, // Current node
|
|
|
|
|
*next; // Next node
|
2024-12-12 02:14:32 +01:00
|
|
|
|
size_t num_frags; // Number of line fragments
|
|
|
|
|
linefrag_t frags[LINEFRAG_MAX], // Line fragments
|
|
|
|
|
*frag; // Current fragment
|
|
|
|
|
mmd_type_t type; // Current node type
|
|
|
|
|
const char *text, // Current text
|
|
|
|
|
*url; // Current URL, if any
|
|
|
|
|
bool ws; // Current whitespace
|
|
|
|
|
pdfio_obj_t *image; // Current image, if any
|
|
|
|
|
size_t imagenum; // Current image number
|
|
|
|
|
doccolor_t color = DOCCOLOR_BLACK; // Current text color
|
|
|
|
|
docfont_t font = deffont; // Current text font
|
|
|
|
|
double x, // Current position
|
|
|
|
|
width, // Width of current fragment
|
|
|
|
|
wswidth, // Width of whitespace
|
|
|
|
|
margin_top, // Top margin
|
2024-12-11 21:37:03 +01:00
|
|
|
|
height, // Height of current fragment
|
2024-12-12 02:14:32 +01:00
|
|
|
|
lineheight; // Height of current line
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
blocktype = mmdGetType(block);
|
|
|
|
|
margin_top = fsize * LINE_HEIGHT;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
if (leader)
|
|
|
|
|
{
|
|
|
|
|
// Add leader text on first line...
|
2024-12-12 02:14:32 +01:00
|
|
|
|
frags[0].width = pdfioContentTextMeasure(dd->fonts[deffont], leader, fsize);
|
|
|
|
|
frags[0].height = fsize;
|
|
|
|
|
frags[0].x = left - frags[0].width;
|
|
|
|
|
frags[0].text = leader;
|
|
|
|
|
frags[0].font = deffont;
|
|
|
|
|
frags[0].color = DOCCOLOR_BLACK;
|
|
|
|
|
|
|
|
|
|
num_frags = 1;
|
|
|
|
|
lineheight = fsize * LINE_HEIGHT;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// No leader text...
|
2024-12-12 02:14:32 +01:00
|
|
|
|
num_frags = 0;
|
|
|
|
|
lineheight = 0.0;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
frag = frags + num_frags;
|
|
|
|
|
|
|
|
|
|
// Loop through the block and render lines...
|
2024-12-10 22:35:12 +01:00
|
|
|
|
for (current = mmdGetFirstChild(block), x = left; current; current = next)
|
|
|
|
|
{
|
|
|
|
|
// Get information about the current node...
|
2024-12-12 02:14:32 +01:00
|
|
|
|
type = mmdGetType(current);
|
|
|
|
|
text = mmdGetText(current);
|
|
|
|
|
image = NULL;
|
|
|
|
|
imagenum = 0;
|
|
|
|
|
url = mmdGetURL(current);
|
|
|
|
|
ws = mmdGetWhitespace(current);
|
|
|
|
|
wswidth = 0.0;
|
|
|
|
|
next = mmd_walk_next(block, current);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
// Process the node...
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (type == MMD_TYPE_IMAGE && url)
|
2024-12-10 22:35:12 +01:00
|
|
|
|
{
|
2024-12-11 21:37:03 +01:00
|
|
|
|
// Embed an image
|
|
|
|
|
size_t i; // Looping var
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
for (i = 0; i < dd->num_images; i ++)
|
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (!strcmp(dd->images[i].url, url))
|
2024-12-10 22:35:12 +01:00
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
image = dd->images[i].obj;
|
|
|
|
|
imagenum = i;
|
2024-12-11 21:37:03 +01:00
|
|
|
|
break;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (!image)
|
2024-12-11 21:37:03 +01:00
|
|
|
|
continue;
|
2024-12-11 00:41:23 +01:00
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
// Image - treat as 100dpi
|
2024-12-12 02:14:32 +01:00
|
|
|
|
width = 72.0 * pdfioImageGetWidth(image) / 100.0;
|
|
|
|
|
height = 72.0 * pdfioImageGetHeight(image) / 100.0;
|
|
|
|
|
text = NULL;
|
2024-12-11 21:37:03 +01:00
|
|
|
|
|
|
|
|
|
if (width > (right - left))
|
|
|
|
|
{
|
|
|
|
|
// Too wide, scale to width...
|
|
|
|
|
width = right - left;
|
2024-12-12 02:14:32 +01:00
|
|
|
|
height = width * pdfioImageGetHeight(image) / pdfioImageGetWidth(image);
|
2024-12-11 21:37:03 +01:00
|
|
|
|
}
|
|
|
|
|
else if (height > (dd->art_box.y2 - dd->art_box.y1))
|
|
|
|
|
{
|
|
|
|
|
// Too tall, scale to height...
|
|
|
|
|
height = dd->art_box.y2 - dd->art_box.y1;
|
2024-12-12 02:14:32 +01:00
|
|
|
|
width = height * pdfioImageGetWidth(image) / pdfioImageGetHeight(image);
|
2024-12-11 21:37:03 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-12-12 02:14:32 +01:00
|
|
|
|
else if (!text)
|
2024-12-11 21:37:03 +01:00
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
continue;
|
2024-12-11 21:37:03 +01:00
|
|
|
|
}
|
2024-12-10 22:35:12 +01:00
|
|
|
|
else
|
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
// Text fragment...
|
|
|
|
|
if (type == MMD_TYPE_EMPHASIZED_TEXT)
|
|
|
|
|
font = DOCFONT_ITALIC;
|
|
|
|
|
else if (type == MMD_TYPE_STRONG_TEXT)
|
|
|
|
|
font = DOCFONT_BOLD;
|
|
|
|
|
else if (type == MMD_TYPE_CODE_TEXT)
|
|
|
|
|
font = DOCFONT_MONOSPACE;
|
2024-12-11 00:41:23 +01:00
|
|
|
|
else
|
2024-12-12 02:14:32 +01:00
|
|
|
|
font = deffont;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (type == MMD_TYPE_CODE_TEXT)
|
|
|
|
|
color = DOCCOLOR_RED;
|
|
|
|
|
else if (type == MMD_TYPE_LINKED_TEXT)
|
|
|
|
|
color = DOCCOLOR_BLUE;
|
|
|
|
|
else
|
|
|
|
|
color = DOCCOLOR_BLACK;
|
2024-12-11 21:37:03 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
width = pdfioContentTextMeasure(dd->fonts[font], text, fsize);
|
|
|
|
|
height = fsize * LINE_HEIGHT;
|
2024-12-11 21:37:03 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (ws)
|
|
|
|
|
wswidth = pdfioContentTextMeasure(dd->fonts[font], " ", fsize);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
// See if this node will fit on the current line...
|
|
|
|
|
if ((num_frags > 0 && (x + width + wswidth) >= right) || num_frags == LINEFRAG_MAX)
|
2024-12-10 22:35:12 +01:00
|
|
|
|
{
|
2024-12-12 02:14:32 +01:00
|
|
|
|
// No, render this line and start over...
|
|
|
|
|
render_line(dd, margin_top, lineheight, num_frags, frags);
|
|
|
|
|
|
|
|
|
|
num_frags = 0;
|
|
|
|
|
frag = frags;
|
|
|
|
|
x = left;
|
|
|
|
|
lineheight = 0.0;
|
|
|
|
|
margin_top = 0.0;
|
|
|
|
|
}
|
2024-12-11 00:41:23 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
// Add the current node to the fragment list
|
|
|
|
|
if (num_frags == 0)
|
|
|
|
|
wswidth = 0.0;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
frag->x = x;
|
|
|
|
|
frag->width = width + wswidth;
|
|
|
|
|
frag->height = text ? fsize : height;
|
|
|
|
|
frag->imagenum = imagenum;
|
|
|
|
|
frag->text = text;
|
|
|
|
|
frag->ws = ws;
|
|
|
|
|
frag->font = font;
|
|
|
|
|
frag->color = color;
|
|
|
|
|
|
|
|
|
|
num_frags ++;
|
|
|
|
|
frag ++;
|
|
|
|
|
x += width + wswidth;
|
|
|
|
|
if (height > lineheight)
|
|
|
|
|
lineheight = height;
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
if (num_frags > 0)
|
|
|
|
|
render_line(dd, margin_top, lineheight, num_frags, frags);
|
2024-12-10 22:35:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'format_doc()' - Format a document.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
format_doc(docdata_t *dd, // I - Document data
|
|
|
|
|
mmd_t *doc, // I - Document node to format
|
|
|
|
|
double left, // I - Left margin
|
|
|
|
|
double right) // I - Right margin
|
|
|
|
|
{
|
|
|
|
|
int i; // Child number
|
|
|
|
|
mmd_type_t doctype; // Document node type
|
|
|
|
|
mmd_t *current; // Current node
|
|
|
|
|
mmd_type_t curtype; // Current node type
|
|
|
|
|
char leader[32]; // Leader
|
|
|
|
|
static const double heading_sizes[] = // Heading font sizes
|
|
|
|
|
{
|
|
|
|
|
SIZE_HEADING_1,
|
|
|
|
|
SIZE_HEADING_2,
|
|
|
|
|
SIZE_HEADING_3,
|
|
|
|
|
SIZE_HEADING_4,
|
|
|
|
|
SIZE_HEADING_5,
|
|
|
|
|
SIZE_HEADING_6
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
doctype = mmdGetType(doc);
|
|
|
|
|
|
|
|
|
|
for (i = 1, current = mmdGetFirstChild(doc); current; i ++, current = mmdGetNextSibling(current))
|
|
|
|
|
{
|
|
|
|
|
switch (curtype = mmdGetType(current))
|
|
|
|
|
{
|
|
|
|
|
default :
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_BLOCK_QUOTE :
|
|
|
|
|
format_doc(dd, current, left + 36.0, right - 36.0);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_ORDERED_LIST :
|
|
|
|
|
case MMD_TYPE_UNORDERED_LIST :
|
|
|
|
|
format_doc(dd, current, left + 36.0, right);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_LIST_ITEM :
|
|
|
|
|
if (doctype == MMD_TYPE_ORDERED_LIST)
|
|
|
|
|
{
|
|
|
|
|
snprintf(leader, sizeof(leader), "%d. ", i);
|
|
|
|
|
format_block(dd, current, DOCFONT_REGULAR, SIZE_BODY, left, right, leader);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
format_block(dd, current, DOCFONT_REGULAR, SIZE_BODY, left, right, /*leader*/"• ");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_HEADING_1 :
|
|
|
|
|
case MMD_TYPE_HEADING_2 :
|
|
|
|
|
case MMD_TYPE_HEADING_3 :
|
|
|
|
|
case MMD_TYPE_HEADING_4 :
|
|
|
|
|
case MMD_TYPE_HEADING_5 :
|
|
|
|
|
case MMD_TYPE_HEADING_6 :
|
|
|
|
|
if (dd->heading)
|
|
|
|
|
free(dd->heading);
|
|
|
|
|
|
|
|
|
|
dd->heading = mmdCopyAllText(current);
|
|
|
|
|
|
|
|
|
|
format_block(dd, current, DOCFONT_BOLD, heading_sizes[curtype - MMD_TYPE_HEADING_1], left, right, /*leader*/NULL);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_PARAGRAPH :
|
|
|
|
|
format_block(dd, current, DOCFONT_REGULAR, SIZE_BODY, left, right, /*leader*/NULL);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case MMD_TYPE_CODE_BLOCK :
|
2024-12-12 02:14:32 +01:00
|
|
|
|
{
|
|
|
|
|
mmd_t *code; // Current code block
|
|
|
|
|
linefrag_t frag; // Line fragment
|
|
|
|
|
double margin_top; // Top margin
|
|
|
|
|
|
|
|
|
|
frag.x = left + 36.0;
|
|
|
|
|
frag.width = 0.0;
|
|
|
|
|
frag.height = SIZE_CODEBLOCK;
|
|
|
|
|
frag.imagenum = 0;
|
|
|
|
|
frag.ws = false;
|
|
|
|
|
frag.font = DOCFONT_MONOSPACE;
|
|
|
|
|
frag.color = DOCCOLOR_RED;
|
|
|
|
|
margin_top = SIZE_CODEBLOCK * LINE_HEIGHT;
|
|
|
|
|
|
|
|
|
|
for (code = mmdGetFirstChild(current); code; code = mmdGetNextSibling(code))
|
|
|
|
|
{
|
|
|
|
|
frag.text = mmdGetText(code);
|
|
|
|
|
|
|
|
|
|
render_line(dd, margin_top, SIZE_CODEBLOCK * LINE_HEIGHT, 1, &frag);
|
|
|
|
|
|
|
|
|
|
margin_top = 0.0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-12-10 22:35:12 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'output_cb()' - Write PDF data to the standard output...
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
static ssize_t // O - Number of bytes written
|
|
|
|
|
output_cb(void *output_cbdata, // I - Callback data (not used)
|
|
|
|
|
const void *buffer, // I - Buffer to write
|
|
|
|
|
size_t bytes) // I - Number of bytes to write
|
|
|
|
|
{
|
|
|
|
|
(void)output_cbdata;
|
|
|
|
|
|
|
|
|
|
return ((ssize_t)fwrite(buffer, 1, bytes, stdout));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// 'main()' - Convert markdown to PDF.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
int // O - Exit status
|
|
|
|
|
main(int argc, // I - Number of command-line arguments
|
|
|
|
|
char *argv[]) // I - Command-line arguments
|
|
|
|
|
{
|
|
|
|
|
docdata_t dd; // Document data
|
|
|
|
|
docfont_t fontface; // Current font
|
|
|
|
|
mmd_t *doc; // Markdown document
|
2024-12-11 00:53:51 +01:00
|
|
|
|
const char *value; // Metadata value
|
2024-12-10 22:35:12 +01:00
|
|
|
|
|
|
|
|
|
|
2024-12-12 02:14:32 +01:00
|
|
|
|
setbuf(stderr, NULL);
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
// Get the markdown file from the command-line...
|
|
|
|
|
if (argc != 2)
|
|
|
|
|
{
|
|
|
|
|
fputs("Usage: md2pdf FILENANE.md >FILENAME.pdf\n", stderr);
|
|
|
|
|
return (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((doc = mmdLoad(/*root*/NULL, argv[1])) == NULL)
|
|
|
|
|
return (1);
|
|
|
|
|
|
|
|
|
|
// Initialize the document data
|
|
|
|
|
memset(&dd, 0, sizeof(dd));
|
|
|
|
|
|
|
|
|
|
dd.media_box.x2 = PAGE_WIDTH;
|
|
|
|
|
dd.media_box.y2 = PAGE_LENGTH;
|
|
|
|
|
|
|
|
|
|
dd.crop_box.x1 = PAGE_LEFT;
|
|
|
|
|
dd.crop_box.y1 = PAGE_FOOTER;
|
|
|
|
|
dd.crop_box.x2 = PAGE_RIGHT;
|
|
|
|
|
dd.crop_box.y2 = PAGE_HEADER;
|
|
|
|
|
|
|
|
|
|
dd.art_box.x1 = PAGE_LEFT;
|
|
|
|
|
dd.art_box.y1 = PAGE_BOTTOM;
|
|
|
|
|
dd.art_box.x2 = PAGE_RIGHT;
|
|
|
|
|
dd.art_box.y2 = PAGE_TOP;
|
|
|
|
|
|
|
|
|
|
dd.title = mmdGetMetadata(doc, "title");
|
|
|
|
|
|
|
|
|
|
// Output a PDF file to the standard output...
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
setmode(1, O_BINARY); // Force binary output on Windows
|
|
|
|
|
#endif // _WIN32
|
|
|
|
|
|
|
|
|
|
if ((dd.pdf = pdfioFileCreateOutput(output_cb, /*output_cbdata*/NULL, /*version*/NULL, /*media_box*/NULL, /*crop_box*/NULL, /*error_cb*/NULL, /*error_data*/NULL)) == NULL)
|
|
|
|
|
return (1);
|
|
|
|
|
|
2024-12-11 00:53:51 +01:00
|
|
|
|
if ((value = mmdGetMetadata(doc, "author")) != NULL)
|
|
|
|
|
pdfioFileSetAuthor(dd.pdf, value);
|
|
|
|
|
|
|
|
|
|
if ((value = mmdGetMetadata(doc, "keywords")) != NULL)
|
|
|
|
|
pdfioFileSetKeywords(dd.pdf, value);
|
|
|
|
|
|
|
|
|
|
if ((value = mmdGetMetadata(doc, "subject")) != NULL)
|
|
|
|
|
pdfioFileSetSubject(dd.pdf, value);
|
|
|
|
|
else if ((value = mmdGetMetadata(doc, "copyright")) != NULL)
|
|
|
|
|
pdfioFileSetSubject(dd.pdf, value);
|
|
|
|
|
|
|
|
|
|
if (dd.title)
|
|
|
|
|
pdfioFileSetTitle(dd.pdf, dd.title);
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
// Add fonts...
|
|
|
|
|
for (fontface = DOCFONT_REGULAR; fontface < DOCFONT_MAX; fontface ++)
|
|
|
|
|
{
|
2024-12-11 00:41:23 +01:00
|
|
|
|
if ((dd.fonts[fontface] = pdfioFileCreateFontObjFromFile(dd.pdf, docfont_filenames[fontface], UNICODE_VALUE)) == NULL)
|
2024-12-10 22:35:12 +01:00
|
|
|
|
return (1);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-11 21:37:03 +01:00
|
|
|
|
// Add images...
|
|
|
|
|
add_images(&dd, doc);
|
|
|
|
|
|
2024-12-10 22:35:12 +01:00
|
|
|
|
// Parse the markdown document...
|
|
|
|
|
format_doc(&dd, doc, dd.art_box.x1, dd.art_box.x2);
|
|
|
|
|
|
|
|
|
|
// Close the PDF and return...
|
|
|
|
|
pdfioStreamClose(dd.st);
|
|
|
|
|
pdfioFileClose(dd.pdf);
|
|
|
|
|
|
|
|
|
|
mmdFree(doc);
|
|
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
|
}
|