#include "mupdf/pdf.h"
#include <ft2build.h>
#include FT_FREETYPE_H
#include FT_ADVANCES_H
#define ALLOWED_TEXT_POS_ERROR (0.001f)
typedef struct pdf_device_s pdf_device;
typedef struct gstate_s gstate;
struct gstate_s
{
fz_buffer *buf;
void (*on_pop)(fz_context*,pdf_device*,void *);
void *on_pop_arg;
fz_colorspace *colorspace[2];
float color[2][4];
fz_matrix ctm;
fz_stroke_state *stroke_state;
float alpha[2];
int font;
float font_size;
float char_spacing;
float word_spacing;
float horizontal_scaling;
float leading;
int text_rendering_mode;
float rise;
int knockout;
fz_matrix tm;
};
typedef struct image_entry_s image_entry;
struct image_entry_s
{
char digest[16];
int id;
pdf_obj *ref;
};
typedef struct alpha_entry_s alpha_entry;
struct alpha_entry_s
{
float alpha;
int stroke;
};
typedef struct font_entry_s font_entry;
struct font_entry_s
{
fz_font *font;
};
typedef struct group_entry_s group_entry;
struct group_entry_s
{
int alpha;
int isolated;
int knockout;
fz_colorspace *colorspace;
pdf_obj *ref;
};
struct pdf_device_s
{
fz_device super;
pdf_document *doc;
pdf_obj *contents;
pdf_obj *resources;
fz_buffer *buffer;
int in_text;
int num_forms;
int num_smasks;
int num_gstates;
int max_gstates;
gstate *gstates;
int num_imgs;
int max_imgs;
image_entry *images;
int num_alphas;
int max_alphas;
alpha_entry *alphas;
int num_fonts;
int max_fonts;
font_entry *fonts;
int num_groups;
int max_groups;
group_entry *groups;
};
#define CURRENT_GSTATE(pdev) (&(pdev)->gstates[(pdev)->num_gstates-1])
static int
send_image(fz_context *ctx, pdf_device *pdev, fz_image *image, int mask, int smask)
{
fz_pixmap *pixmap = NULL;
pdf_obj *imobj = NULL;
pdf_obj *imref = NULL;
fz_compressed_buffer *cbuffer = NULL;
fz_compression_params *cp = NULL;
fz_buffer *buffer = NULL;
int i, num;
fz_md5 state;
unsigned char digest[16];
fz_colorspace *colorspace = image->colorspace;
pdf_document *doc = pdev->doc;
cbuffer = image->buffer;
fz_var(pixmap);
fz_var(buffer);
fz_var(imobj);
fz_var(imref);
fz_try(ctx)
{
if (cbuffer != NULL && cbuffer->params.type != FZ_IMAGE_PNG && cbuffer->params.type != FZ_IMAGE_TIFF)
{
buffer = fz_keep_buffer(ctx, cbuffer->buffer);
cp = &cbuffer->params;
}
else
{
unsigned int size;
int n;
pixmap = image->get_pixmap(ctx, image, image->w, image->h);
colorspace = pixmap->colorspace;
n = (pixmap->n == 1 ? 1 : pixmap->n-1);
size = image->w * image->h * n;
buffer = fz_new_buffer(ctx, size);
buffer->len = size;
if (pixmap->n == 1)
{
memcpy(buffer->data, pixmap->samples, size);
}
else
{
unsigned char *d = buffer->data;
unsigned char *s = pixmap->samples;
int mod = n;
while (size--)
{
*d++ = *s++;
mod--;
if (mod == 0)
s++, mod = n;
}
}
}
fz_md5_init(&state);
fz_md5_update(&state, buffer->data, buffer->len);
fz_md5_final(&state, digest);
for(i=0; i < pdev->num_imgs; i++)
{
if (!memcmp(&digest, pdev->images[i].digest, sizeof(16)))
{
num = i;
break;
}
}
if (i < pdev->num_imgs)
break;
if (pdev->num_imgs == pdev->max_imgs)
{
int newmax = pdev->max_imgs * 2;
if (newmax == 0)
newmax = 4;
pdev->images = fz_resize_array(ctx, pdev->images, newmax, sizeof(*pdev->images));
pdev->max_imgs = newmax;
}
num = pdev->num_imgs++;
memcpy(pdev->images[num].digest,digest,16);
pdev->images[num].ref = NULL;
imobj = pdf_new_dict(ctx, doc, 3);
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Type, PDF_NAME_XObject);
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Subtype, PDF_NAME_Image);
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Width, pdf_new_int(ctx, doc, image->w));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Height, pdf_new_int(ctx, doc, image->h));
if (mask)
{}
else if (!colorspace || colorspace->n == 1)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_ColorSpace, PDF_NAME_DeviceGray);
else if (colorspace->n == 3)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_ColorSpace, PDF_NAME_DeviceRGB);
else if (colorspace->n == 4)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_ColorSpace, PDF_NAME_DeviceCMYK);
if (!mask)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_BitsPerComponent, pdf_new_int(ctx, doc, image->bpc));
switch (cp ? cp->type : FZ_IMAGE_UNKNOWN)
{
case FZ_IMAGE_UNKNOWN:
default:
break;
case FZ_IMAGE_JPEG:
if (cp->u.jpeg.color_transform != -1)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_ColorTransform, pdf_new_int(ctx, doc, cp->u.jpeg.color_transform));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_DCTDecode);
break;
case FZ_IMAGE_JPX:
if (cp->u.jpx.smask_in_data)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_SMaskInData, pdf_new_int(ctx, doc, cp->u.jpx.smask_in_data));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_JPXDecode);
break;
case FZ_IMAGE_FAX:
if (cp->u.fax.columns)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Columns, pdf_new_int(ctx, doc, cp->u.fax.columns));
if (cp->u.fax.rows)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Rows, pdf_new_int(ctx, doc, cp->u.fax.rows));
if (cp->u.fax.k)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_K, pdf_new_int(ctx, doc, cp->u.fax.k));
if (cp->u.fax.end_of_line)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_EndOfLine, pdf_new_int(ctx, doc, cp->u.fax.end_of_line));
if (cp->u.fax.encoded_byte_align)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_EncodedByteAlign, pdf_new_int(ctx, doc, cp->u.fax.encoded_byte_align));
if (cp->u.fax.end_of_block)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_EndOfBlock, pdf_new_int(ctx, doc, cp->u.fax.end_of_block));
if (cp->u.fax.black_is_1)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_BlackIs1, pdf_new_int(ctx, doc, cp->u.fax.black_is_1));
if (cp->u.fax.damaged_rows_before_error)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_DamagedRowsBeforeError, pdf_new_int(ctx, doc, cp->u.fax.damaged_rows_before_error));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_CCITTFaxDecode);
break;
case FZ_IMAGE_JBIG2:
cp->type = FZ_IMAGE_UNKNOWN;
break;
case FZ_IMAGE_FLATE:
if (cp->u.flate.columns)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Columns, pdf_new_int(ctx, doc, cp->u.flate.columns));
if (cp->u.flate.colors)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Colors, pdf_new_int(ctx, doc, cp->u.flate.colors));
if (cp->u.flate.predictor)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Predictor, pdf_new_int(ctx, doc, cp->u.flate.predictor));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_FlateDecode);
pdf_dict_put_drop(ctx, imobj, PDF_NAME_BitsPerComponent, pdf_new_int(ctx, doc, image->bpc));
break;
case FZ_IMAGE_LZW:
if (cp->u.lzw.columns)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Columns, pdf_new_int(ctx, doc, cp->u.lzw.columns));
if (cp->u.lzw.colors)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Colors, pdf_new_int(ctx, doc, cp->u.lzw.colors));
if (cp->u.lzw.predictor)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Predictor, pdf_new_int(ctx, doc, cp->u.lzw.predictor));
if (cp->u.lzw.early_change)
pdf_dict_put_drop(ctx, imobj, PDF_NAME_EarlyChange, pdf_new_int(ctx, doc, cp->u.lzw.early_change));
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_LZWDecode);
break;
case FZ_IMAGE_RLD:
pdf_dict_put_drop(ctx, imobj, PDF_NAME_Filter, PDF_NAME_RunLengthDecode);
break;
}
if (mask)
{
pdf_dict_put_drop(ctx, imobj, PDF_NAME_ImageMask, pdf_new_bool(ctx, doc, 1));
}
if (image->mask)
{
int smasknum = send_image(ctx, pdev, image->mask, 0, 1);
pdf_dict_put(ctx, imobj, PDF_NAME_SMask, pdev->images[smasknum].ref);
}
imref = pdf_new_ref(ctx, doc, imobj);
pdf_update_stream(ctx, doc, imref, buffer, 1);
{
char text[32];
snprintf(text, sizeof(text), "XObject/Img%d", num);
pdf_dict_putp(ctx, pdev->resources, text, imref);
}
pdev->images[num].ref = imref;
}
fz_always(ctx)
{
fz_drop_buffer(ctx, buffer);
pdf_drop_obj(ctx, imobj);
fz_drop_pixmap(ctx, pixmap);
}
fz_catch(ctx)
{
pdf_drop_obj(ctx, imref);
fz_rethrow(ctx);
}
return num;
}
static void
pdf_dev_stroke_state(fz_context *ctx, pdf_device *pdev, fz_stroke_state *stroke_state)
{
gstate *gs = CURRENT_GSTATE(pdev);
if (stroke_state == gs->stroke_state)
return;
if (gs->stroke_state && !memcmp(stroke_state, gs->stroke_state, sizeof(*stroke_state)))
return;
if (!gs->stroke_state || gs->stroke_state->linewidth != stroke_state->linewidth)
{
fz_buffer_printf(ctx, gs->buf, "%f w\n", stroke_state->linewidth);
}
if (!gs->stroke_state || gs->stroke_state->start_cap != stroke_state->start_cap)
{
int cap = stroke_state->start_cap;
if (cap == FZ_LINECAP_TRIANGLE)
cap = FZ_LINECAP_BUTT;
fz_buffer_printf(ctx, gs->buf, "%d J\n", cap);
}
if (!gs->stroke_state || gs->stroke_state->linejoin != stroke_state->linejoin)
{
int join = stroke_state->linejoin;
if (join == FZ_LINEJOIN_MITER_XPS)
join = FZ_LINEJOIN_MITER;
fz_buffer_printf(ctx, gs->buf, "%d j\n", join);
}
if (!gs->stroke_state || gs->stroke_state->miterlimit != stroke_state->miterlimit)
{
fz_buffer_printf(ctx, gs->buf, "%f M\n", stroke_state->miterlimit);
}
if (gs->stroke_state == NULL && stroke_state->dash_len == 0)
{}
else if (!gs->stroke_state || gs->stroke_state->dash_phase != stroke_state->dash_phase || gs->stroke_state->dash_len != stroke_state->dash_len ||
memcmp(gs->stroke_state->dash_list, stroke_state->dash_list, sizeof(float)*stroke_state->dash_len))
{
int i;
if (stroke_state->dash_len == 0)
fz_buffer_printf(ctx, gs->buf, "[");
for (i = 0; i < stroke_state->dash_len; i++)
fz_buffer_printf(ctx, gs->buf, "%c%f", (i == 0 ? '[' : ' '), stroke_state->dash_list[i]);
fz_buffer_printf(ctx, gs->buf, "]%f d\n", stroke_state->dash_phase);
}
fz_drop_stroke_state(ctx, gs->stroke_state);
gs->stroke_state = fz_keep_stroke_state(ctx, stroke_state);
}
typedef struct
{
fz_context *ctx;
fz_buffer *buf;
} pdf_dev_path_arg;
static void
pdf_dev_path_moveto(fz_context *ctx, void *arg, float x, float y)
{
fz_buffer *buf = (fz_buffer *)arg;
fz_buffer_printf(ctx, buf, "%f %f m\n", x, y);
}
static void
pdf_dev_path_lineto(fz_context *ctx, void *arg, float x, float y)
{
fz_buffer *buf = (fz_buffer *)arg;
fz_buffer_printf(ctx, buf, "%f %f l\n", x, y);
}
static void
pdf_dev_path_curveto(fz_context *ctx, void *arg, float x1, float y1, float x2, float y2, float x3, float y3)
{
fz_buffer *buf = (fz_buffer *)arg;
fz_buffer_printf(ctx, buf, "%f %f %f %f %f %f c\n", x1, y1, x2, y2, x3, y3);
}
static void
pdf_dev_path_close(fz_context *ctx, void *arg)
{
fz_buffer *buf = (fz_buffer *)arg;
fz_buffer_printf(ctx, buf, "h\n");
}
static const fz_path_processor pdf_dev_path_proc =
{
pdf_dev_path_moveto,
pdf_dev_path_lineto,
pdf_dev_path_curveto,
pdf_dev_path_close
};
static void
pdf_dev_path(fz_context *ctx, pdf_device *pdev, fz_path *path)
{
gstate *gs = CURRENT_GSTATE(pdev);
fz_process_path(ctx, &pdf_dev_path_proc, (void *)gs->buf, path);
}
static void
pdf_dev_ctm(fz_context *ctx, pdf_device *pdev, const fz_matrix *ctm)
{
fz_matrix inverse;
gstate *gs = CURRENT_GSTATE(pdev);
if (memcmp(&gs->ctm, ctm, sizeof(*ctm)) == 0)
return;
fz_invert_matrix(&inverse, &gs->ctm);
fz_concat(&inverse, ctm, &inverse);
memcpy(&gs->ctm, ctm, sizeof(*ctm));
fz_buffer_printf(ctx, gs->buf, "%f %f %f %f %f %f cm\n", inverse.a, inverse.b, inverse.c, inverse.d, inverse.e, inverse.f);
}
static void
pdf_dev_color(fz_context *ctx, pdf_device *pdev, fz_colorspace *colorspace, float *color, int stroke)
{
int diff = 0;
int i;
int cspace = 0;
float rgb[FZ_MAX_COLORS];
gstate *gs = CURRENT_GSTATE(pdev);
if (colorspace == fz_device_gray(ctx))
cspace = 1;
else if (colorspace == fz_device_rgb(ctx))
cspace = 3;
else if (colorspace == fz_device_cmyk(ctx))
cspace = 4;
if (cspace == 0)
{
colorspace->to_rgb(ctx, colorspace, color, rgb);
color = rgb;
colorspace = fz_device_rgb(ctx);
}
if (gs->colorspace[stroke] != colorspace)
{
gs->colorspace[stroke] = colorspace;
diff = 1;
}
for (i=0; i < colorspace->n; i++)
if (gs->color[stroke][i] != color[i])
{
gs->color[stroke][i] = color[i];
diff = 1;
}
if (diff == 0)
return;
switch (cspace + stroke*8)
{
case 1:
fz_buffer_printf(ctx, gs->buf, "%f g\n", color[0]);
break;
case 3:
fz_buffer_printf(ctx, gs->buf, "%f %f %f rg\n", color[0], color[1], color[2]);
break;
case 4:
fz_buffer_printf(ctx, gs->buf, "%f %f %f %f k\n", color[0], color[1], color[2], color[3]);
break;
case 1+8:
fz_buffer_printf(ctx, gs->buf, "%f G\n", color[0]);
break;
case 3+8:
fz_buffer_printf(ctx, gs->buf, "%f %f %f RG\n", color[0], color[1], color[2]);
break;
case 4+8:
fz_buffer_printf(ctx, gs->buf, "%f %f %f %f K\n", color[0], color[1], color[2], color[3]);
break;
}
}
static void
pdf_dev_alpha(fz_context *ctx, pdf_device *pdev, float alpha, int stroke)
{
int i;
pdf_document *doc = pdev->doc;
gstate *gs = CURRENT_GSTATE(pdev);
if (gs->alpha[stroke] == alpha)
return;
for (i = 0; i < pdev->num_alphas; i++)
if (pdev->alphas[i].alpha == alpha && pdev->alphas[i].stroke == stroke)
break;
if (i == pdev->num_alphas)
{
pdf_obj *o;
pdf_obj *ref = NULL;
fz_var(ref);
if (pdev->num_alphas == pdev->max_alphas)
{
int newmax = pdev->max_alphas * 2;
if (newmax == 0)
newmax = 4;
pdev->alphas = fz_resize_array(ctx, pdev->alphas, newmax, sizeof(*pdev->alphas));
pdev->max_alphas = newmax;
}
pdev->alphas[i].alpha = alpha;
pdev->alphas[i].stroke = stroke;
o = pdf_new_dict(ctx, doc, 1);
fz_try(ctx)
{
char text[32];
pdf_dict_put_drop(ctx, o, (stroke ? PDF_NAME_CA : PDF_NAME_ca), pdf_new_real(ctx, doc, alpha));
ref = pdf_new_ref(ctx, doc, o);
snprintf(text, sizeof(text), "ExtGState/Alp%d", i);
pdf_dict_putp(ctx, pdev->resources, text, ref);
}
fz_always(ctx)
{
pdf_drop_obj(ctx, o);
pdf_drop_obj(ctx, ref);
}
fz_catch(ctx)
{
fz_rethrow(ctx);
}
pdev->num_alphas++;
}
fz_buffer_printf(ctx, gs->buf, "/Alp%d gs\n", i);
}
static void
pdf_dev_font(fz_context *ctx, pdf_device *pdev, fz_font *font, float size)
{
int i;
pdf_document *doc = pdev->doc;
gstate *gs = CURRENT_GSTATE(pdev);
if (gs->font >= 0 && pdev->fonts[gs->font].font == font)
return;
if (font->ft_buffer != NULL || font->ft_substitute)
fz_throw(ctx, FZ_ERROR_GENERIC, "pdf device supports only base 14 fonts currently");
for (i = 0; i < pdev->num_fonts; i++)
if (pdev->fonts[i].font == font)
break;
if (i == pdev->num_fonts)
{
pdf_obj *o;
pdf_obj *ref = NULL;
fz_var(ref);
if (pdev->num_fonts == pdev->max_fonts)
{
int newmax = pdev->max_fonts * 2;
if (newmax == 0)
newmax = 4;
pdev->fonts = fz_resize_array(ctx, pdev->fonts, newmax, sizeof(*pdev->fonts));
pdev->max_fonts = newmax;
}
pdev->fonts[i].font = fz_keep_font(ctx, font);
o = pdf_new_dict(ctx, doc, 3);
fz_try(ctx)
{
char text[32];
pdf_dict_put_drop(ctx, o, PDF_NAME_Type, PDF_NAME_Font);
pdf_dict_put_drop(ctx, o, PDF_NAME_Subtype, PDF_NAME_Type1);
pdf_dict_put_drop(ctx, o, PDF_NAME_BaseFont, pdf_new_name(ctx, doc, font->name));
pdf_dict_put_drop(ctx, o, PDF_NAME_Encoding, PDF_NAME_WinAnsiEncoding);
ref = pdf_new_ref(ctx, doc, o);
snprintf(text, sizeof(text), "Font/F%d", i);
pdf_dict_putp(ctx, pdev->resources, text, ref);
}
fz_always(ctx)
{
pdf_drop_obj(ctx, o);
pdf_drop_obj(ctx, ref);
}
fz_catch(ctx)
{
fz_rethrow(ctx);
}
pdev->num_fonts++;
}
fz_buffer_printf(ctx, gs->buf, "/F%d %f Tf\n", i, size);
}
static void
pdf_dev_tm(fz_context *ctx, pdf_device *pdev, const fz_matrix *tm)
{
gstate *gs = CURRENT_GSTATE(pdev);
if (memcmp(&gs->tm, tm, sizeof(*tm)) == 0)
return;
fz_buffer_printf(ctx, gs->buf, "%f %f %f %f %f %f Tm\n", tm->a, tm->b, tm->c, tm->d, tm->e, tm->f);
gs->tm = *tm;
}
static void
pdf_dev_push_new_buf(fz_context *ctx, pdf_device *pdev, fz_buffer *buf, void (*on_pop)(fz_context*,pdf_device*,void*), void *on_pop_arg)
{
if (pdev->num_gstates == pdev->max_gstates)
{
int newmax = pdev->max_gstates*2;
pdev->gstates = fz_resize_array(ctx, pdev->gstates, newmax, sizeof(*pdev->gstates));
pdev->max_gstates = newmax;
}
memcpy(&pdev->gstates[pdev->num_gstates], &pdev->gstates[pdev->num_gstates-1], sizeof(*pdev->gstates));
fz_keep_stroke_state(ctx, pdev->gstates[pdev->num_gstates].stroke_state);
if (buf)
pdev->gstates[pdev->num_gstates].buf = buf;
else
fz_keep_buffer(ctx, pdev->gstates[pdev->num_gstates].buf);
pdev->gstates[pdev->num_gstates].on_pop = on_pop;
pdev->gstates[pdev->num_gstates].on_pop_arg = on_pop_arg;
fz_buffer_printf(ctx, pdev->gstates[pdev->num_gstates].buf, "q\n");
pdev->num_gstates++;
}
static void
pdf_dev_push(fz_context *ctx, pdf_device *pdev)
{
pdf_dev_push_new_buf(ctx, pdev, NULL, NULL, NULL);
}
static void *
pdf_dev_pop(fz_context *ctx, pdf_device *pdev)
{
gstate *gs = CURRENT_GSTATE(pdev);
void *arg = gs->on_pop_arg;
fz_buffer_printf(ctx, gs->buf, "Q\n");
if (gs->on_pop)
gs->on_pop(ctx, pdev, arg);
pdev->num_gstates--;
fz_drop_stroke_state(ctx, pdev->gstates[pdev->num_gstates].stroke_state);
fz_drop_buffer(ctx, pdev->gstates[pdev->num_gstates].buf);
return arg;
}
static void
pdf_dev_text(fz_context *ctx, pdf_device *pdev, fz_text *text, float size)
{
int mask = FT_LOAD_NO_SCALE | FT_LOAD_IGNORE_TRANSFORM;
int i;
fz_matrix trm;
fz_matrix inverse;
gstate *gs = CURRENT_GSTATE(pdev);
fz_matrix trunc_trm;
trm = gs->tm;
trunc_trm.a = trm.a;
trunc_trm.b = trm.b;
trunc_trm.c = trm.c;
trunc_trm.d = trm.d;
trunc_trm.e = 0;
trunc_trm.f = 0;
fz_invert_matrix(&inverse, &trunc_trm);
i = 0;
while (i < text->len)
{
fz_text_item *it = &text->items[i];
fz_point delta;
float x;
int j;
delta.x = it->x - trm.e;
delta.y = it->y - trm.f;
fz_transform_point(&delta, &inverse);
if (delta.x != 0 || delta.y != 0)
{
fz_buffer_printf(ctx, gs->buf, "%f %f Td ", delta.x, delta.y);
trm.e = it->x;
trm.f = it->y;
}
j = i+1;
if (text->font->ft_face)
{
x = it->x;
while (j < text->len)
{
FT_Fixed adv;
FT_Get_Advance(text->font->ft_face, text->items[j-1].gid, mask, &adv);
x += (float)adv * size /((FT_Face)text->font->ft_face)->units_per_EM;
if (fabs(x - text->items[j].x) > ALLOWED_TEXT_POS_ERROR || fabs(it->y - text->items[j].y) > ALLOWED_TEXT_POS_ERROR)
break;
j++;
}
}
fz_buffer_printf(ctx, gs->buf, "<");
for (; i < j; i++)
{
fz_buffer_printf(ctx, gs->buf, "%02x", text->items[i].ucs);
}
fz_buffer_printf(ctx, gs->buf, "> Tj\n");
}
gs->tm.e = trm.e;
gs->tm.f = trm.f;
}
static void
pdf_dev_trm(fz_context *ctx, pdf_device *pdev, int trm)
{
gstate *gs = CURRENT_GSTATE(pdev);
if (gs->text_rendering_mode == trm)
return;
gs->text_rendering_mode = trm;
fz_buffer_printf(ctx, gs->buf, "%d Tr\n", trm);
}
static void
pdf_dev_begin_text(fz_context *ctx, pdf_device *pdev, const fz_matrix *tm, int trm)
{
pdf_dev_trm(ctx, pdev, trm);
if (!pdev->in_text)
{
gstate *gs = CURRENT_GSTATE(pdev);
fz_buffer_printf(ctx, gs->buf, "BT\n");
gs->tm.a = 1;
gs->tm.b = 0;
gs->tm.c = 0;
gs->tm.d = 1;
gs->tm.e = 0;
gs->tm.f = 0;
pdev->in_text = 1;
}
pdf_dev_tm(ctx, pdev, tm);
}
static void
pdf_dev_end_text(fz_context *ctx, pdf_device *pdev)
{
gstate *gs = CURRENT_GSTATE(pdev);
if (!pdev->in_text)
return;
pdev->in_text = 0;
fz_buffer_printf(ctx, gs->buf, "ET\n");
}
static int
pdf_dev_new_form(fz_context *ctx, pdf_obj **form_ref, pdf_device *pdev, const fz_rect *bbox, int isolated, int knockout, float alpha, fz_colorspace *colorspace)
{
pdf_document *doc = pdev->doc;
int num;
pdf_obj *group_ref;
pdf_obj *group;
pdf_obj *form;
*form_ref = NULL;
for(num = 0; num < pdev->num_groups; num++)
{
group_entry *g = &pdev->groups[num];
if (g->isolated == isolated && g->knockout == knockout && g->alpha == alpha && g->colorspace == colorspace)
{
group_ref = pdev->groups[num].ref;
break;
}
}
if (num == pdev->num_groups)
{
if (pdev->num_groups == pdev->max_groups)
{
int newmax = pdev->max_groups * 2;
if (newmax == 0)
newmax = 4;
pdev->groups = fz_resize_array(ctx, pdev->groups, newmax, sizeof(*pdev->groups));
pdev->max_groups = newmax;
}
pdev->num_groups++;
pdev->groups[num].isolated = isolated;
pdev->groups[num].knockout = knockout;
pdev->groups[num].alpha = alpha;
pdev->groups[num].colorspace = fz_keep_colorspace(ctx, colorspace);
pdev->groups[num].ref = NULL;
group = pdf_new_dict(ctx, doc, 5);
fz_try(ctx)
{
pdf_dict_put_drop(ctx, group, PDF_NAME_Type, PDF_NAME_Group);
pdf_dict_put_drop(ctx, group, PDF_NAME_S, PDF_NAME_Transparency);
pdf_dict_put_drop(ctx, group, PDF_NAME_K, pdf_new_bool(ctx, doc, knockout));
pdf_dict_put_drop(ctx, group, PDF_NAME_I, pdf_new_bool(ctx, doc, isolated));
if (!colorspace)
{}
else if (colorspace->n == 1)
pdf_dict_put_drop(ctx, group, PDF_NAME_CS, PDF_NAME_DeviceGray);
else if (colorspace->n == 4)
pdf_dict_put_drop(ctx, group, PDF_NAME_CS, PDF_NAME_DeviceCMYK);
else
pdf_dict_put_drop(ctx, group, PDF_NAME_CS, PDF_NAME_DeviceRGB);
group_ref = pdev->groups[num].ref = pdf_new_ref(ctx, doc, group);
}
fz_always(ctx)
{
pdf_drop_obj(ctx, group);
}
fz_catch(ctx)
{
fz_rethrow(ctx);
}
}
form = pdf_new_dict(ctx, doc, 4);
fz_try(ctx)
{
pdf_dict_put_drop(ctx, form, PDF_NAME_Subtype, PDF_NAME_Form);
pdf_dict_put(ctx, form, PDF_NAME_Group, group_ref);
pdf_dict_put_drop(ctx, form, PDF_NAME_FormType, pdf_new_int(ctx, doc, 1));
pdf_dict_put_drop(ctx, form, PDF_NAME_BBox, pdf_new_rect(ctx, doc, bbox));
*form_ref = pdf_new_ref(ctx, doc, form);
}
fz_catch(ctx)
{
pdf_drop_obj(ctx, form);
fz_rethrow(ctx);
}
{
char text[32];
num = pdev->num_forms++;
snprintf(text, sizeof(text), "XObject/Fm%d", num);
pdf_dict_putp(ctx, pdev->resources, text, *form_ref);
}
return num;
}
static void
pdf_dev_fill_path(fz_context *ctx, fz_device *dev, fz_path *path, int even_odd, const fz_matrix *ctm,
fz_colorspace *colorspace, float *color, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
gstate *gs = CURRENT_GSTATE(pdev);
pdf_dev_end_text(ctx, pdev);
pdf_dev_alpha(ctx, pdev, alpha, 0);
pdf_dev_color(ctx, pdev, colorspace, color, 0);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_path(ctx, pdev, path);
fz_buffer_printf(ctx, gs->buf, (even_odd ? "f*\n" : "f\n"));
}
static void
pdf_dev_stroke_path(fz_context *ctx, fz_device *dev, fz_path *path, fz_stroke_state *stroke, const fz_matrix *ctm,
fz_colorspace *colorspace, float *color, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
gstate *gs = CURRENT_GSTATE(pdev);
pdf_dev_end_text(ctx, pdev);
pdf_dev_alpha(ctx, pdev, alpha, 1);
pdf_dev_color(ctx, pdev, colorspace, color, 1);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_stroke_state(ctx, pdev, stroke);
pdf_dev_path(ctx, pdev, path);
fz_buffer_printf(ctx, gs->buf, "S\n");
}
static void
pdf_dev_clip_path(fz_context *ctx, fz_device *dev, fz_path *path, const fz_rect *rect, int even_odd, const fz_matrix *ctm)
{
pdf_device *pdev = (pdf_device*)dev;
gstate *gs;
pdf_dev_end_text(ctx, pdev);
pdf_dev_push(ctx, pdev);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_path(ctx, pdev, path);
gs = CURRENT_GSTATE(pdev);
fz_buffer_printf(ctx, gs->buf, (even_odd ? "W* n\n" : "W n\n"));
}
static void
pdf_dev_clip_stroke_path(fz_context *ctx, fz_device *dev, fz_path *path, const fz_rect *rect, fz_stroke_state *stroke, const fz_matrix *ctm)
{
pdf_device *pdev = (pdf_device*)dev;
gstate *gs;
pdf_dev_end_text(ctx, pdev);
pdf_dev_push(ctx, pdev);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_path(ctx, pdev, path);
gs = CURRENT_GSTATE(pdev);
fz_buffer_printf(ctx, gs->buf, "W n\n");
}
static void
pdf_dev_fill_text(fz_context *ctx, fz_device *dev, fz_text *text, const fz_matrix *ctm,
fz_colorspace *colorspace, float *color, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
fz_matrix trm = text->trm;
float size = fz_matrix_expansion(&trm);
fz_pre_scale(&trm, 1/size, 1/size);
pdf_dev_begin_text(ctx, pdev, &trm, 0);
pdf_dev_font(ctx, pdev, text->font, size);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_alpha(ctx, pdev, alpha, 0);
pdf_dev_color(ctx, pdev, colorspace, color, 0);
pdf_dev_text(ctx, pdev, text, size);
}
static void
pdf_dev_stroke_text(fz_context *ctx, fz_device *dev, fz_text *text, fz_stroke_state *stroke, const fz_matrix *ctm,
fz_colorspace *colorspace, float *color, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
fz_matrix trm = text->trm;
float size = fz_matrix_expansion(&trm);
fz_pre_scale(&trm, 1/size, 1/size);
pdf_dev_begin_text(ctx, pdev, &text->trm, 1);
pdf_dev_font(ctx, pdev, text->font, 1);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_alpha(ctx, pdev, alpha, 1);
pdf_dev_color(ctx, pdev, colorspace, color, 1);
pdf_dev_text(ctx, pdev, text, size);
}
static void
pdf_dev_clip_text(fz_context *ctx, fz_device *dev, fz_text *text, const fz_matrix *ctm, int accumulate)
{
pdf_device *pdev = (pdf_device*)dev;
fz_matrix trm = text->trm;
float size = fz_matrix_expansion(&trm);
fz_pre_scale(&trm, 1/size, 1/size);
pdf_dev_begin_text(ctx, pdev, &text->trm, 0);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_font(ctx, pdev, text->font, 7);
pdf_dev_text(ctx, pdev, text, size);
}
static void
pdf_dev_clip_stroke_text(fz_context *ctx, fz_device *dev, fz_text *text, fz_stroke_state *stroke, const fz_matrix *ctm)
{
pdf_device *pdev = (pdf_device*)dev;
fz_matrix trm = text->trm;
float size = fz_matrix_expansion(&trm);
fz_pre_scale(&trm, 1/size, 1/size);
pdf_dev_begin_text(ctx, pdev, &text->trm, 0);
pdf_dev_font(ctx, pdev, text->font, 5);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_text(ctx, pdev, text, size);
}
static void
pdf_dev_ignore_text(fz_context *ctx, fz_device *dev, fz_text *text, const fz_matrix *ctm)
{
pdf_device *pdev = (pdf_device*)dev;
fz_matrix trm = text->trm;
float size = fz_matrix_expansion(&trm);
fz_pre_scale(&trm, 1/size, 1/size);
pdf_dev_begin_text(ctx, pdev, &text->trm, 0);
pdf_dev_ctm(ctx, pdev, ctm);
pdf_dev_font(ctx, pdev, text->font, 3);
pdf_dev_text(ctx, pdev, text, size);
}
static void
pdf_dev_fill_image(fz_context *ctx, fz_device *dev, fz_image *image, const fz_matrix *ctm, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
int num;
gstate *gs = CURRENT_GSTATE(pdev);
fz_matrix local_ctm = *ctm;
pdf_dev_end_text(ctx, pdev);
num = send_image(ctx, pdev, image, 0, 0);
pdf_dev_alpha(ctx, pdev, alpha, 0);
fz_pre_scale(&local_ctm, 1, -1);
fz_pre_translate(&local_ctm, 0, -1);
pdf_dev_ctm(ctx, pdev, &local_ctm);
fz_buffer_printf(ctx, gs->buf, "/Img%d Do\n", num);
}
static void
pdf_dev_fill_shade(fz_context *ctx, fz_device *dev, fz_shade *shade, const fz_matrix *ctm, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_dev_end_text(ctx, pdev);
}
static void
pdf_dev_fill_image_mask(fz_context *ctx, fz_device *dev, fz_image *image, const fz_matrix *ctm,
fz_colorspace *colorspace, float *color, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
gstate *gs = CURRENT_GSTATE(pdev);
int num;
fz_matrix local_ctm = *ctm;
pdf_dev_end_text(ctx, pdev);
num = send_image(ctx, pdev, image, 1, 0);
fz_buffer_printf(ctx, gs->buf, "q\n");
pdf_dev_alpha(ctx, pdev, alpha, 0);
pdf_dev_color(ctx, pdev, colorspace, color, 0);
fz_pre_scale(&local_ctm, 1, -1);
fz_pre_translate(&local_ctm, 0, -1);
pdf_dev_ctm(ctx, pdev, &local_ctm);
fz_buffer_printf(ctx, gs->buf, "/Img%d Do Q\n", num);
}
static void
pdf_dev_clip_image_mask(fz_context *ctx, fz_device *dev, fz_image *image, const fz_rect *rect, const fz_matrix *ctm)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_dev_end_text(ctx, pdev);
pdf_dev_push(ctx, pdev);
}
static void
pdf_dev_pop_clip(fz_context *ctx, fz_device *dev)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_dev_end_text(ctx, pdev);
pdf_dev_pop(ctx, pdev);
}
static void
pdf_dev_begin_mask(fz_context *ctx, fz_device *dev, const fz_rect *bbox, int luminosity, fz_colorspace *colorspace, float *color)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_document *doc = pdev->doc;
gstate *gs;
pdf_obj *smask = NULL;
pdf_obj *egs = NULL;
pdf_obj *egs_ref;
pdf_obj *form_ref;
pdf_obj *color_obj = NULL;
int i;
fz_var(smask);
fz_var(egs);
fz_var(color_obj);
pdf_dev_end_text(ctx, pdev);
pdf_dev_new_form(ctx, &form_ref, pdev, bbox, 0, 0, 1, colorspace);
fz_try(ctx)
{
smask = pdf_new_dict(ctx, doc, 4);
pdf_dict_put_drop(ctx, smask, PDF_NAME_Type, PDF_NAME_Mask);
pdf_dict_put_drop(ctx, smask, PDF_NAME_S, (luminosity ? PDF_NAME_Luminosity : PDF_NAME_Alpha));
pdf_dict_put(ctx, smask, PDF_NAME_G, form_ref);
color_obj = pdf_new_array(ctx, doc, colorspace->n);
for (i = 0; i < colorspace->n; i++)
pdf_array_push(ctx, color_obj, pdf_new_real(ctx, doc, color[i]));
pdf_dict_put_drop(ctx, smask, PDF_NAME_BC, color_obj);
color_obj = NULL;
egs = pdf_new_dict(ctx, doc, 5);
pdf_dict_put_drop(ctx, egs, PDF_NAME_Type, PDF_NAME_ExtGState);
pdf_dict_put_drop(ctx, egs, PDF_NAME_SMask, pdf_new_ref(ctx, doc, smask));
egs_ref = pdf_new_ref(ctx, doc, egs);
{
char text[32];
snprintf(text, sizeof(text), "ExtGState/SM%d", pdev->num_smasks++);
pdf_dict_putp(ctx, pdev->resources, text, egs_ref);
pdf_drop_obj(ctx, egs_ref);
}
gs = CURRENT_GSTATE(pdev);
fz_buffer_printf(ctx, gs->buf, "/SM%d gs\n", pdev->num_smasks-1);
}
fz_always(ctx)
{
pdf_drop_obj(ctx, smask);
}
fz_catch(ctx)
{
pdf_drop_obj(ctx, form_ref);
pdf_drop_obj(ctx, color_obj);
fz_rethrow(ctx);
}
pdf_dev_push_new_buf(ctx, pdev, fz_new_buffer(ctx, 1024), NULL, form_ref);
}
static void
pdf_dev_end_mask(fz_context *ctx, fz_device *dev)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_document *doc = pdev->doc;
gstate *gs = CURRENT_GSTATE(pdev);
fz_buffer *buf = fz_keep_buffer(ctx, gs->buf);
pdf_obj *form_ref = (pdf_obj *)gs->on_pop_arg;
pdf_dev_end_text(ctx, pdev);
fz_buffer_printf(ctx, buf, "Q\n");
pdf_update_stream(ctx, doc, form_ref, buf, 0);
fz_drop_buffer(ctx, buf);
gs->buf = fz_keep_buffer(ctx, gs[-1].buf);
gs->on_pop_arg = NULL;
pdf_drop_obj(ctx, form_ref);
fz_buffer_printf(ctx, gs->buf, "q\n");
}
static void
pdf_dev_begin_group(fz_context *ctx, fz_device *dev, const fz_rect *bbox, int isolated, int knockout, int blendmode, float alpha)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_document *doc = pdev->doc;
int num;
pdf_obj *form_ref;
gstate *gs;
pdf_dev_end_text(ctx, pdev);
num = pdf_dev_new_form(ctx, &form_ref, pdev, bbox, isolated, knockout, alpha, NULL);
{
char text[32];
pdf_obj *obj;
snprintf(text, sizeof(text), "ExtGState/BlendMode%d", blendmode);
obj = pdf_dict_getp(ctx, pdev->resources, text);
if (obj == NULL)
{
obj = pdf_new_dict(ctx, doc, 2);
pdf_dict_put_drop(ctx, obj, PDF_NAME_Type, PDF_NAME_ExtGState);
pdf_dict_put_drop(ctx, obj, PDF_NAME_BM, pdf_new_name(ctx, doc, fz_blendmode_name(blendmode)));
pdf_dict_putp_drop(ctx, pdev->resources, text, obj);
}
}
gs = CURRENT_GSTATE(pdev);
fz_buffer_printf(ctx, gs->buf, "/BlendMode%d gs /Fm%d Do\n", blendmode, num);
pdf_dev_push_new_buf(ctx, pdev, fz_new_buffer(ctx, 1024), NULL, form_ref);
}
static void
pdf_dev_end_group(fz_context *ctx, fz_device *dev)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_document *doc = pdev->doc;
gstate *gs = CURRENT_GSTATE(pdev);
fz_buffer *buf = fz_keep_buffer(ctx, gs->buf);
pdf_obj *form_ref;
pdf_dev_end_text(ctx, pdev);
form_ref = (pdf_obj *)pdf_dev_pop(ctx, pdev);
pdf_update_stream(ctx, doc, form_ref, buf, 0);
fz_drop_buffer(ctx, buf);
pdf_drop_obj(ctx, form_ref);
}
static int
pdf_dev_begin_tile(fz_context *ctx, fz_device *dev, const fz_rect *area, const fz_rect *view, float xstep, float ystep, const fz_matrix *ctm, int id)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_dev_end_text(ctx, pdev);
return 0;
}
static void
pdf_dev_end_tile(fz_context *ctx, fz_device *dev)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_dev_end_text(ctx, pdev);
}
static void
pdf_dev_drop_imp(fz_context *ctx, fz_device *dev)
{
pdf_device *pdev = (pdf_device*)dev;
pdf_document *doc = pdev->doc;
int i;
pdf_dev_end_text(ctx, pdev);
for (i = pdev->num_gstates-1; i >= 0; i--)
{
fz_drop_stroke_state(ctx, pdev->gstates[i].stroke_state);
}
for (i = pdev->num_fonts-1; i >= 0; i--)
{
fz_drop_font(ctx, pdev->fonts[i].font);
}
for (i = pdev->num_imgs-1; i >= 0; i--)
{
pdf_drop_obj(ctx, pdev->images[i].ref);
}
if (pdev->contents)
{
pdf_update_stream(ctx, doc, pdev->contents, pdev->gstates[0].buf, 0);
pdf_drop_obj(ctx, pdev->contents);
}
if (pdev->buffer != pdev->gstates[0].buf)
{
fz_drop_buffer(ctx, pdev->gstates[0].buf);
}
pdf_drop_obj(ctx, pdev->resources);
fz_free(ctx, pdev->images);
fz_free(ctx, pdev->alphas);
fz_free(ctx, pdev->gstates);
}
fz_device *pdf_new_pdf_device(fz_context *ctx, pdf_document *doc, pdf_obj *contents, pdf_obj *resources, const fz_matrix *ctm, fz_buffer *buf)
{
pdf_device *dev = fz_new_device(ctx, sizeof *dev);
dev->super.drop_imp = pdf_dev_drop_imp;
dev->super.fill_path = pdf_dev_fill_path;
dev->super.stroke_path = pdf_dev_stroke_path;
dev->super.clip_path = pdf_dev_clip_path;
dev->super.clip_stroke_path = pdf_dev_clip_stroke_path;
dev->super.fill_text = pdf_dev_fill_text;
dev->super.stroke_text = pdf_dev_stroke_text;
dev->super.clip_text = pdf_dev_clip_text;
dev->super.clip_stroke_text = pdf_dev_clip_stroke_text;
dev->super.ignore_text = pdf_dev_ignore_text;
dev->super.fill_shade = pdf_dev_fill_shade;
dev->super.fill_image = pdf_dev_fill_image;
dev->super.fill_image_mask = pdf_dev_fill_image_mask;
dev->super.clip_image_mask = pdf_dev_clip_image_mask;
dev->super.pop_clip = pdf_dev_pop_clip;
dev->super.begin_mask = pdf_dev_begin_mask;
dev->super.end_mask = pdf_dev_end_mask;
dev->super.begin_group = pdf_dev_begin_group;
dev->super.end_group = pdf_dev_end_group;
dev->super.begin_tile = pdf_dev_begin_tile;
dev->super.end_tile = pdf_dev_end_tile;
fz_try(ctx)
{
dev->buffer = fz_keep_buffer(ctx, buf);
if (!buf)
buf = fz_new_buffer(ctx, 256);
dev->doc = doc;
dev->contents = pdf_keep_obj(ctx, contents);
dev->resources = pdf_keep_obj(ctx, resources);
dev->gstates = fz_malloc_struct(ctx, gstate);
dev->gstates[0].buf = buf;
dev->gstates[0].ctm = *ctm;
dev->gstates[0].colorspace[0] = fz_device_gray(ctx);
dev->gstates[0].colorspace[1] = fz_device_gray(ctx);
dev->gstates[0].color[0][0] = 1;
dev->gstates[0].color[1][0] = 1;
dev->gstates[0].alpha[0] = 1.0;
dev->gstates[0].alpha[1] = 1.0;
dev->gstates[0].font = -1;
dev->gstates[0].horizontal_scaling = 100;
dev->num_gstates = 1;
dev->max_gstates = 1;
}
fz_catch(ctx)
{
if (dev->gstates && dev->buffer == NULL)
fz_drop_buffer(ctx, dev->gstates[0].buf);
fz_free(ctx, dev);
fz_rethrow(ctx);
}
return (fz_device*)dev;
}
fz_device *pdf_page_write(fz_context *ctx, pdf_document *doc, pdf_page *page)
{
pdf_obj *resources = pdf_dict_get(ctx, page->me, PDF_NAME_Resources);
fz_matrix ctm;
pdf_obj *obj;
fz_pre_translate(fz_scale(&ctm, 1, -1), 0, page->mediabox.y0-page->mediabox.y1);
if (resources == NULL)
{
resources = pdf_new_dict(ctx, doc, 0);
pdf_dict_put_drop(ctx, page->me, PDF_NAME_Resources, resources);
}
obj = pdf_new_dict(ctx, doc, 0);
fz_try(ctx)
{
pdf_obj *new_contents = pdf_new_ref(ctx, doc, obj);
pdf_dict_put(ctx, page->me, PDF_NAME_Contents, new_contents);
pdf_drop_obj(ctx, page->contents);
page->contents = new_contents;
}
fz_always(ctx)
{
pdf_drop_obj(ctx, obj);
}
fz_catch(ctx)
{
fz_rethrow(ctx);
}
return pdf_new_pdf_device(ctx, doc, page->contents, resources, &ctm, NULL);
}