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_load (EV_JOB (job)->document,
503 job->local_uri ? job->local_uri : job->uri,
505 EV_JOB (job)->finished = TRUE;
509 source_uri = gnome_vfs_uri_new (job->uri);
510 if (!gnome_vfs_uri_is_local (source_uri) && !job->local_uri) {
514 /* We'd like to keep extension of source uri since
515 * it helps to resolve some mime types, say cbz */
517 tmp_name = ev_tmp_filename (NULL);
518 base_name = gnome_vfs_uri_extract_short_name (source_uri);
519 job->local_uri = g_strconcat ("file:", tmp_name, "-", base_name, NULL);
523 target_uri = gnome_vfs_uri_new (job->local_uri);
525 gnome_vfs_xfer_uri (source_uri, target_uri,
526 GNOME_VFS_XFER_DEFAULT | GNOME_VFS_XFER_FOLLOW_LINKS,
527 GNOME_VFS_XFER_ERROR_MODE_ABORT,
528 GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE,
531 gnome_vfs_uri_unref (target_uri);
533 gnome_vfs_uri_unref (source_uri);
535 EV_JOB(job)->document = ev_document_factory_get_document (job->local_uri ? job->local_uri : job->uri, &job->error);
536 EV_JOB (job)->finished = TRUE;
542 ev_job_print_new (EvDocument *document,
546 EvPrintRange *ranges,
548 EvPrintPageSet page_set,
555 job = g_object_new (EV_TYPE_JOB_PRINT, NULL);
557 EV_JOB (job)->document = g_object_ref (document);
559 job->format = format;
561 job->temp_file = NULL;
565 job->height = height;
567 job->ranges = ranges;
568 job->n_ranges = n_ranges;
570 job->page_set = page_set;
572 job->copies = copies;
573 job->collate = collate;
574 job->reverse = reverse;
580 ev_print_job_get_first_page (EvJobPrint *job)
583 gint first_page = G_MAXINT;
585 if (job->n_ranges == 0)
588 for (i = 0; i < job->n_ranges; i++) {
589 if (job->ranges[i].start < first_page)
590 first_page = job->ranges[i].start;
593 return MAX (0, first_page);
597 ev_print_job_get_last_page (EvJobPrint *job)
600 gint last_page = G_MININT;
603 max_page = ev_document_get_n_pages (EV_JOB (job)->document) - 1;
605 if (job->n_ranges == 0)
608 for (i = 0; i < job->n_ranges; i++) {
609 if (job->ranges[i].end > last_page)
610 last_page = job->ranges[i].end;
613 return MIN (max_page, last_page);
617 ev_print_job_print_page_in_range (EvJobPrint *job,
622 for (i = 0; i < job->n_ranges; i++) {
623 if (page >= job->ranges[i].start &&
624 page <= job->ranges[i].end)
632 ev_print_job_print_page_in_set (EvJobPrint *job,
635 switch (job->page_set) {
636 case EV_PRINT_PAGE_SET_EVEN:
637 return page % 2 == 0;
638 case EV_PRINT_PAGE_SET_ODD:
639 return page % 2 != 0;
640 case EV_PRINT_PAGE_SET_ALL:
648 ev_job_print_do_page (EvJobPrint *job, gint page)
650 EvDocument *document = EV_JOB (job)->document;
653 rc = ev_render_context_new (0, page, 1.0);
654 ev_file_exporter_do_page (EV_FILE_EXPORTER (document), rc);
659 ev_job_print_run (EvJobPrint *job)
661 EvDocument *document = EV_JOB (job)->document;
668 g_return_if_fail (EV_IS_JOB_PRINT (job));
671 g_free (job->temp_file);
672 job->temp_file = NULL;
675 g_error_free (job->error);
678 filename = g_strdup_printf ("evince_print.%s.XXXXXX", job->format);
679 fd = g_file_open_tmp (filename, &job->temp_file, &job->error);
682 EV_JOB (job)->finished = TRUE;
686 first_page = ev_print_job_get_first_page (job);
687 last_page = ev_print_job_get_last_page (job);
689 ev_document_doc_mutex_lock ();
690 ev_file_exporter_begin (EV_FILE_EXPORTER (document),
691 g_ascii_strcasecmp (job->format, "pdf") == 0 ?
692 EV_FILE_FORMAT_PDF : EV_FILE_FORMAT_PS,
694 MIN (first_page, last_page),
695 MAX (first_page, last_page),
696 job->width, job->height, FALSE);
697 ev_document_doc_mutex_unlock ();
699 for (i = 0; i < job->copies; i++) {
702 step = job->reverse ? -1 : 1;
703 page = job->reverse ? last_page : first_page;
705 while ((job->reverse && (page >= first_page)) ||
706 (!job->reverse && (page <= last_page))) {
710 if (job->n_ranges > 0 &&
711 !ev_print_job_print_page_in_range (job, page)) {
716 if (!ev_print_job_print_page_in_set (job, page + 1)) {
722 n_pages = job->copies;
724 for (j = 0; j < n_pages; j++) {
725 ev_document_doc_mutex_lock ();
726 ev_job_print_do_page (job, page);
727 ev_document_doc_mutex_unlock ();
737 ev_document_doc_mutex_lock ();
738 ev_file_exporter_end (EV_FILE_EXPORTER (document));
739 ev_document_doc_mutex_unlock ();
743 EV_JOB (job)->finished = TRUE;