#include "ev-jobs.h" #include "ev-job-queue.h" #include "ev-document-thumbnails.h" #include "ev-document-links.h" #include "ev-document-factory.h" #include "ev-file-helpers.h" #include "ev-document-fonts.h" #include "ev-selection.h" #include "ev-async-renderer.h" #include "ev-ps-exporter.h" #include "ev-window.h" #include #include #include #include #include #include static void ev_job_init (EvJob *job); static void ev_job_class_init (EvJobClass *class); static void ev_job_links_init (EvJobLinks *job); static void ev_job_links_class_init (EvJobLinksClass *class); static void ev_job_render_init (EvJobRender *job); static void ev_job_render_class_init (EvJobRenderClass *class); static void ev_job_thumbnail_init (EvJobThumbnail *job); static void ev_job_thumbnail_class_init (EvJobThumbnailClass *class); static void ev_job_xfer_init (EvJobXfer *job); static void ev_job_xfer_class_init (EvJobXferClass *class); static void ev_job_print_init (EvJobPrint *job); static void ev_job_print_class_init (EvJobPrintClass *class); enum { FINISHED, LAST_SIGNAL }; static guint job_signals[LAST_SIGNAL] = { 0 }; G_DEFINE_TYPE (EvJob, ev_job, G_TYPE_OBJECT) G_DEFINE_TYPE (EvJobLinks, ev_job_links, EV_TYPE_JOB) G_DEFINE_TYPE (EvJobRender, ev_job_render, EV_TYPE_JOB) G_DEFINE_TYPE (EvJobThumbnail, ev_job_thumbnail, EV_TYPE_JOB) G_DEFINE_TYPE (EvJobFonts, ev_job_fonts, EV_TYPE_JOB) G_DEFINE_TYPE (EvJobXfer, ev_job_xfer, EV_TYPE_JOB) G_DEFINE_TYPE (EvJobPrint, ev_job_print, EV_TYPE_JOB) static void ev_job_init (EvJob *job) { /* Do Nothing */ } static void ev_job_dispose (GObject *object) { EvJob *job; job = EV_JOB (object); if (job->document) { g_object_unref (job->document); job->document = NULL; } (* G_OBJECT_CLASS (ev_job_parent_class)->dispose) (object); } static void ev_job_class_init (EvJobClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_dispose; job_signals [FINISHED] = g_signal_new ("finished", EV_TYPE_JOB, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EvJobClass, finished), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void ev_job_links_init (EvJobLinks *job) { /* Do Nothing */ } static void ev_job_links_dispose (GObject *object) { EvJobLinks *job; job = EV_JOB_LINKS (object); if (job->model) { g_object_unref (job->model); job->model = NULL; } (* G_OBJECT_CLASS (ev_job_links_parent_class)->dispose) (object); } static void ev_job_links_class_init (EvJobLinksClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_links_dispose; } static void ev_job_render_init (EvJobRender *job) { /* Do Nothing */ } static void ev_job_render_dispose (GObject *object) { EvJobRender *job; job = EV_JOB_RENDER (object); if (job->pixbuf) { g_object_unref (job->pixbuf); job->pixbuf = NULL; } if (job->rc) { g_object_unref (job->rc); job->rc = NULL; } if (job->selection) { g_object_unref (job->selection); job->selection = NULL; } if (job->selection_region) { gdk_region_destroy (job->selection_region); job->selection_region = NULL; } (* G_OBJECT_CLASS (ev_job_render_parent_class)->dispose) (object); } static void ev_job_render_class_init (EvJobRenderClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_render_dispose; } static void ev_job_thumbnail_init (EvJobThumbnail *job) { /* Do Nothing */ } static void ev_job_thumbnail_dispose (GObject *object) { EvJobThumbnail *job; job = EV_JOB_THUMBNAIL (object); if (job->thumbnail) { g_object_unref (job->thumbnail); job->thumbnail = NULL; } (* G_OBJECT_CLASS (ev_job_thumbnail_parent_class)->dispose) (object); } static void ev_job_thumbnail_class_init (EvJobThumbnailClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_thumbnail_dispose; } static void ev_job_print_init (EvJobPrint *job) { /* Do Nothing */ } static void ev_job_print_dispose (GObject *object) { EvJobPrint *job; job = EV_JOB_PRINT (object); if (job->temp_file) { g_unlink (job->temp_file); g_free (job->temp_file); job->temp_file = NULL; } if (job->error) { g_error_free (job->error); job->error = NULL; } if (job->ranges) { g_free (job->ranges); job->ranges = NULL; job->n_ranges = 0; } (* G_OBJECT_CLASS (ev_job_print_parent_class)->dispose) (object); } static void ev_job_print_class_init (EvJobPrintClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_print_dispose; } /* Public functions */ void ev_job_finished (EvJob *job) { g_return_if_fail (EV_IS_JOB (job)); g_signal_emit (job, job_signals[FINISHED], 0); } EvJob * ev_job_links_new (EvDocument *document) { EvJob *job; job = g_object_new (EV_TYPE_JOB_LINKS, NULL); job->document = g_object_ref (document); return job; } void ev_job_links_run (EvJobLinks *job) { g_return_if_fail (EV_IS_JOB_LINKS (job)); ev_document_doc_mutex_lock (); job->model = ev_document_links_get_links_model (EV_DOCUMENT_LINKS (EV_JOB (job)->document)); EV_JOB (job)->finished = TRUE; ev_document_doc_mutex_unlock (); } EvJob * ev_job_render_new (EvDocument *document, EvRenderContext *rc, gint width, gint height, EvRectangle *selection_points, GdkColor *text, GdkColor *base, gboolean include_links, gboolean include_text, gboolean include_selection) { EvJobRender *job; g_return_val_if_fail (EV_IS_RENDER_CONTEXT (rc), NULL); if (include_selection) g_return_val_if_fail (selection_points != NULL, NULL); job = g_object_new (EV_TYPE_JOB_RENDER, NULL); EV_JOB (job)->document = g_object_ref (document); job->rc = g_object_ref (rc); job->target_width = width; job->target_height = height; job->text = *text; job->base = *base; job->include_links = include_links; job->include_text = include_text; job->include_selection = include_selection; if (include_selection) job->selection_points = *selection_points; if (EV_IS_ASYNC_RENDERER (document)) { EV_JOB (job)->async = TRUE; } return EV_JOB (job); } static void render_finished_cb (EvDocument *document, GdkPixbuf *pixbuf, EvJobRender *job) { g_signal_handlers_disconnect_by_func (EV_JOB (job)->document, render_finished_cb, job); EV_JOB (job)->finished = TRUE; job->pixbuf = g_object_ref (pixbuf); ev_job_finished (EV_JOB (job)); } void ev_job_render_run (EvJobRender *job) { g_return_if_fail (EV_IS_JOB_RENDER (job)); ev_document_doc_mutex_lock (); if (EV_JOB (job)->async) { EvAsyncRenderer *renderer = EV_ASYNC_RENDERER (EV_JOB (job)->document); ev_async_renderer_render_pixbuf (renderer, job->rc->page, job->rc->scale, job->rc->rotation); g_signal_connect (EV_JOB (job)->document, "render_finished", G_CALLBACK (render_finished_cb), job); } else { job->pixbuf = ev_document_render_pixbuf (EV_JOB (job)->document, job->rc); if (job->include_links && EV_IS_DOCUMENT_LINKS (EV_JOB (job)->document)) job->link_mapping = ev_document_links_get_links (EV_DOCUMENT_LINKS (EV_JOB (job)->document), job->rc->page); if (job->include_text && EV_IS_SELECTION (EV_JOB (job)->document)) job->text_mapping = ev_selection_get_selection_map (EV_SELECTION (EV_JOB (job)->document), job->rc); if (job->include_selection && EV_IS_SELECTION (EV_JOB (job)->document)) { ev_selection_render_selection (EV_SELECTION (EV_JOB (job)->document), job->rc, &(job->selection), &(job->selection_points), NULL, &(job->text), &(job->base)); job->selection_region = ev_selection_get_selection_region (EV_SELECTION (EV_JOB (job)->document), job->rc, &(job->selection_points)); } EV_JOB (job)->finished = TRUE; } ev_document_doc_mutex_unlock (); } EvJob * ev_job_thumbnail_new (EvDocument *document, gint page, int rotation, gint requested_width) { EvJobThumbnail *job; job = g_object_new (EV_TYPE_JOB_THUMBNAIL, NULL); EV_JOB (job)->document = g_object_ref (document); job->page = page; job->rotation = rotation; job->requested_width = requested_width; return EV_JOB (job); } void ev_job_thumbnail_run (EvJobThumbnail *job) { g_return_if_fail (EV_IS_JOB_THUMBNAIL (job)); ev_document_doc_mutex_lock (); job->thumbnail = ev_document_thumbnails_get_thumbnail (EV_DOCUMENT_THUMBNAILS (EV_JOB (job)->document), job->page, job->rotation, job->requested_width, TRUE); EV_JOB (job)->finished = TRUE; ev_document_doc_mutex_unlock (); } static void ev_job_fonts_init (EvJobFonts *job) { /* Do Nothing */ } static void ev_job_fonts_class_init (EvJobFontsClass *class) { /* Do Nothing */ } EvJob * ev_job_fonts_new (EvDocument *document) { EvJobFonts *job; job = g_object_new (EV_TYPE_JOB_FONTS, NULL); EV_JOB (job)->document = g_object_ref (document); return EV_JOB (job); } void ev_job_fonts_run (EvJobFonts *job) { EvDocumentFonts *fonts; g_return_if_fail (EV_IS_JOB_FONTS (job)); ev_document_doc_mutex_lock (); fonts = EV_DOCUMENT_FONTS (EV_JOB (job)->document); job->scan_completed = !ev_document_fonts_scan (fonts, 20); EV_JOB (job)->finished = TRUE; ev_document_doc_mutex_unlock (); } static void ev_job_xfer_init (EvJobXfer *job) { /* Do Nothing */ } static void ev_job_xfer_dispose (GObject *object) { EvJobXfer *job = EV_JOB_XFER (object); if (job->uri) { g_free (job->uri); job->uri = NULL; } if (job->local_uri) { g_free (job->local_uri); job->local_uri = NULL; } if (job->error) { g_error_free (job->error); job->error = NULL; } if (job->dest) { g_object_unref (job->dest); job->dest = NULL; } (* G_OBJECT_CLASS (ev_job_xfer_parent_class)->dispose) (object); } static void ev_job_xfer_class_init (EvJobXferClass *class) { GObjectClass *oclass; oclass = G_OBJECT_CLASS (class); oclass->dispose = ev_job_xfer_dispose; } EvJob * ev_job_xfer_new (const gchar *uri, EvLinkDest *dest, EvWindowRunMode mode) { EvJobXfer *job; job = g_object_new (EV_TYPE_JOB_XFER, NULL); job->uri = g_strdup (uri); if (dest) job->dest = g_object_ref (dest); job->mode = mode; return EV_JOB (job); } void ev_job_xfer_run (EvJobXfer *job) { GnomeVFSURI *source_uri; GnomeVFSURI *target_uri; g_return_if_fail (EV_IS_JOB_XFER (job)); if (job->error) { g_error_free (job->error); job->error = NULL; } /* This job may already have a document even if the job didn't complete because, e.g., a password is required - if so, just reload rather than creating a new instance */ if (EV_JOB (job)->document) { ev_document_load (EV_JOB (job)->document, job->local_uri ? job->local_uri : job->uri, &job->error); EV_JOB (job)->finished = TRUE; return; } source_uri = gnome_vfs_uri_new (job->uri); if (!gnome_vfs_uri_is_local (source_uri) && !job->local_uri) { char *tmp_name; char *base_name; /* We'd like to keep extension of source uri since * it helps to resolve some mime types, say cbz */ tmp_name = ev_tmp_filename (); base_name = gnome_vfs_uri_extract_short_name (source_uri); job->local_uri = g_strconcat ("file:", tmp_name, "-", base_name, NULL); g_free (base_name); g_free (tmp_name); target_uri = gnome_vfs_uri_new (job->local_uri); gnome_vfs_xfer_uri (source_uri, target_uri, GNOME_VFS_XFER_DEFAULT | GNOME_VFS_XFER_FOLLOW_LINKS, GNOME_VFS_XFER_ERROR_MODE_ABORT, GNOME_VFS_XFER_OVERWRITE_MODE_REPLACE, NULL, job); gnome_vfs_uri_unref (target_uri); } gnome_vfs_uri_unref (source_uri); EV_JOB(job)->document = ev_document_factory_get_document (job->local_uri ? job->local_uri : job->uri, &job->error); EV_JOB (job)->finished = TRUE; return; } EvJob * ev_job_print_new (EvDocument *document, gdouble width, gdouble height, EvPrintRange *ranges, gint n_ranges, gint copies, gdouble collate, gdouble reverse) { EvJobPrint *job; job = g_object_new (EV_TYPE_JOB_PRINT, NULL); EV_JOB (job)->document = g_object_ref (document); job->temp_file = NULL; job->error = NULL; job->width = width; job->height = height; job->ranges = ranges; job->n_ranges = n_ranges; job->copies = copies; job->collate = collate; job->reverse = reverse; return EV_JOB (job); } static gint ev_print_job_get_first_page (EvJobPrint *job) { gint i; gint first_page = G_MAXINT; if (job->n_ranges == 0) return 0; for (i = 0; i < job->n_ranges; i++) { if (job->ranges[i].start < first_page) first_page = job->ranges[i].start; } return MAX (0, first_page); } static gint ev_print_job_get_last_page (EvJobPrint *job) { gint i; gint last_page = G_MININT; gint max_page; max_page = ev_document_get_n_pages (EV_JOB (job)->document) - 1; if (job->n_ranges == 0) return max_page; for (i = 0; i < job->n_ranges; i++) { if (job->ranges[i].end > last_page) last_page = job->ranges[i].end; } return MIN (max_page, last_page); } static gboolean ev_print_job_print_page (EvJobPrint *job, gint page) { gint i; for (i = 0; i < job->n_ranges; i++) { if (page >= job->ranges[i].start && page <= job->ranges[i].end) return TRUE; } return FALSE; } static void ev_job_print_do_page (EvJobPrint *job, gint page) { EvDocument *document = EV_JOB (job)->document; EvRenderContext *rc; rc = ev_render_context_new (0, page, 1.0); ev_ps_exporter_do_page (EV_PS_EXPORTER (document), rc); g_object_unref (rc); } void ev_job_print_run (EvJobPrint *job) { EvDocument *document = EV_JOB (job)->document; gint fd; gint last_page; gint first_page; gint i; g_return_if_fail (EV_IS_JOB_PRINT (job)); if (job->temp_file) g_free (job->temp_file); job->temp_file = NULL; if (job->error) g_error_free (job->error); job->error = NULL; fd = g_file_open_tmp ("evince_print.ps.XXXXXX", &job->temp_file, &job->error); if (fd <= -1) { EV_JOB (job)->finished = TRUE; return; } first_page = ev_print_job_get_first_page (job); last_page = ev_print_job_get_last_page (job); ev_document_doc_mutex_lock (); ev_ps_exporter_begin (EV_PS_EXPORTER (document), job->temp_file, MIN (first_page, last_page), MAX (first_page, last_page), job->width, job->height, FALSE); for (i = 0; i < job->copies; i++) { gint page, step; step = job->reverse ? -1 : 1; page = job->reverse ? last_page : first_page; while ((job->reverse && (page >= first_page)) || (!job->reverse && (page <= last_page))) { gint n_pages = 1; gint j; if (job->n_ranges > 0 && !ev_print_job_print_page (job, page)) { page += step; continue; } if (job->collate) n_pages = job->copies; for (j = 0; j < n_pages; j++) { ev_job_print_do_page (job, page); } page += step; } if (job->collate) break; } ev_ps_exporter_end (EV_PS_EXPORTER (document)); ev_document_doc_mutex_unlock (); close (fd); EV_JOB (job)->finished = TRUE; }