reactos/rosapps/smartpdf/fitz/mupdf/pdf_build.c
Klemens Friedl 435a566751 SmartPDF - lightweight pdf viewer app for rosapps
* sumatrapdf - vendor import
* everything compiles (libjpeg, poppler, fitz, sumatrapdf)
* does NOT link

(remove the comment tags in the parent directory.rbuild file (rosapps dir) to build it)

svn path=/trunk/; revision=29295
2007-09-29 08:39:35 +00:00

803 lines
16 KiB
C

#include <fitz.h>
#include <mupdf.h>
void
pdf_initgstate(pdf_gstate *gs)
{
gs->linewidth = 1.0;
gs->linecap = 0;
gs->linejoin = 0;
gs->miterlimit = 10;
gs->dashphase = 0;
gs->dashlen = 0;
memset(gs->dashlist, 0, sizeof(gs->dashlist));
gs->stroke.kind = PDF_MCOLOR;
gs->stroke.cs = fz_keepcolorspace(pdf_devicegray);
gs->stroke.v[0] = 0;
gs->stroke.indexed = nil;
gs->stroke.pattern = nil;
gs->stroke.shade = nil;
gs->fill.kind = PDF_MCOLOR;
gs->fill.cs = fz_keepcolorspace(pdf_devicegray);
gs->fill.v[0] = 0;
gs->fill.indexed = nil;
gs->fill.pattern = nil;
gs->fill.shade = nil;
gs->charspace = 0;
gs->wordspace = 0;
gs->scale = 1;
gs->leading = 0;
gs->font = nil;
gs->size = -1;
gs->render = 0;
gs->rise = 0;
gs->head = nil;
}
fz_error *
pdf_setcolorspace(pdf_csi *csi, int what, fz_colorspace *cs)
{
pdf_gstate *gs = csi->gstate + csi->gtop;
fz_error *error;
pdf_material *mat;
error = pdf_flushtext(csi);
if (error)
return error;
mat = what == PDF_MFILL ? &gs->fill : &gs->stroke;
fz_dropcolorspace(mat->cs);
mat->kind = PDF_MCOLOR;
mat->cs = fz_keepcolorspace(cs);
mat->v[0] = 0; /* FIXME: default color */
mat->v[1] = 0; /* FIXME: default color */
mat->v[2] = 0; /* FIXME: default color */
mat->v[3] = 1; /* FIXME: default color */
if (!strcmp(cs->name, "Indexed"))
{
mat->kind = PDF_MINDEXED;
mat->indexed = (pdf_indexed*)cs;
mat->cs = mat->indexed->base;
}
if (!strcmp(cs->name, "Lab"))
mat->kind = PDF_MLAB;
return nil;
}
fz_error *
pdf_setcolor(pdf_csi *csi, int what, float *v)
{
pdf_gstate *gs = csi->gstate + csi->gtop;
fz_error *error;
pdf_indexed *ind;
pdf_material *mat;
int i, k;
error = pdf_flushtext(csi);
if (error)
return error;
mat = what == PDF_MFILL ? &gs->fill : &gs->stroke;
switch (mat->kind)
{
case PDF_MPATTERN:
if (!strcmp(mat->cs->name, "Lab"))
goto Llab;
if (!strcmp(mat->cs->name, "Indexed"))
goto Lindexed;
/* fall through */
case PDF_MCOLOR:
for (i = 0; i < mat->cs->n; i++)
mat->v[i] = v[i];
break;
case PDF_MLAB:
Llab:
mat->v[0] = v[0] / 100.0;
mat->v[1] = (v[1] + 100) / 200.0;
mat->v[2] = (v[2] + 100) / 200.0;
break;
case PDF_MINDEXED:
Lindexed:
ind = mat->indexed;
i = CLAMP(v[0], 0, ind->high);
for (k = 0; k < ind->base->n; k++)
mat->v[k] = ind->lookup[ind->base->n * i + k] / 255.0;
break;
default:
return fz_throw("syntaxerror: color not compatible with material");
}
return nil;
}
fz_error *
pdf_setpattern(pdf_csi *csi, int what, pdf_pattern *pat, float *v)
{
pdf_gstate *gs = csi->gstate + csi->gtop;
fz_error *error;
pdf_material *mat;
error = pdf_flushtext(csi);
if (error)
return error;
if (what == PDF_MFILL)
mat = &gs->fill;
else
mat = &gs->stroke;
// TODO: this possibly drops a pattern too many times resulting in droping pattern
// used in other places, so don't drop it (better leak than crash).
// It's possible that overzeleaus dropping happens in some other place
#if 0
if (mat->pattern)
pdf_droppattern(mat->pattern);
#endif
mat->kind = PDF_MPATTERN;
if (pat)
mat->pattern = pdf_keeppattern(pat);
else
mat->pattern = nil;
if (v)
return pdf_setcolor(csi, what, v);
return nil;
}
fz_error *
pdf_setshade(pdf_csi *csi, int what, fz_shade *shade)
{
pdf_gstate *gs = csi->gstate + csi->gtop;
fz_error *error;
pdf_material *mat;
error = pdf_flushtext(csi);
if (error)
return error;
mat = what == PDF_MFILL ? &gs->fill : &gs->stroke;
if (mat->shade)
fz_dropshade(mat->shade);
mat->kind = PDF_MSHADE;
mat->shade = fz_keepshade(shade);
return nil;
}
fz_error *
pdf_buildstrokepath(pdf_gstate *gs, fz_pathnode *path)
{
fz_error *error;
fz_stroke stroke;
fz_dash *dash;
stroke.linecap = gs->linecap;
stroke.linejoin = gs->linejoin;
stroke.linewidth = gs->linewidth;
stroke.miterlimit = gs->miterlimit;
if (gs->dashlen)
{
error = fz_newdash(&dash, gs->dashphase, gs->dashlen, gs->dashlist);
if (error)
return error;
}
else
dash = nil;
error = fz_endpath(path, FZ_STROKE, &stroke, dash);
if (error) {
fz_dropdash(dash);
return error;
}
return nil;
}
fz_error *
pdf_buildfillpath(pdf_gstate *gs, fz_pathnode *path, int eofill)
{
return fz_endpath(path, eofill ? FZ_EOFILL : FZ_FILL, nil, nil);
}
static fz_error *
addcolorshape(pdf_gstate *gs, fz_node *shape, fz_colorspace *cs, float *v)
{
fz_error *error;
fz_node *mask;
fz_node *solid;
error = fz_newmasknode(&mask);
if (error) return error;
error = fz_newsolidnode(&solid, cs, cs->n, v);
if (error) return error;
fz_insertnodelast(mask, shape);
fz_insertnodelast(mask, solid);
fz_insertnodelast(gs->head, mask);
return nil;
}
static fz_error *
addinvisibleshape(pdf_gstate *gs, fz_node *shape)
{
fz_error *error;
fz_node *mask;
fz_pathnode *path;
error = fz_newmasknode(&mask);
if (error) return error;
error = fz_newpathnode(&path);
if (error) return error;
error = fz_endpath(path, FZ_FILL, nil, nil);
if (error) return error;
fz_insertnodelast(mask, (fz_node*)path);
fz_insertnodelast(mask, shape);
fz_insertnodelast(gs->head, mask);
return nil;
}
static fz_matrix getmatrix(fz_node *node)
{
if (node->parent)
{
fz_matrix ptm = getmatrix(node->parent);
if (fz_istransformnode(node))
return fz_concat(((fz_transformnode*)node)->m, ptm);
return ptm;
}
if (fz_istransformnode(node))
return ((fz_transformnode*)node)->m;
return fz_identity();
}
static fz_error *
addpatternshape(pdf_gstate *gs, fz_node *shape,
pdf_pattern *pat, fz_colorspace *cs, float *v)
{
fz_error *error;
fz_node *xform;
fz_node *over;
fz_node *mask;
fz_node *meta;
fz_node *link;
fz_matrix ctm;
fz_matrix inv;
fz_matrix ptm;
fz_rect bbox;
fz_obj *dict;
int x, y, x0, y0, x1, y1;
/* patterns are painted in user space */
ctm = getmatrix(gs->head);
inv = fz_invertmatrix(ctm);
error = fz_newmasknode(&mask);
if (error) return error;
ptm = fz_concat(pat->matrix, fz_invertmatrix(ctm));
error = fz_newtransformnode(&xform, ptm);
if (error) return error;
error = fz_packobj(&dict, "<< /Tree %p /XStep %f /YStep %f "
" /Matrix[%f %f %f %f %f %f] >>",
pat->tree, pat->xstep, pat->ystep,
pat->matrix.a, pat->matrix.b,
pat->matrix.c, pat->matrix.d,
pat->matrix.e, pat->matrix.f);
if (error) return error;
error = fz_newmetanode(&meta, "Pattern", dict);
if (error) return error;
error = fz_newovernode(&over);
if (error) return error;
fz_insertnodelast(mask, shape);
fz_insertnodelast(mask, meta);
fz_insertnodelast(meta, xform);
fz_insertnodelast(xform, over);
/* get bbox of shape in pattern space for stamping */
ptm = fz_concat(ctm, fz_invertmatrix(pat->matrix));
bbox = fz_boundnode(shape, ptm);
/* expand bbox by pattern bbox */
bbox.x0 += pat->bbox.x0;
bbox.y0 += pat->bbox.y0;
bbox.x1 += pat->bbox.x1;
bbox.y1 += pat->bbox.y1;
x0 = fz_floor(bbox.x0 / pat->xstep);
y0 = fz_floor(bbox.y0 / pat->ystep);
x1 = fz_ceil(bbox.x1 / pat->xstep);
y1 = fz_ceil(bbox.y1 / pat->ystep);
for (y = y0; y <= y1; y++)
{
for (x = x0; x <= x1; x++)
{
ptm = fz_translate(x * pat->xstep, y * pat->ystep);
error = fz_newtransformnode(&xform, ptm);
if (error) return error;
error = fz_newlinknode(&link, pat->tree);
if (error) return error;
fz_insertnodelast(xform, link);
fz_insertnodelast(over, xform);
}
}
if (pat->ismask)
return addcolorshape(gs, mask, cs, v);
fz_insertnodelast(gs->head, mask);
return nil;
}
fz_error *
pdf_addshade(pdf_gstate *gs, fz_shade *shade)
{
fz_error *error;
fz_node *node;
error = fz_newshadenode(&node, shade);
if (error) return error;
fz_insertnodelast(gs->head, node);
return nil;
}
static fz_error *
addshadeshape(pdf_gstate *gs, fz_node *shape, fz_shade *shade)
{
fz_error *error;
fz_node *mask;
fz_node *color;
fz_node *xform;
fz_node *over;
fz_node *bgnd;
fz_matrix ctm;
fz_matrix inv;
ctm = getmatrix(gs->head);
inv = fz_invertmatrix(ctm);
error = fz_newtransformnode(&xform, inv);
if (error) return error;
error = fz_newmasknode(&mask);
if (error) return error;
error = fz_newshadenode(&color, shade);
if (error) return error;
if (shade->usebackground)
{
error = fz_newovernode(&over);
if (error) return error;
error = fz_newsolidnode(&bgnd, shade->cs, shade->cs->n, shade->background);
if (error) return error;
fz_insertnodelast(mask, shape);
fz_insertnodelast(over, bgnd);
fz_insertnodelast(over, color);
fz_insertnodelast(xform, over);
fz_insertnodelast(mask, xform);
fz_insertnodelast(gs->head, mask);
}
else
{
fz_insertnodelast(mask, shape);
fz_insertnodelast(xform, color);
fz_insertnodelast(mask, xform);
fz_insertnodelast(gs->head, mask);
}
return nil;
}
fz_error *
pdf_addfillshape(pdf_gstate *gs, fz_node *shape)
{
switch (gs->fill.kind)
{
case PDF_MNONE:
fz_insertnodelast(gs->head, shape);
return nil;
case PDF_MCOLOR:
case PDF_MLAB:
case PDF_MINDEXED:
return addcolorshape(gs, shape, gs->fill.cs, gs->fill.v);
case PDF_MPATTERN:
// this is a work-around to make http://kpdf.kde.org/stuff/nytimes-firefox-final.pdf
// not crash see http://blog.kowalczyk.info/forum_sumatra/topic.php?TopicId=287&Posts=0
if (gs->fill.pattern)
return addpatternshape(gs, shape, gs->fill.pattern, gs->fill.cs, gs->fill.v);
else
return nil;
case PDF_MSHADE:
return addshadeshape(gs, shape, gs->fill.shade);
default:
return fz_throw("unimplemented material");
}
}
fz_error *
pdf_addstrokeshape(pdf_gstate *gs, fz_node *shape)
{
switch (gs->stroke.kind)
{
case PDF_MNONE:
fz_insertnodelast(gs->head, shape);
return nil;
case PDF_MCOLOR:
case PDF_MLAB:
case PDF_MINDEXED:
return addcolorshape(gs, shape, gs->stroke.cs, gs->stroke.v);
case PDF_MPATTERN:
return addpatternshape(gs, shape, gs->stroke.pattern, gs->stroke.cs, gs->stroke.v);
case PDF_MSHADE:
return addshadeshape(gs, shape, gs->stroke.shade);
default:
return fz_throw("unimplemented material");
}
}
fz_error *
pdf_addclipmask(pdf_gstate *gs, fz_node *shape)
{
fz_error *error;
fz_node *mask;
fz_node *over;
error = fz_newmasknode(&mask);
if (error) return error;
error = fz_newovernode(&over);
if (error) return error;
fz_insertnodelast(mask, shape);
fz_insertnodelast(mask, over);
fz_insertnodelast(gs->head, mask);
gs->head = over;
return nil;
}
fz_error *
pdf_addtransform(pdf_gstate *gs, fz_node *transform)
{
fz_error *error;
fz_node *over;
error = fz_newovernode(&over);
if (error) return error;
fz_insertnodelast(gs->head, transform);
fz_insertnodelast(transform, over);
gs->head = over;
return nil;
}
fz_error *
pdf_showimage(pdf_csi *csi, pdf_image *img)
{
fz_error *error;
fz_node *mask;
fz_node *color;
fz_node *shape;
error = fz_newimagenode(&color, (fz_image*)img);
if (error)
return error;
if (img->super.n == 0 && img->super.a == 1)
{
error = pdf_addfillshape(csi->gstate + csi->gtop, color);
if (error) {
fz_dropnode(color);
return error;
}
}
else
{
if (img->mask)
{
error = fz_newimagenode(&shape, (fz_image*)img->mask);
if (error) return error;
error = fz_newmasknode(&mask);
if (error) return error;
fz_insertnodelast(mask, shape);
fz_insertnodelast(mask, color);
fz_insertnodelast(csi->gstate[csi->gtop].head, mask);
}
else
{
fz_insertnodelast(csi->gstate[csi->gtop].head, color);
}
}
return nil;
}
fz_error *
pdf_showpath(pdf_csi *csi,
int doclose, int dofill, int dostroke, int evenodd)
{
pdf_gstate *gstate = csi->gstate + csi->gtop;
fz_error *error;
fz_pathnode *spath;
fz_pathnode *fpath;
if (doclose)
{
error = fz_closepath(csi->path);
if (error) return error;
}
if (dofill && dostroke)
{
fpath = csi->path;
error = fz_clonepathnode(&spath, fpath);
if (error) return error;
}
else
{
spath = fpath = csi->path;
}
if (dofill)
{
error = pdf_buildfillpath(gstate, fpath, evenodd);
if (error) return error;
error = pdf_addfillshape(gstate, (fz_node*)fpath);
if (error) return error;
}
if (dostroke)
{
error = pdf_buildstrokepath(gstate, spath);
if (error) return error;
error = pdf_addstrokeshape(gstate, (fz_node*)spath);
if (error) return error;
}
if (csi->clip)
{
fz_pathnode *clip;
error = fz_clonepathnode(&clip, csi->path);
if (error) return error;
error = fz_endpath(clip, FZ_FILL, nil, nil);
if (error) return error;
error = pdf_addclipmask(gstate, (fz_node*)clip);
if (error) return error;
csi->clip = 0;
}
if (!dofill && !dostroke)
{
fz_dropnode((fz_node*)csi->path);
}
csi->path = nil;
error = fz_newpathnode(&csi->path);
if (error) return error;
return nil;
}
fz_error *
pdf_flushtext(pdf_csi *csi)
{
pdf_gstate *gstate = csi->gstate + csi->gtop;
fz_error *error;
if (csi->text)
{
switch (csi->textmode)
{
case 0: /* fill */
case 1: /* stroke */
case 2: /* stroke + fill */
error = pdf_addfillshape(gstate, (fz_node*)csi->text);
if (error)
return error;
break;
case 3: /* invisible */
error = addinvisibleshape(gstate, (fz_node*)csi->text);
if (error)
return error;
break;
case 4: /* fill + clip */
case 5: /* stroke + clip */
case 6: /* stroke + fill + clip */
{
fz_textnode *temp;
error = fz_clonetextnode(&temp, csi->text);
if (error)
return error;
error = pdf_addfillshape(gstate, (fz_node*)temp);
if (error)
return error;
}
/* fall through */
case 7: /* invisible clip */
if (!csi->textclip)
{
error = fz_newovernode(&csi->textclip);
if (error)
return error;
}
fz_insertnodelast(csi->textclip, (fz_node*)csi->text);
break;
}
csi->text = nil;
}
return nil;
}
fz_error *
showglyph(pdf_csi *csi, int cid)
{
pdf_gstate *gstate = csi->gstate + csi->gtop;
pdf_font *font = gstate->font;
fz_error *error;
fz_matrix tsm, trm;
float w0, w1, tx, ty;
fz_hmtx h;
fz_vmtx v;
tsm.a = gstate->size * gstate->scale;
tsm.b = 0;
tsm.c = 0;
tsm.d = gstate->size;
tsm.e = 0;
tsm.f = gstate->rise;
if (font->super.wmode == 1)
{
v = fz_getvmtx((fz_font*)font, cid);
tsm.e -= v.x * gstate->size / 1000.0;
tsm.f -= v.y * gstate->size / 1000.0;
}
trm = fz_concat(tsm, csi->tm);
/* flush buffered text if face or matrix or rendermode has changed */
if (!csi->text ||
((fz_font*)font) != csi->text->font ||
fabs(trm.a - csi->text->trm.a) > FLT_EPSILON ||
fabs(trm.b - csi->text->trm.b) > FLT_EPSILON ||
fabs(trm.c - csi->text->trm.c) > FLT_EPSILON ||
fabs(trm.d - csi->text->trm.d) > FLT_EPSILON ||
gstate->render != csi->textmode)
{
error = pdf_flushtext(csi);
if (error) return error;
error = fz_newtextnode(&csi->text, (fz_font*)font);
if (error) return error;
csi->text->trm = trm;
csi->text->trm.e = 0;
csi->text->trm.f = 0;
csi->textmode = gstate->render;
}
/* add glyph to textobject */
error = fz_addtext(csi->text, cid, trm.e, trm.f);
if (error)
return error;
if (font->super.wmode == 0)
{
h = fz_gethmtx((fz_font*)font, cid);
w0 = h.w / 1000.0;
tx = (w0 * gstate->size + gstate->charspace) * gstate->scale;
csi->tm = fz_concat(fz_translate(tx, 0), csi->tm);
}
else
{
w1 = v.w / 1000.0;
ty = w1 * gstate->size + gstate->charspace;
csi->tm = fz_concat(fz_translate(0, ty), csi->tm);
}
return nil;
}
void
showspace(pdf_csi *csi, float tadj)
{
pdf_gstate *gstate = csi->gstate + csi->gtop;
pdf_font *font = gstate->font;
if (font->super.wmode == 0)
csi->tm = fz_concat(fz_translate(tadj * gstate->scale, 0), csi->tm);
else
csi->tm = fz_concat(fz_translate(0, tadj), csi->tm);
}
fz_error *
pdf_showtext(pdf_csi *csi, fz_obj *text)
{
pdf_gstate *gstate = csi->gstate + csi->gtop;
pdf_font *font = gstate->font;
fz_error *error;
unsigned char *buf;
unsigned char *end;
int i, len;
int cpt, cid;
if (fz_isarray(text))
{
for (i = 0; i < fz_arraylen(text); i++)
{
fz_obj *item = fz_arrayget(text, i);
if (fz_isstring(item))
{
error = pdf_showtext(csi, item);
if (error) return error;
}
else
{
showspace(csi, - fz_toreal(item) * gstate->size / 1000.0);
}
}
return nil;
}
buf = fz_tostrbuf(text);
len = fz_tostrlen(text);
end = buf + len;
while (buf < end)
{
buf = pdf_decodecmap(font->encoding, buf, &cpt);
cid = pdf_lookupcmap(font->encoding, cpt);
if (cid == -1)
cid = 0;
error = showglyph(csi, cid);
if (error)
return error;
if (cpt == 32)
showspace(csi, gstate->wordspace);
}
return nil;
}