75940ff56f2f9ecad8d6c54f2cdde14b7f11bd00
1 /** \file
2 * Native PDF import using libpoppler.
3 *
4 * Authors:
5 * miklos erdelyi
6 *
7 * Copyright (C) 2007 Authors
8 *
9 * Released under GNU GPL, read the file 'COPYING' for more information
10 *
11 */
13 #ifdef HAVE_CONFIG_H
14 # include <config.h>
15 #endif
17 #ifdef HAVE_POPPLER
19 #include "goo/GooString.h"
20 #include "ErrorCodes.h"
21 #include "GlobalParams.h"
22 #include "PDFDoc.h"
23 #include "Page.h"
24 #include "Catalog.h"
26 #include "pdf-input.h"
27 #include "extension/system.h"
28 #include "extension/input.h"
29 #include "svg-builder.h"
30 #include "pdf-parser.h"
32 #include "document-private.h"
34 #include <gtk/gtkdialog.h>
36 namespace Inkscape {
37 namespace Extension {
38 namespace Internal {
40 /**
41 * \brief The PDF import dialog
42 * FIXME: Probably this should be placed into src/ui/dialog
43 */
45 static Glib::ustring crop_setting_choices[] = {
46 Glib::ustring(_("media box")),
47 Glib::ustring(_("crop box")),
48 Glib::ustring(_("trim box")),
49 Glib::ustring(_("bleed box")),
50 Glib::ustring(_("art box"))
51 };
53 PdfImportDialog::PdfImportDialog(PDFDoc *doc)
54 {
56 _pdf_doc = doc;
58 cancelbutton = Gtk::manage(new class Gtk::Button(Gtk::StockID("gtk-cancel")));
59 okbutton = Gtk::manage(new class Gtk::Button(Gtk::StockID("gtk-ok")));
60 _labelSelect = Gtk::manage(new class Gtk::Label(_("Select page:")));
62 // Page number
63 Gtk::Adjustment *_pageNumberSpin_adj = Gtk::manage(
64 new class Gtk::Adjustment(1, 1, _pdf_doc->getNumPages(), 1, 10, 0));
65 _pageNumberSpin = Gtk::manage(new class Gtk::SpinButton(*_pageNumberSpin_adj, 1, 1));
66 hbox2 = Gtk::manage(new class Gtk::HBox(false, 0));
68 // Crop settings
69 _cropCheck = Gtk::manage(new class Gtk::CheckButton(_("Crop to:")));
70 _cropTypeCombo = Gtk::manage(new class Gtk::ComboBoxText());
71 int num_crop_choices = sizeof(crop_setting_choices) / sizeof(crop_setting_choices[0]);
72 for ( int i = 0 ; i < num_crop_choices ; i++ ) {
73 _cropTypeCombo->append_text(crop_setting_choices[i]);
74 }
75 _cropTypeCombo->set_active_text(crop_setting_choices[0]);
76 _cropTypeCombo->set_sensitive(false);
78 hbox3 = Gtk::manage(new class Gtk::HBox(false, 0));
79 vbox2 = Gtk::manage(new class Gtk::VBox(false, 0));
80 alignment3 = Gtk::manage(new class Gtk::Alignment(0.5, 0.5, 1, 1));
81 _labelPageSettings = Gtk::manage(new class Gtk::Label(_("Page Settings")));
82 _pageSettingsFrame = Gtk::manage(new class Gtk::Frame());
83 _labelPrecision = Gtk::manage(new class Gtk::Label(_("Precision of approximation for gradient meshes:")));
85 _fallbackPrecisionSlider_adj = Gtk::manage(new class Gtk::Adjustment(2, 1, 256, 1, 10, 10));
86 _fallbackPrecisionSlider = Gtk::manage(new class Gtk::HScale(*_fallbackPrecisionSlider_adj));
87 _fallbackPrecisionSlider->set_value(2.0);
88 _labelPrecisionComment = Gtk::manage(new class Gtk::Label(_("rough")));
89 hbox6 = Gtk::manage(new class Gtk::HBox(false, 0));
91 // Text options
92 _labelText = Gtk::manage(new class Gtk::Label(_("Text handling:")));
93 _textHandlingCombo = Gtk::manage(new class Gtk::ComboBoxText());
94 _textHandlingCombo->append_text(_("import text as text"));
95 _textHandlingCombo->set_active_text(_("import text as text"));
97 hbox5 = Gtk::manage(new class Gtk::HBox(false, 0));
98 _embedImagesCheck = Gtk::manage(new class Gtk::CheckButton(_("Embed images")));
99 vbox3 = Gtk::manage(new class Gtk::VBox(false, 0));
100 alignment4 = Gtk::manage(new class Gtk::Alignment(0.5, 0.5, 1, 1));
101 _labelImportSettings = Gtk::manage(new class Gtk::Label(_("Import Settings")));
102 _importSettingsFrame = Gtk::manage(new class Gtk::Frame());
103 vbox1 = Gtk::manage(new class Gtk::VBox(false, 0));
104 _previewArea = Gtk::manage(new class Gtk::DrawingArea());
105 hbox1 = Gtk::manage(new class Gtk::HBox(false, 0));
106 cancelbutton->set_flags(Gtk::CAN_FOCUS);
107 cancelbutton->set_flags(Gtk::CAN_DEFAULT);
108 cancelbutton->set_relief(Gtk::RELIEF_NORMAL);
109 okbutton->set_flags(Gtk::CAN_FOCUS);
110 okbutton->set_flags(Gtk::CAN_DEFAULT);
111 okbutton->set_relief(Gtk::RELIEF_NORMAL);
112 this->get_action_area()->property_layout_style().set_value(Gtk::BUTTONBOX_END);
113 _labelSelect->set_alignment(0.5,0.5);
114 _labelSelect->set_padding(0,0);
115 _labelSelect->set_justify(Gtk::JUSTIFY_LEFT);
116 _labelSelect->set_line_wrap(false);
117 _labelSelect->set_use_markup(false);
118 _labelSelect->set_selectable(false);
119 _pageNumberSpin->set_flags(Gtk::CAN_FOCUS);
120 _pageNumberSpin->set_update_policy(Gtk::UPDATE_ALWAYS);
121 _pageNumberSpin->set_numeric(true);
122 _pageNumberSpin->set_digits(0);
123 _pageNumberSpin->set_wrap(false);
124 hbox2->pack_start(*_labelSelect, Gtk::PACK_SHRINK, 0);
125 hbox2->pack_start(*_pageNumberSpin, Gtk::PACK_SHRINK, 4);
126 _cropCheck->set_flags(Gtk::CAN_FOCUS);
127 _cropCheck->set_relief(Gtk::RELIEF_NORMAL);
128 _cropCheck->set_mode(true);
129 _cropCheck->set_active(false);
130 _cropTypeCombo->set_border_width(1);
131 hbox3->pack_start(*_cropCheck, Gtk::PACK_SHRINK, 0);
132 hbox3->pack_start(*_cropTypeCombo, Gtk::PACK_SHRINK, 4);
133 vbox2->pack_start(*hbox2);
134 vbox2->pack_start(*hbox3);
135 alignment3->add(*vbox2);
136 _labelPageSettings->set_alignment(0.5,0.5);
137 _labelPageSettings->set_padding(0,0);
138 _labelPageSettings->set_justify(Gtk::JUSTIFY_LEFT);
139 _labelPageSettings->set_line_wrap(false);
140 _labelPageSettings->set_use_markup(true);
141 _labelPageSettings->set_selectable(false);
142 _pageSettingsFrame->set_border_width(4);
143 _pageSettingsFrame->set_shadow_type(Gtk::SHADOW_ETCHED_IN);
144 _pageSettingsFrame->set_label_align(0,0.5);
145 _pageSettingsFrame->add(*alignment3);
146 _pageSettingsFrame->set_label_widget(*_labelPageSettings);
147 _labelPrecision->set_alignment(0.5,0.5);
148 _labelPrecision->set_padding(0,0);
149 _labelPrecision->set_justify(Gtk::JUSTIFY_LEFT);
150 _labelPrecision->set_line_wrap(false);
151 _labelPrecision->set_use_markup(false);
152 _labelPrecision->set_selectable(false);
153 _fallbackPrecisionSlider->set_size_request(180,-1);
154 _fallbackPrecisionSlider->set_flags(Gtk::CAN_FOCUS);
155 _fallbackPrecisionSlider->set_update_policy(Gtk::UPDATE_CONTINUOUS);
156 _fallbackPrecisionSlider->set_inverted(false);
157 _fallbackPrecisionSlider->set_digits(1);
158 _fallbackPrecisionSlider->set_draw_value(true);
159 _fallbackPrecisionSlider->set_value_pos(Gtk::POS_TOP);
160 _labelPrecisionComment->set_size_request(90,-1);
161 _labelPrecisionComment->set_alignment(0.5,0.5);
162 _labelPrecisionComment->set_padding(0,0);
163 _labelPrecisionComment->set_justify(Gtk::JUSTIFY_LEFT);
164 _labelPrecisionComment->set_line_wrap(false);
165 _labelPrecisionComment->set_use_markup(false);
166 _labelPrecisionComment->set_selectable(false);
167 hbox6->pack_start(*_fallbackPrecisionSlider, Gtk::PACK_SHRINK, 4);
168 hbox6->pack_start(*_labelPrecisionComment, Gtk::PACK_SHRINK, 4);
169 _labelText->set_alignment(0.5,0.5);
170 _labelText->set_padding(0,0);
171 _labelText->set_justify(Gtk::JUSTIFY_LEFT);
172 _labelText->set_line_wrap(false);
173 _labelText->set_use_markup(false);
174 _labelText->set_selectable(false);
175 hbox5->pack_start(*_labelText, Gtk::PACK_SHRINK, 0);
176 hbox5->pack_start(*_textHandlingCombo, Gtk::PACK_SHRINK, 0);
177 _embedImagesCheck->set_flags(Gtk::CAN_FOCUS);
178 _embedImagesCheck->set_relief(Gtk::RELIEF_NORMAL);
179 _embedImagesCheck->set_mode(true);
180 _embedImagesCheck->set_active(true);
181 vbox3->pack_start(*_labelPrecision, Gtk::PACK_SHRINK, 4);
182 vbox3->pack_start(*hbox6, Gtk::PACK_SHRINK, 0);
183 vbox3->pack_start(*hbox5);
184 vbox3->pack_start(*_embedImagesCheck, Gtk::PACK_SHRINK, 0);
185 alignment4->add(*vbox3);
186 _labelImportSettings->set_alignment(0.5,0.5);
187 _labelImportSettings->set_padding(0,0);
188 _labelImportSettings->set_justify(Gtk::JUSTIFY_LEFT);
189 _labelImportSettings->set_line_wrap(false);
190 _labelImportSettings->set_use_markup(true);
191 _labelImportSettings->set_selectable(false);
192 _importSettingsFrame->set_border_width(4);
193 _importSettingsFrame->set_shadow_type(Gtk::SHADOW_ETCHED_IN);
194 _importSettingsFrame->set_label_align(0,0.5);
195 _importSettingsFrame->add(*alignment4);
196 _importSettingsFrame->set_label_widget(*_labelImportSettings);
197 vbox1->pack_start(*_pageSettingsFrame, Gtk::PACK_EXPAND_PADDING, 0);
198 vbox1->pack_start(*_importSettingsFrame, Gtk::PACK_EXPAND_PADDING, 0);
199 hbox1->pack_start(*vbox1);
200 hbox1->pack_start(*_previewArea, Gtk::PACK_EXPAND_WIDGET, 4);
201 this->get_vbox()->set_homogeneous(false);
202 this->get_vbox()->set_spacing(0);
203 this->get_vbox()->pack_start(*hbox1);
204 this->set_title(_("PDF Import Settings"));
205 this->set_modal(true);
206 this->property_window_position().set_value(Gtk::WIN_POS_NONE);
207 this->set_resizable(true);
208 this->property_destroy_with_parent().set_value(false);
209 this->set_has_separator(true);
210 this->add_action_widget(*cancelbutton, -6);
211 this->add_action_widget(*okbutton, -5);
212 cancelbutton->show();
213 okbutton->show();
214 _labelSelect->show();
215 _pageNumberSpin->show();
216 hbox2->show();
217 _cropCheck->show();
218 _cropTypeCombo->show();
219 hbox3->show();
220 vbox2->show();
221 alignment3->show();
222 _labelPageSettings->show();
223 _pageSettingsFrame->show();
224 _labelPrecision->show();
225 _fallbackPrecisionSlider->show();
226 _labelPrecisionComment->show();
227 hbox6->show();
228 _labelText->show();
229 _textHandlingCombo->show();
230 hbox5->show();
231 _embedImagesCheck->show();
232 vbox3->show();
233 alignment4->show();
234 _labelImportSettings->show();
235 _importSettingsFrame->show();
236 vbox1->show();
237 _previewArea->show();
238 hbox1->show();
240 // Connect signals
241 _previewArea->signal_expose_event().connect(sigc::mem_fun(*this, &PdfImportDialog::_onExposePreview));
242 _pageNumberSpin_adj->signal_value_changed().connect(sigc::mem_fun(*this, &PdfImportDialog::_onPageNumberChanged));
243 _cropCheck->signal_toggled().connect(sigc::mem_fun(*this, &PdfImportDialog::_onToggleCropping));
244 _fallbackPrecisionSlider_adj->signal_value_changed().connect(sigc::mem_fun(*this, &PdfImportDialog::_onPrecisionChanged));
246 // Create an OutputDev
247 _preview_output_dev = new CairoOutputDev();
248 _preview_output_dev->startDoc(_pdf_doc->getXRef());
250 // Set default preview size
251 _preview_width = 200;
252 _preview_height = 300;
254 // Init preview
255 _thumb_data = NULL;
256 _cairo_surface = NULL;
257 _pageNumberSpin_adj->set_value(1.0);
258 }
260 PdfImportDialog::~PdfImportDialog() {
261 if (_preview_output_dev) {
262 delete _preview_output_dev;
263 }
264 if (_cairo_surface) {
265 cairo_surface_destroy(_cairo_surface);
266 }
267 if (_thumb_data) {
268 delete _thumb_data;
269 }
270 }
272 bool PdfImportDialog::showDialog() {
273 show();
274 gint b = run();
275 hide();
276 if ( b == Gtk::RESPONSE_OK ) {
277 return TRUE;
278 } else {
279 return FALSE;
280 }
281 }
283 int PdfImportDialog::getSelectedPage() {
284 return _current_page;
285 }
287 /**
288 * \brief Retrieves the current settings into a repr which SvgBuilder will use
289 * for determining the behaviour desired by the user
290 */
291 void PdfImportDialog::getImportSettings(Inkscape::XML::Node *prefs) {
292 sp_repr_set_svg_double(prefs, "selectedPage", (double)_current_page);
293 if (_cropCheck->get_active()) {
294 Glib::ustring current_choice = _cropTypeCombo->get_active_text();
295 int num_crop_choices = sizeof(crop_setting_choices) / sizeof(crop_setting_choices[0]);
296 int i = 0;
297 for ( ; i < num_crop_choices ; i++ ) {
298 if ( current_choice == crop_setting_choices[i] ) {
299 break;
300 }
301 }
302 sp_repr_set_svg_double(prefs, "cropTo", (double)i);
303 } else {
304 sp_repr_set_svg_double(prefs, "cropTo", -1.0);
305 }
306 sp_repr_set_svg_double(prefs, "approximationPrecision",
307 _fallbackPrecisionSlider->get_value());
308 if (_embedImagesCheck->get_active()) {
309 prefs->setAttribute("embedImages", "1");
310 } else {
311 prefs->setAttribute("embedImages", "0");
312 }
313 }
315 /**
316 * \brief Redisplay the comment on the current approximation precision setting
317 * Evenly divides the interval of possible values between the available labels.
318 */
319 void PdfImportDialog::_onPrecisionChanged() {
321 static Glib::ustring precision_comments[] = {
322 Glib::ustring(_("rough")),
323 Glib::ustring(_("medium")),
324 Glib::ustring(_("fine")),
325 Glib::ustring(_("very fine"))
326 };
328 double min = _fallbackPrecisionSlider_adj->get_lower();
329 double max = _fallbackPrecisionSlider_adj->get_upper();
330 int num_intervals = sizeof(precision_comments) / sizeof(precision_comments[0]);
331 double interval_len = ( max - min ) / (double)num_intervals;
332 double value = _fallbackPrecisionSlider_adj->get_value();
333 int comment_idx = (int)floor( ( value - min ) / interval_len );
334 _labelPrecisionComment->set_label(precision_comments[comment_idx]);
335 }
337 void PdfImportDialog::_onToggleCropping() {
338 _cropTypeCombo->set_sensitive(_cropCheck->get_active());
339 }
341 void PdfImportDialog::_onPageNumberChanged() {
342 int page = _pageNumberSpin->get_value_as_int();
343 _current_page = CLAMP(page, 1, _pdf_doc->getCatalog()->getNumPages());
344 _setPreviewPage(_current_page);
345 }
347 /**
348 * \brief Copies image data from a Cairo surface to a pixbuf
349 *
350 * Borrowed from libpoppler, from the file poppler-page.cc
351 * Copyright (C) 2005, Red Hat, Inc.
352 *
353 */
354 static void copy_cairo_surface_to_pixbuf (cairo_surface_t *surface,
355 unsigned char *data,
356 GdkPixbuf *pixbuf)
357 {
358 int cairo_width, cairo_height, cairo_rowstride;
359 unsigned char *pixbuf_data, *dst, *cairo_data;
360 int pixbuf_rowstride, pixbuf_n_channels;
361 unsigned int *src;
362 int x, y;
364 cairo_width = cairo_image_surface_get_width (surface);
365 cairo_height = cairo_image_surface_get_height (surface);
366 cairo_rowstride = cairo_width * 4;
367 cairo_data = data;
369 pixbuf_data = gdk_pixbuf_get_pixels (pixbuf);
370 pixbuf_rowstride = gdk_pixbuf_get_rowstride (pixbuf);
371 pixbuf_n_channels = gdk_pixbuf_get_n_channels (pixbuf);
373 if (cairo_width > gdk_pixbuf_get_width (pixbuf))
374 cairo_width = gdk_pixbuf_get_width (pixbuf);
375 if (cairo_height > gdk_pixbuf_get_height (pixbuf))
376 cairo_height = gdk_pixbuf_get_height (pixbuf);
377 for (y = 0; y < cairo_height; y++)
378 {
379 src = (unsigned int *) (cairo_data + y * cairo_rowstride);
380 dst = pixbuf_data + y * pixbuf_rowstride;
381 for (x = 0; x < cairo_width; x++)
382 {
383 dst[0] = (*src >> 16) & 0xff;
384 dst[1] = (*src >> 8) & 0xff;
385 dst[2] = (*src >> 0) & 0xff;
386 if (pixbuf_n_channels == 4)
387 dst[3] = (*src >> 24) & 0xff;
388 dst += pixbuf_n_channels;
389 src++;
390 }
391 }
392 }
394 /**
395 * \brief Updates the preview area with the previously rendered thumbnail
396 */
397 bool PdfImportDialog::_onExposePreview(GdkEventExpose *event) {
399 Glib::RefPtr<Gdk::Pixbuf> thumb = Gdk::Pixbuf::create(Gdk::COLORSPACE_RGB, true,
400 8, _thumb_width, _thumb_height);
401 // Set background to white
402 thumb->fill(0xffffffff);
403 Glib::RefPtr<Gdk::Pixmap> back_pixmap = Gdk::Pixmap::create(_previewArea->get_window(),
404 _thumb_width, _thumb_height, -1);
405 back_pixmap->draw_pixbuf(Glib::RefPtr<Gdk::GC>(), thumb, 0, 0, 0, 0,
406 static_cast<int>(_thumb_width),
407 static_cast<int>(_thumb_height),
408 Gdk::RGB_DITHER_NONE, 0, 0);
409 _previewArea->get_window()->set_back_pixmap(back_pixmap, false);
410 _previewArea->get_window()->clear();
411 // Copy the thumbnail image from the Cairo surface
412 copy_cairo_surface_to_pixbuf(_cairo_surface, _thumb_data, thumb->gobj());
413 _previewArea->get_window()->draw_pixbuf(Glib::RefPtr<Gdk::GC>(), thumb,
414 0, 0, 0, 0, -1, -1,
415 Gdk::RGB_DITHER_NONE, 0, 0);
417 return true;
418 }
420 /**
421 * \brief Renders the given page's thumbnail using Cairo
422 */
423 void PdfImportDialog::_setPreviewPage(int page) {
425 _previewed_page = _pdf_doc->getCatalog()->getPage(page);
426 // TODO: When available, obtain the thumbnail from the PDF document itself
427 // Get page size by accounting for rotation
428 double width, height;
429 int rotate = _previewed_page->getRotate();
430 if ( rotate == 90 || rotate == 270 ) {
431 height = _previewed_page->getCropWidth();
432 width = _previewed_page->getCropHeight();
433 } else {
434 width = _previewed_page->getCropWidth();
435 height = _previewed_page->getCropHeight();
436 }
437 // Calculate the needed scaling for the page
438 double scale_x = (double)_preview_width / width;
439 double scale_y = (double)_preview_height / height;
440 double scale_factor = ( scale_x > scale_y ) ? scale_y : scale_x;
441 // Create new Cairo surface
442 _thumb_width = (int)ceil( width * scale_factor );
443 _thumb_height = (int)ceil( height * scale_factor );
444 int rowstride = _thumb_width * 4;
445 if (_thumb_data) {
446 delete _thumb_data;
447 }
448 _thumb_data = new unsigned char[ rowstride * _thumb_height ];
449 if (_cairo_surface) {
450 cairo_surface_destroy(_cairo_surface);
451 }
452 _cairo_surface = cairo_image_surface_create_for_data(_thumb_data,
453 CAIRO_FORMAT_ARGB32, _thumb_width, _thumb_height, rowstride);
454 cairo_t *cr = cairo_create(_cairo_surface);
455 cairo_scale(cr, scale_factor, scale_factor); // Use Cairo for resizing the image
456 _preview_output_dev->setCairo(cr);
457 // Render page
458 _previewed_page->displaySlice(_preview_output_dev,
459 72.0, 72.0, 0,
460 FALSE, /* useMediaBox */
461 TRUE, /* crop */
462 0, 0,
463 (int)ceil(_previewed_page->getCropWidth()),
464 (int)ceil(_previewed_page->getCropHeight()),
465 FALSE, /* printing */
466 _pdf_doc->getCatalog());
467 // Clean up
468 _preview_output_dev->setCairo(NULL);
469 cairo_destroy(cr);
470 // Redraw preview area
471 _previewArea->set_size_request(_preview_width, _preview_height);
472 _previewArea->queue_draw();
473 }
475 ////////////////////////////////////////////////////////////////////////////////
477 /**
478 * Parses the selected page of the given PDF document using PdfParser.
479 */
480 SPDocument *
481 PdfInput::open(::Inkscape::Extension::Input * mod, const gchar * uri) {
483 // Initialize the globalParams variable for poppler
484 if (!globalParams) {
485 globalParams = new GlobalParams();
486 }
487 GooString *filename_goo = new GooString(uri);
488 PDFDoc *pdf_doc = new PDFDoc(filename_goo, NULL, NULL, NULL); // TODO: Could ask for password
489 if (!pdf_doc->isOk()) {
490 int error = pdf_doc->getErrorCode();
491 delete pdf_doc;
492 if (error == errEncrypted) {
493 g_message("Document is encrypted.");
494 } else {
495 g_message("Failed to load document from data (error %d)", error);
496 }
498 return NULL;
499 }
500 PdfImportDialog *dlg = new PdfImportDialog(pdf_doc);
501 if (!dlg->showDialog()) {
502 delete dlg;
503 delete pdf_doc;
505 return NULL;
506 }
508 // Get needed page
509 int page_num = dlg->getSelectedPage();
510 Catalog *catalog = pdf_doc->getCatalog();
511 Page *page = catalog->getPage(page_num);
513 SPDocument *doc = sp_document_new(NULL, TRUE, TRUE);
514 bool saved = sp_document_get_undo_sensitive(doc);
515 sp_document_set_undo_sensitive(doc, false); // No need to undo in this temporary document
517 // Create builder
518 gchar *docname = g_path_get_basename(uri);
519 gchar *dot = g_strrstr(docname, ".");
520 if (dot) {
521 *dot = 0;
522 }
523 SvgBuilder *builder = new SvgBuilder(doc, docname, pdf_doc->getXRef());
525 // Get preferences
526 Inkscape::XML::Node *prefs = builder->getPreferences();
527 dlg->getImportSettings(prefs);
529 // Apply crop settings
530 PDFRectangle *clipToBox = NULL;
531 double crop_setting;
532 sp_repr_get_double(prefs, "cropTo", &crop_setting);
533 if ( crop_setting >= 0.0 ) { // Do page clipping
534 int crop_choice = (int)crop_setting;
535 switch (crop_choice) {
536 case 0: // Media box
537 clipToBox = page->getMediaBox();
538 break;
539 case 1: // Crop box
540 clipToBox = page->getCropBox();
541 break;
542 case 2: // Bleed box
543 clipToBox = page->getBleedBox();
544 break;
545 case 3: // Trim box
546 clipToBox = page->getTrimBox();
547 break;
548 case 4: // Art box
549 clipToBox = page->getArtBox();
550 break;
551 default:
552 break;
553 }
554 }
556 // Create parser
557 PdfParser *pdf_parser = new PdfParser(pdf_doc->getXRef(), builder, page_num-1, page->getRotate(),
558 page->getResourceDict(), page->getCropBox(), clipToBox);
560 // Set up approximation precision for parser
561 double color_delta;
562 sp_repr_get_double(prefs, "approximationPrecision", &color_delta);
563 if ( color_delta <= 0.0 ) {
564 color_delta = 1.0 / 2.0;
565 } else {
566 color_delta = 1.0 / color_delta;
567 }
568 for ( int i = 1 ; i <= pdfNumShadingTypes ; i++ ) {
569 pdf_parser->setApproximationPrecision(i, color_delta, 6);
570 }
572 // Parse the document structure
573 Object obj;
574 page->getContents(&obj);
575 if (!obj.isNull()) {
576 pdf_parser->parse(&obj);
577 }
579 // Cleanup
580 obj.free();
581 delete pdf_parser;
582 delete builder;
583 g_free(docname);
584 delete pdf_doc;
586 // Restore undo
587 sp_document_set_undo_sensitive(doc, saved);
589 return doc;
590 }
592 #include "../clear-n_.h"
594 void
595 PdfInput::init(void) {
596 Inkscape::Extension::Extension * ext;
598 /* PDF in */
599 ext = Inkscape::Extension::build_from_mem(
600 "<inkscape-extension>\n"
601 "<name>PDF Input</name>\n"
602 "<id>org.inkscape.input.pdf</id>\n"
603 "<input>\n"
604 "<extension>.pdf</extension>\n"
605 "<mimetype>application/pdf</mimetype>\n"
606 "<filetypename>Adobe PDF (*.pdf) [via poppler]</filetypename>\n"
607 "<filetypetooltip>Adobe Portable Document Format</filetypetooltip>\n"
608 "</input>\n"
609 "</inkscape-extension>", new PdfInput());
611 /* AI in */
612 ext = Inkscape::Extension::build_from_mem(
613 "<inkscape-extension>\n"
614 "<name>AI Input</name>\n"
615 "<id>org.inkscape.input.ai</id>\n"
616 "<input>\n"
617 "<extension>.ai</extension>\n"
618 "<mimetype>image/x-adobe-illustrator</mimetype>\n"
619 "<filetypename>Adobe Illustrator (*.ai) [PDF-based]</filetypename>\n"
620 "<filetypetooltip>Open files saved with recent versions of Adobe Illustrator</filetypetooltip>\n"
621 "</input>\n"
622 "</inkscape-extension>", new PdfInput());
623 } // init
625 } } } /* namespace Inkscape, Extension, Implementation */
627 #endif /* HAVE_POPPLER */
629 /*
630 Local Variables:
631 mode:c++
632 c-file-style:"stroustrup"
633 c-file-offsets:((innamespace . 0)(inline-open . 0))
634 indent-tabs-mode:nil
635 fill-column:99
636 End:
637 */
638 // vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4 :