2 #include "ev-job-queue.h"
3 #include "ev-document-thumbnails.h"
4 #include "ev-document-links.h"
5 #include "ev-document-images.h"
6 #include "ev-document-factory.h"
7 #include "ev-file-helpers.h"
8 #include "ev-document-fonts.h"
9 #include "ev-selection.h"
10 #include "ev-async-renderer.h"
11 #include "ev-file-exporter.h"
12 #include "ev-window.h"
14 #include <glib/gstdio.h>
16 #include <libgnomevfs/gnome-vfs-uri.h>
17 #include <libgnomevfs/gnome-vfs-utils.h>
18 #include <libgnomevfs/gnome-vfs-ops.h>
19 #include <libgnomevfs/gnome-vfs-xfer.h>
21 static void ev_job_init (EvJob *job);
22 static void ev_job_class_init (EvJobClass *class);
23 static void ev_job_links_init (EvJobLinks *job);
24 static void ev_job_links_class_init (EvJobLinksClass *class);
25 static void ev_job_render_init (EvJobRender *job);
26 static void ev_job_render_class_init (EvJobRenderClass *class);
27 static void ev_job_thumbnail_init (EvJobThumbnail *job);
28 static void ev_job_thumbnail_class_init (EvJobThumbnailClass *class);
29 static void ev_job_xfer_init (EvJobXfer *job);
30 static void ev_job_xfer_class_init (EvJobXferClass *class);
31 static void ev_job_print_init (EvJobPrint *job);
32 static void ev_job_print_class_init (EvJobPrintClass *class);
40 static guint job_signals[LAST_SIGNAL] = { 0 };
42 G_DEFINE_TYPE (EvJob, ev_job, G_TYPE_OBJECT)
43 G_DEFINE_TYPE (EvJobLinks, ev_job_links, EV_TYPE_JOB)
44 G_DEFINE_TYPE (EvJobRender, ev_job_render, EV_TYPE_JOB)
45 G_DEFINE_TYPE (EvJobThumbnail, ev_job_thumbnail, EV_TYPE_JOB)
46 G_DEFINE_TYPE (EvJobFonts, ev_job_fonts, EV_TYPE_JOB)
47 G_DEFINE_TYPE (EvJobXfer, ev_job_xfer, EV_TYPE_JOB)
48 G_DEFINE_TYPE (EvJobPrint, ev_job_print, EV_TYPE_JOB)
50 static void ev_job_init (EvJob *job) { /* Do Nothing */ }
53 ev_job_dispose (GObject *object)
57 job = EV_JOB (object);
60 g_object_unref (job->document);
64 (* G_OBJECT_CLASS (ev_job_parent_class)->dispose) (object);
68 ev_job_class_init (EvJobClass *class)
72 oclass = G_OBJECT_CLASS (class);
74 oclass->dispose = ev_job_dispose;
76 job_signals [FINISHED] =
77 g_signal_new ("finished",
80 G_STRUCT_OFFSET (EvJobClass, finished),
82 g_cclosure_marshal_VOID__VOID,
87 static void ev_job_links_init (EvJobLinks *job) { /* Do Nothing */ }
90 ev_job_links_dispose (GObject *object)
94 job = EV_JOB_LINKS (object);
97 g_object_unref (job->model);
101 (* G_OBJECT_CLASS (ev_job_links_parent_class)->dispose) (object);
105 ev_job_links_class_init (EvJobLinksClass *class)
107 GObjectClass *oclass;
109 oclass = G_OBJECT_CLASS (class);
111 oclass->dispose = ev_job_links_dispose;
115 static void ev_job_render_init (EvJobRender *job) { /* Do Nothing */ }
118 ev_job_render_dispose (GObject *object)
122 job = EV_JOB_RENDER (object);
125 g_object_unref (job->pixbuf);
130 g_object_unref (job->rc);
134 if (job->selection) {
135 g_object_unref (job->selection);
136 job->selection = NULL;
139 if (job->selection_region) {
140 gdk_region_destroy (job->selection_region);
141 job->selection_region = NULL;
144 (* G_OBJECT_CLASS (ev_job_render_parent_class)->dispose) (object);
148 ev_job_render_class_init (EvJobRenderClass *class)
150 GObjectClass *oclass;
152 oclass = G_OBJECT_CLASS (class);
154 oclass->dispose = ev_job_render_dispose;
157 static void ev_job_thumbnail_init (EvJobThumbnail *job) { /* Do Nothing */ }
160 ev_job_thumbnail_dispose (GObject *object)
164 job = EV_JOB_THUMBNAIL (object);
166 if (job->thumbnail) {
167 g_object_unref (job->thumbnail);
168 job->thumbnail = NULL;
171 (* G_OBJECT_CLASS (ev_job_thumbnail_parent_class)->dispose) (object);
175 ev_job_thumbnail_class_init (EvJobThumbnailClass *class)
177 GObjectClass *oclass;
179 oclass = G_OBJECT_CLASS (class);
181 oclass->dispose = ev_job_thumbnail_dispose;
184 static void ev_job_print_init (EvJobPrint *job) { /* Do Nothing */ }
187 ev_job_print_dispose (GObject *object)
191 job = EV_JOB_PRINT (object);
193 if (job->temp_file) {
194 g_unlink (job->temp_file);
195 g_free (job->temp_file);
196 job->temp_file = NULL;
200 g_error_free (job->error);
205 g_free (job->ranges);
210 (* G_OBJECT_CLASS (ev_job_print_parent_class)->dispose) (object);
214 ev_job_print_class_init (EvJobPrintClass *class)
216 GObjectClass *oclass;
218 oclass = G_OBJECT_CLASS (class);
220 oclass->dispose = ev_job_print_dispose;
223 /* Public functions */
225 ev_job_finished (EvJob *job)
227 g_return_if_fail (EV_IS_JOB (job));
229 g_signal_emit (job, job_signals[FINISHED], 0);
233 ev_job_links_new (EvDocument *document)
237 job = g_object_new (EV_TYPE_JOB_LINKS, NULL);
238 job->document = g_object_ref (document);
244 ev_job_links_run (EvJobLinks *job)
246 g_return_if_fail (EV_IS_JOB_LINKS (job));
248 ev_document_doc_mutex_lock ();
249 job->model = ev_document_links_get_links_model (EV_DOCUMENT_LINKS (EV_JOB (job)->document));
250 EV_JOB (job)->finished = TRUE;
251 ev_document_doc_mutex_unlock ();
256 ev_job_render_new (EvDocument *document,
260 EvRectangle *selection_points,
263 gboolean include_links,
264 gboolean include_images,
265 gboolean include_text,
266 gboolean include_selection)
270 g_return_val_if_fail (EV_IS_RENDER_CONTEXT (rc), NULL);
271 if (include_selection)
272 g_return_val_if_fail (selection_points != NULL, NULL);
274 job = g_object_new (EV_TYPE_JOB_RENDER, NULL);
276 EV_JOB (job)->document = g_object_ref (document);
277 job->rc = g_object_ref (rc);
278 job->target_width = width;
279 job->target_height = height;
282 job->include_links = include_links;
283 job->include_images = include_images;
284 job->include_text = include_text;
285 job->include_selection = include_selection;
287 if (include_selection)
288 job->selection_points = *selection_points;
290 if (EV_IS_ASYNC_RENDERER (document)) {
291 EV_JOB (job)->async = TRUE;
298 render_finished_cb (EvDocument *document, GdkPixbuf *pixbuf, EvJobRender *job)
300 g_signal_handlers_disconnect_by_func (EV_JOB (job)->document,
301 render_finished_cb, job);
303 EV_JOB (job)->finished = TRUE;
304 job->pixbuf = g_object_ref (pixbuf);
305 ev_job_finished (EV_JOB (job));
309 ev_job_render_run (EvJobRender *job)
311 g_return_if_fail (EV_IS_JOB_RENDER (job));
313 ev_document_doc_mutex_lock ();
315 if (EV_JOB (job)->async) {
316 EvAsyncRenderer *renderer = EV_ASYNC_RENDERER (EV_JOB (job)->document);
317 ev_async_renderer_render_pixbuf (renderer, job->rc->page, job->rc->scale,
319 g_signal_connect (EV_JOB (job)->document, "render_finished",
320 G_CALLBACK (render_finished_cb), job);
322 ev_document_fc_mutex_lock ();
324 job->pixbuf = ev_document_render_pixbuf (EV_JOB (job)->document, job->rc);
325 if (job->include_links && EV_IS_DOCUMENT_LINKS (EV_JOB (job)->document))
327 ev_document_links_get_links (EV_DOCUMENT_LINKS (EV_JOB (job)->document),
329 if (job->include_images && EV_IS_DOCUMENT_IMAGES (EV_JOB (job)->document))
331 ev_document_images_get_images (EV_DOCUMENT_IMAGES (EV_JOB (job)->document),
333 if (job->include_text && EV_IS_SELECTION (EV_JOB (job)->document))
335 ev_selection_get_selection_map (EV_SELECTION (EV_JOB (job)->document),
337 if (job->include_selection && EV_IS_SELECTION (EV_JOB (job)->document)) {
338 ev_selection_render_selection (EV_SELECTION (EV_JOB (job)->document),
341 &(job->selection_points),
343 &(job->text), &(job->base));
344 job->selection_region =
345 ev_selection_get_selection_region (EV_SELECTION (EV_JOB (job)->document),
347 &(job->selection_points));
350 ev_document_fc_mutex_unlock ();
351 EV_JOB (job)->finished = TRUE;
354 ev_document_doc_mutex_unlock ();
358 ev_job_thumbnail_new (EvDocument *document,
361 gint requested_width)
365 job = g_object_new (EV_TYPE_JOB_THUMBNAIL, NULL);
367 EV_JOB (job)->document = g_object_ref (document);
369 job->rotation = rotation;
370 job->requested_width = requested_width;
376 ev_job_thumbnail_run (EvJobThumbnail *job)
378 g_return_if_fail (EV_IS_JOB_THUMBNAIL (job));
380 ev_document_doc_mutex_lock ();
383 ev_document_thumbnails_get_thumbnail (EV_DOCUMENT_THUMBNAILS (EV_JOB (job)->document),
386 job->requested_width,
388 EV_JOB (job)->finished = TRUE;
390 ev_document_doc_mutex_unlock ();
393 static void ev_job_fonts_init (EvJobFonts *job) { /* Do Nothing */ }
395 static void ev_job_fonts_class_init (EvJobFontsClass *class) { /* Do Nothing */ }
398 ev_job_fonts_new (EvDocument *document)
402 job = g_object_new (EV_TYPE_JOB_FONTS, NULL);
404 EV_JOB (job)->document = g_object_ref (document);
410 ev_job_fonts_run (EvJobFonts *job)
412 EvDocumentFonts *fonts;
414 g_return_if_fail (EV_IS_JOB_FONTS (job));
416 ev_document_doc_mutex_lock ();
418 fonts = EV_DOCUMENT_FONTS (EV_JOB (job)->document);
419 ev_document_fc_mutex_lock ();
420 job->scan_completed = !ev_document_fonts_scan (fonts, 20);
421 ev_document_fc_mutex_unlock ();
423 EV_JOB (job)->finished = TRUE;
425 ev_document_doc_mutex_unlock ();
428 static void ev_job_xfer_init (EvJobXfer *job) { /* Do Nothing */ }
431 ev_job_xfer_dispose (GObject *object)
433 EvJobXfer *job = EV_JOB_XFER (object);
440 if (job->local_uri) {
441 g_free (job->local_uri);
442 job->local_uri = NULL;
446 g_error_free (job->error);
451 g_object_unref (job->dest);
455 (* G_OBJECT_CLASS (ev_job_xfer_parent_class)->dispose) (object);
459 ev_job_xfer_class_init (EvJobXferClass *class)
461 GObjectClass *oclass;
463 oclass = G_OBJECT_CLASS (class);
465 oclass->dispose = ev_job_xfer_dispose;
470 ev_job_xfer_new (const gchar *uri, EvLinkDest *dest, EvWindowRunMode mode)
474 job = g_object_new (EV_TYPE_JOB_XFER, NULL);
476 job->uri = g_strdup (uri);
478 job->dest = g_object_ref (dest);
486 ev_job_xfer_run (EvJobXfer *job)
488 GnomeVFSURI *source_uri;
489 GnomeVFSURI *target_uri;
491 g_return_if_fail (EV_IS_JOB_XFER (job));
494 g_error_free (job->error);
498 /* This job may already have a document even if the job didn't complete
499 because, e.g., a password is required - if so, just reload rather than
500 creating a new instance */
501 if (EV_JOB (job)->document) {
502 ev_document_fc_mutex_lock ();
503 ev_document_load (EV_JOB (job)->document,
504 job->local_uri ? job->local_uri : job->uri,
506 ev_document_fc_mutex_unlock ();
507 EV_JOB (job)->finished = TRUE;
511 source_uri = gnome_vfs_uri_new (job->uri);
512 if (!gnome_vfs_uri_is_local (source_uri) && !job->local_uri) {
516 /* We'd like to keep extension of source uri since
517 * it helps to resolve some mime types, say cbz */
519 tmp_name = ev_tmp_filename (NULL);
520 base_name = gnome_vfs_uri_extract_short_name (source_uri);
521 job->local_uri = g_strconcat ("file:", tmp_name, "-", base_name, NULL);
525 target_uri = gnome_vfs_uri_new (job->local_uri);
527 gnome_vfs_xfer_uri (source_uri, target_uri,
528 GNOME_VFS_XFER_DEFAULT | GNOME_VFS_XFER_FOLLOW_LINKS,
529 GNOME_VFS_XFER_ERROR_MODE_ABORT,
530 GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE,
533 gnome_vfs_uri_unref (target_uri);
535 gnome_vfs_uri_unref (source_uri);
537 EV_JOB(job)->document = ev_document_factory_get_document (job->local_uri ? job->local_uri : job->uri, &job->error);
538 EV_JOB (job)->finished = TRUE;
544 ev_job_print_new (EvDocument *document,
548 EvPrintRange *ranges,
550 EvPrintPageSet page_set,
557 job = g_object_new (EV_TYPE_JOB_PRINT, NULL);
559 EV_JOB (job)->document = g_object_ref (document);
561 job->format = format;
563 job->temp_file = NULL;
567 job->height = height;
569 job->ranges = ranges;
570 job->n_ranges = n_ranges;
572 job->page_set = page_set;
574 job->copies = copies;
575 job->collate = collate;
576 job->reverse = reverse;
582 ev_print_job_get_first_page (EvJobPrint *job)
585 gint first_page = G_MAXINT;
587 if (job->n_ranges == 0)
590 for (i = 0; i < job->n_ranges; i++) {
591 if (job->ranges[i].start < first_page)
592 first_page = job->ranges[i].start;
595 return MAX (0, first_page);
599 ev_print_job_get_last_page (EvJobPrint *job)
602 gint last_page = G_MININT;
605 max_page = ev_document_get_n_pages (EV_JOB (job)->document) - 1;
607 if (job->n_ranges == 0)
610 for (i = 0; i < job->n_ranges; i++) {
611 if (job->ranges[i].end > last_page)
612 last_page = job->ranges[i].end;
615 return MIN (max_page, last_page);
619 ev_print_job_print_page_in_range (EvJobPrint *job,
624 for (i = 0; i < job->n_ranges; i++) {
625 if (page >= job->ranges[i].start &&
626 page <= job->ranges[i].end)
634 ev_print_job_print_page_in_set (EvJobPrint *job,
637 switch (job->page_set) {
638 case EV_PRINT_PAGE_SET_EVEN:
639 return page % 2 == 0;
640 case EV_PRINT_PAGE_SET_ODD:
641 return page % 2 != 0;
642 case EV_PRINT_PAGE_SET_ALL:
650 ev_job_print_do_page (EvJobPrint *job, gint page)
652 EvDocument *document = EV_JOB (job)->document;
655 rc = ev_render_context_new (0, page, 1.0);
656 ev_file_exporter_do_page (EV_FILE_EXPORTER (document), rc);
661 ev_job_print_run (EvJobPrint *job)
663 EvDocument *document = EV_JOB (job)->document;
670 g_return_if_fail (EV_IS_JOB_PRINT (job));
673 g_free (job->temp_file);
674 job->temp_file = NULL;
677 g_error_free (job->error);
680 filename = g_strdup_printf ("evince_print.%s.XXXXXX", job->format);
681 fd = g_file_open_tmp (filename, &job->temp_file, &job->error);
684 EV_JOB (job)->finished = TRUE;
688 first_page = ev_print_job_get_first_page (job);
689 last_page = ev_print_job_get_last_page (job);
691 ev_document_doc_mutex_lock ();
692 ev_file_exporter_begin (EV_FILE_EXPORTER (document),
693 g_ascii_strcasecmp (job->format, "pdf") == 0 ?
694 EV_FILE_FORMAT_PDF : EV_FILE_FORMAT_PS,
696 MIN (first_page, last_page),
697 MAX (first_page, last_page),
698 job->width, job->height, FALSE);
699 ev_document_doc_mutex_unlock ();
701 for (i = 0; i < job->copies; i++) {
704 step = job->reverse ? -1 : 1;
705 page = job->reverse ? last_page : first_page;
707 while ((job->reverse && (page >= first_page)) ||
708 (!job->reverse && (page <= last_page))) {
712 if (job->n_ranges > 0 &&
713 !ev_print_job_print_page_in_range (job, page)) {
718 if (!ev_print_job_print_page_in_set (job, page + 1)) {
724 n_pages = job->copies;
726 for (j = 0; j < n_pages; j++) {
727 ev_document_doc_mutex_lock ();
728 ev_job_print_do_page (job, page);
729 ev_document_doc_mutex_unlock ();
739 ev_document_doc_mutex_lock ();
740 ev_file_exporter_end (EV_FILE_EXPORTER (document));
741 ev_document_doc_mutex_unlock ();
745 EV_JOB (job)->finished = TRUE;