]> www.fi.muni.cz Git - evince.git/blob - pdf/xpdf/Page.cc
Imported Xpdf 2.03 and fixed build.
[evince.git] / pdf / xpdf / Page.cc
1 //========================================================================
2 //
3 // Page.cc
4 //
5 // Copyright 1996-2003 Glyph & Cog, LLC
6 //
7 //========================================================================
8
9 #include <aconf.h>
10
11 #ifdef USE_GCC_PRAGMAS
12 #pragma implementation
13 #endif
14
15 #include <stddef.h>
16 #include "GlobalParams.h"
17 #include "Object.h"
18 #include "Array.h"
19 #include "Dict.h"
20 #include "XRef.h"
21 #include "Link.h"
22 #include "OutputDev.h"
23 #ifndef PDF_PARSER_ONLY
24 #include "Gfx.h"
25 #include "Annot.h"
26 #endif
27 #include "Error.h"
28 #include "Page.h"
29
30 //------------------------------------------------------------------------
31 // PageAttrs
32 //------------------------------------------------------------------------
33
34 PageAttrs::PageAttrs(PageAttrs *attrs, Dict *dict) {
35   Object obj1;
36   double w, h;
37
38   // get old/default values
39   if (attrs) {
40     mediaBox = attrs->mediaBox;
41     cropBox = attrs->cropBox;
42     haveCropBox = attrs->haveCropBox;
43     rotate = attrs->rotate;
44     attrs->resources.copy(&resources);
45   } else {
46     // set default MediaBox to 8.5" x 11" -- this shouldn't be necessary
47     // but some (non-compliant) PDF files don't specify a MediaBox
48     mediaBox.x1 = 0;
49     mediaBox.y1 = 0;
50     mediaBox.x2 = 612;
51     mediaBox.y2 = 792;
52     cropBox.x1 = cropBox.y1 = cropBox.x2 = cropBox.y2 = 0;
53     haveCropBox = gFalse;
54     rotate = 0;
55     resources.initNull();
56   }
57
58   // media box
59   readBox(dict, "MediaBox", &mediaBox);
60
61   // crop box
62   if (readBox(dict, "CropBox", &cropBox)) {
63     haveCropBox = gTrue;
64   }
65   if (!haveCropBox) {
66     cropBox = mediaBox;
67   }
68
69   // if the MediaBox is excessively larger than the CropBox,
70   // just use the CropBox
71   limitToCropBox = gFalse;
72   if (haveCropBox) {
73     w = 0.25 * (cropBox.x2 - cropBox.x1);
74     h = 0.25 * (cropBox.y2 - cropBox.y1);
75     if ((cropBox.x1 - mediaBox.x1) + (mediaBox.x2 - cropBox.x2) > w ||
76         (cropBox.y1 - mediaBox.y1) + (mediaBox.y2 - cropBox.y2) > h) {
77       limitToCropBox = gTrue;
78     }
79   }
80
81   // other boxes
82   bleedBox = cropBox;
83   readBox(dict, "BleedBox", &bleedBox);
84   trimBox = cropBox;
85   readBox(dict, "TrimBox", &trimBox);
86   artBox = cropBox;
87   readBox(dict, "ArtBox", &artBox);
88
89   // rotate
90   dict->lookup("Rotate", &obj1);
91   if (obj1.isInt()) {
92     rotate = obj1.getInt();
93   }
94   obj1.free();
95   while (rotate < 0) {
96     rotate += 360;
97   }
98   while (rotate >= 360) {
99     rotate -= 360;
100   }
101
102   // misc attributes
103   dict->lookup("LastModified", &lastModified);
104   dict->lookup("BoxColorInfo", &boxColorInfo);
105   dict->lookup("Group", &group);
106   dict->lookup("Metadata", &metadata);
107   dict->lookup("PieceInfo", &pieceInfo);
108   dict->lookup("SeparationInfo", &separationInfo);
109
110   // resource dictionary
111   dict->lookup("Resources", &obj1);
112   if (obj1.isDict()) {
113     resources.free();
114     obj1.copy(&resources);
115   }
116   obj1.free();
117 }
118
119 PageAttrs::~PageAttrs() {
120   lastModified.free();
121   boxColorInfo.free();
122   group.free();
123   metadata.free();
124   pieceInfo.free();
125   separationInfo.free();
126   resources.free();
127 }
128
129 GBool PageAttrs::readBox(Dict *dict, char *key, PDFRectangle *box) {
130   PDFRectangle tmp;
131   Object obj1, obj2;
132   GBool ok;
133
134   dict->lookup(key, &obj1);
135   if (obj1.isArray() && obj1.arrayGetLength() == 4) {
136     ok = gTrue;
137     obj1.arrayGet(0, &obj2);
138     if (obj2.isNum()) {
139       tmp.x1 = obj2.getNum();
140     } else {
141       ok = gFalse;
142     }
143     obj2.free();
144     obj1.arrayGet(1, &obj2);
145     if (obj2.isNum()) {
146       tmp.y1 = obj2.getNum();
147     } else {
148       ok = gFalse;
149     }
150     obj2.free();
151     obj1.arrayGet(2, &obj2);
152     if (obj2.isNum()) {
153       tmp.x2 = obj2.getNum();
154     } else {
155       ok = gFalse;
156     }
157     obj2.free();
158     obj1.arrayGet(3, &obj2);
159     if (obj2.isNum()) {
160       tmp.y2 = obj2.getNum();
161     } else {
162       ok = gFalse;
163     }
164     obj2.free();
165     if (ok) {
166       *box = tmp;
167     }
168   } else {
169     ok = gFalse;
170   }
171   obj1.free();
172   return ok;
173 }
174
175 //------------------------------------------------------------------------
176 // Page
177 //------------------------------------------------------------------------
178
179 Page::Page(XRef *xrefA, int numA, Dict *pageDict, PageAttrs *attrsA) {
180   ok = gTrue;
181   xref = xrefA;
182   num = numA;
183
184   // get attributes
185   attrs = attrsA;
186
187   // annotations
188   pageDict->lookupNF("Annots", &annots);
189   if (!(annots.isRef() || annots.isArray() || annots.isNull())) {
190     error(-1, "Page annotations object (page %d) is wrong type (%s)",
191           num, annots.getTypeName());
192     annots.free();
193     goto err2;
194   }
195
196   // contents
197   pageDict->lookupNF("Contents", &contents);
198   if (!(contents.isRef() || contents.isArray() ||
199         contents.isNull())) {
200     error(-1, "Page contents object (page %d) is wrong type (%s)",
201           num, contents.getTypeName());
202     contents.free();
203     goto err1;
204   }
205
206   // thumb
207   pageDict->lookupNF("Thumb", &thumb);
208   if (!(thumb.isStream() || thumb.isNull() || thumb.isRef())) {
209       error(-1, "Page thumb object (page %d) is wrong type (%s)",
210             num, thumb.getTypeName());
211       thumb.initNull(); 
212   }
213   
214   return;
215
216  err2:
217   annots.initNull();
218  err1:
219   contents.initNull();
220   ok = gFalse;
221 }
222
223 Page::~Page() {
224   delete attrs;
225   annots.free();
226   contents.free();
227 }
228
229 void Page::display(OutputDev *out, double hDPI, double vDPI, int rotate,
230                    Links *links, Catalog *catalog,
231                    GBool (*abortCheckCbk)(void *data),
232                    void *abortCheckCbkData,
233                    GBool (*annotDisplayDecideCbk)(Annot *annot, void *user_data),
234                    void *annotDisplayDecideCbkData) {
235   displaySlice(out, hDPI, vDPI, rotate, -1, -1, -1, -1, links, catalog,
236                abortCheckCbk, abortCheckCbkData,
237                annotDisplayDecideCbk, annotDisplayDecideCbkData);
238 }
239
240 void Page::displaySlice(OutputDev *out, double hDPI, double vDPI, int rotate,
241                         int sliceX, int sliceY, int sliceW, int sliceH,
242                         Links *links, Catalog *catalog,
243                         GBool (*abortCheckCbk)(void *data),
244                         void *abortCheckCbkData,
245                         GBool (*annotDisplayDecideCbk)(Annot *annot, void *user_data),
246                         void *annotDisplayDecideCbkData) {
247 #ifndef PDF_PARSER_ONLY
248   PDFRectangle *mediaBox, *cropBox;
249   PDFRectangle box;
250   Gfx *gfx;
251   Object obj;
252   Link *link;
253   Annots *annotList;
254   double kx, ky;
255   int i;
256
257   rotate += getRotate();
258   if (rotate >= 360) {
259     rotate -= 360;
260   } else if (rotate < 0) {
261     rotate += 360;
262   }
263
264   mediaBox = getBox();
265   if (sliceW >= 0 && sliceH >= 0) {
266     kx = 72.0 / hDPI;
267     ky = 72.0 / vDPI;
268     if (rotate == 90) {
269       if (out->upsideDown()) {
270         box.x1 = mediaBox->x1 + ky * sliceY;
271         box.x2 = mediaBox->x1 + ky * (sliceY + sliceH);
272       } else {
273         box.x1 = mediaBox->x2 - ky * (sliceY + sliceH);
274         box.x2 = mediaBox->x2 - ky * sliceY;
275       }
276       box.y1 = mediaBox->y1 + kx * sliceX;
277       box.y2 = mediaBox->y1 + kx * (sliceX + sliceW);
278     } else if (rotate == 180) {
279       box.x1 = mediaBox->x2 - kx * (sliceX + sliceW);
280       box.x2 = mediaBox->x2 - kx * sliceX;
281       if (out->upsideDown()) {
282         box.y1 = mediaBox->y1 + ky * sliceY;
283         box.y2 = mediaBox->y1 + ky * (sliceY + sliceH);
284       } else {
285         box.y1 = mediaBox->y2 - ky * (sliceY + sliceH);
286         box.y2 = mediaBox->y2 - ky * sliceY;
287       }
288     } else if (rotate == 270) {
289       if (out->upsideDown()) {
290         box.x1 = mediaBox->x2 - ky * (sliceY + sliceH);
291         box.x2 = mediaBox->x2 - ky * sliceY;
292       } else {
293         box.x1 = mediaBox->x1 + ky * sliceY;
294         box.x2 = mediaBox->x1 + ky * (sliceY + sliceH);
295       }
296       box.y1 = mediaBox->y2 - kx * (sliceX + sliceW);
297       box.y2 = mediaBox->y2 - kx * sliceX;
298     } else {
299       box.x1 = mediaBox->x1 + kx * sliceX;
300       box.x2 = mediaBox->x1 + kx * (sliceX + sliceW);
301       if (out->upsideDown()) {
302         box.y1 = mediaBox->y2 - ky * (sliceY + sliceH);
303         box.y2 = mediaBox->y2 - ky * sliceY;
304       } else {
305         box.y1 = mediaBox->y1 + ky * sliceY;
306         box.y2 = mediaBox->y1 + ky * (sliceY + sliceH);
307       }
308     }
309   } else {
310     box = *mediaBox;
311   }
312   cropBox = getCropBox();
313
314   if (globalParams->getPrintCommands()) {
315     printf("***** MediaBox = ll:%g,%g ur:%g,%g\n",
316            box.x1, box.y1, box.x2, box.y2);
317     if (isCropped()) {
318       printf("***** CropBox = ll:%g,%g ur:%g,%g\n",
319              cropBox->x1, cropBox->y1, cropBox->x2, cropBox->y2);
320     }
321     printf("***** Rotate = %d\n", attrs->getRotate());
322   }
323
324   gfx = new Gfx(xref, out, num, attrs->getResourceDict(),
325                 hDPI, vDPI, &box, isCropped(), cropBox, rotate,
326                 abortCheckCbk, abortCheckCbkData);
327   contents.fetch(xref, &obj);
328   if (!obj.isNull()) {
329     gfx->saveState();
330     gfx->display(&obj);
331     gfx->restoreState();
332   }
333   obj.free();
334
335   // draw links
336   if (links) {
337     gfx->saveState();
338     for (i = 0; i < links->getNumLinks(); ++i) {
339       link = links->getLink(i);
340       out->drawLink(link, catalog);
341     }
342     gfx->restoreState();
343     out->dump();
344   }
345
346   // draw non-link annotations
347   annotList = new Annots(xref, annots.fetch(xref, &obj));
348   obj.free();
349 #ifdef USE_ANNOTS_VIEW
350   if (annotList->getNumAnnots() > 0) {
351     if (globalParams->getPrintCommands()) {
352       printf("***** Annotations\n");
353     }
354     for (i = 0; i < annotList->getNumAnnots(); ++i) {
355         Annot *annot = annotList->getAnnot(i);
356         if ((annotDisplayDecideCbk &&
357              (*annotDisplayDecideCbk)(annot, annotDisplayDecideCbkData)) || 
358             !annotDisplayDecideCbk)
359           annot->draw(gfx); 
360     }
361     out->dump();
362   }
363 #endif
364   delete annotList;
365
366   delete gfx;
367 #endif
368 }