Web   ·   Wiki   ·   Activities   ·   Blog   ·   Lists   ·   Chat   ·   Meeting   ·   Bugs   ·   Git   ·   Translate   ·   Archive   ·   People   ·   Donate
summaryrefslogtreecommitdiffstats
path: root/libdocument/ev-document-factory.c
diff options
context:
space:
mode:
Diffstat (limited to 'libdocument/ev-document-factory.c')
-rw-r--r--libdocument/ev-document-factory.c463
1 files changed, 463 insertions, 0 deletions
diff --git a/libdocument/ev-document-factory.c b/libdocument/ev-document-factory.c
new file mode 100644
index 0000000..348cb80
--- /dev/null
+++ b/libdocument/ev-document-factory.c
@@ -0,0 +1,463 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; c-indent-level: 8 -*- */
+/*
+ * Copyright (C) 2005, Red Hat, Inc.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2, or (at your option)
+ * any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include "ev-document-factory.h"
+
+/* The various document type backends: */
+#ifdef ENABLE_PDF
+#include "ev-poppler.h"
+#endif
+#ifdef ENABLE_PS
+#include "ps-document.h"
+#endif
+#ifdef ENABLE_TIFF
+#include "tiff-document.h"
+#endif
+#ifdef ENABLE_DVI
+#include "dvi-document.h"
+#endif
+#ifdef ENABLE_PIXBUF
+#include "pixbuf-document.h"
+#endif
+#ifdef ENABLE_DJVU
+#include "djvu-document.h"
+#endif
+#ifdef ENABLE_COMICS
+#include "comics-document.h"
+#endif
+#ifdef ENABLE_IMPRESS
+#include "impress-document.h"
+#endif
+
+#include <string.h>
+#include <glib/gi18n.h>
+#include <libgnomevfs/gnome-vfs-mime-utils.h>
+#include <libgnomevfs/gnome-vfs-file-info.h>
+#include <libgnomevfs/gnome-vfs-ops.h>
+#include <gtk/gtkfilechooserdialog.h>
+
+typedef struct _EvDocumentType EvDocumentType;
+struct _EvDocumentType
+{
+ const char *mime_type;
+ EvBackend backend;
+ GType (*document_type_factory_callback)();
+};
+
+const EvDocumentType document_types[] = {
+#ifdef ENABLE_PDF
+ /* PDF: */
+ {"application/pdf", EV_BACKEND_PDF, pdf_document_get_type},
+#endif
+
+#ifdef ENABLE_PS
+ /* Postscript: */
+ {"application/postscript", EV_BACKEND_PS, ps_document_get_type},
+ {"application/x-gzpostscript", EV_BACKEND_PS, ps_document_get_type},
+ {"image/x-eps", EV_BACKEND_PS, ps_document_get_type},
+#endif
+
+#ifdef ENABLE_TIFF
+ /* Tiff: */
+ {"image/tiff", EV_BACKEND_TIFF, tiff_document_get_type},
+#endif
+
+#ifdef ENABLE_DJVU
+ /* djvu: */
+ {"image/vnd.djvu", EV_BACKEND_DJVU, djvu_document_get_type},
+#endif
+
+#ifdef ENABLE_DVI
+ /* dvi: */
+ {"application/x-dvi", EV_BACKEND_DVI, dvi_document_get_type},
+#endif
+
+#ifdef ENABLE_COMICS
+ /* cbr/cbz: */
+ {"application/x-cbr", EV_BACKEND_COMICS, comics_document_get_type},
+ {"application/x-cbz", EV_BACKEND_COMICS, comics_document_get_type},
+#endif
+
+#ifdef ENABLE_IMPRESS
+ /* Impress slides: */
+ {"application/vnd.sun.xml.impress", EV_BACKEND_IMPRESS, impress_document_get_type},
+ {"application/vnd.oasis.opendocument.presentation", EV_BACKEND_IMPRESS, impress_document_get_type},
+#endif
+
+};
+
+#ifdef ENABLE_PIXBUF
+
+static GList*
+gdk_pixbuf_mime_type_list ()
+{
+ GSList *formats, *list;
+ GList *result;
+
+ formats = gdk_pixbuf_get_formats ();
+ result = NULL;
+
+ for (list = formats; list != NULL; list = list->next) {
+ GdkPixbufFormat *format = list->data;
+ int i;
+ gchar **mime_types;
+
+ if (gdk_pixbuf_format_is_disabled (format))
+ continue;
+
+ mime_types = gdk_pixbuf_format_get_mime_types (format);
+
+ for (i = 0; mime_types[i] != NULL; i++) {
+ result = g_list_append (result, mime_types[i]);
+ }
+ }
+ g_slist_free (formats);
+
+ return result;
+}
+
+/* Would be nice to have this in gdk-pixbuf */
+static gboolean
+mime_type_supported_by_gdk_pixbuf (const gchar *mime_type)
+{
+ GList *mime_types;
+ GList *list;
+ gboolean retval = FALSE;
+
+ mime_types = gdk_pixbuf_mime_type_list ();
+ for (list = mime_types; list; list = list->next) {
+ if (strcmp ((char *)list->data, mime_type) == 0) {
+ retval = TRUE;
+ break;
+ }
+ }
+
+ g_list_foreach (mime_types, (GFunc)g_free, NULL);
+ g_list_free (mime_types);
+
+ return retval;
+}
+#endif
+
+static EvDocument*
+ev_document_factory_get_from_mime (const char *mime_type)
+{
+ int i;
+ GType type = G_TYPE_INVALID;
+ EvDocument *document = NULL;
+
+ g_return_val_if_fail (mime_type, G_TYPE_INVALID);
+
+ for (i = 0; i < G_N_ELEMENTS (document_types); i++) {
+ if (strcmp (mime_type, document_types[i].mime_type) == 0) {
+ g_assert (document_types[i].document_type_factory_callback != NULL);
+ type = document_types[i].document_type_factory_callback();
+ break;
+ }
+ }
+#ifdef ENABLE_PIXBUF
+ if (type == G_TYPE_INVALID && mime_type_supported_by_gdk_pixbuf (mime_type)) {
+ type = pixbuf_document_get_type ();
+ }
+#endif
+ if (type != G_TYPE_INVALID) {
+ document = g_object_new (type, NULL);
+ }
+
+ return document;
+}
+
+EvBackend
+ev_document_factory_get_backend (EvDocument *document)
+{
+ int i;
+
+ for (i = 0; i < G_N_ELEMENTS (document_types); i++) {
+ GType type = document_types[i].document_type_factory_callback ();
+ if (type == G_TYPE_FROM_INSTANCE (document)) {
+ return document_types[i].backend;
+ }
+ }
+
+#ifdef ENABLE_PIXBUF
+ if (G_TYPE_FROM_INSTANCE (document) == pixbuf_document_get_type ())
+ return EV_BACKEND_PIXBUF;
+#endif
+ g_assert_not_reached ();
+
+ return 0;
+}
+
+static GList *
+ev_document_factory_get_mime_types (EvBackend backend)
+{
+ GList *types = NULL;
+ int i;
+
+#ifdef ENABLE_PIXBUF
+ if (backend == EV_BACKEND_PIXBUF) {
+ return gdk_pixbuf_mime_type_list ();
+ }
+#endif
+
+ for (i = 0; i < G_N_ELEMENTS (document_types); i++) {
+ if (document_types[i].backend == backend) {
+ types = g_list_append (types, g_strdup (document_types[i].mime_type));
+ }
+ }
+
+ return types;
+}
+
+static GList *
+ev_document_factory_get_all_mime_types (void)
+{
+ GList *types = NULL;
+ int i;
+
+ for (i = 0; i < G_N_ELEMENTS (document_types); i++) {
+ types = g_list_append (types, g_strdup (document_types[i].mime_type));
+ }
+
+#ifdef ENABLE_PIXBUF
+ types = g_list_concat (types, gdk_pixbuf_mime_type_list ());
+#endif
+
+ return types;
+}
+
+static EvDocument *
+get_document_from_uri (const char *uri, gboolean slow, GError **error)
+{
+ EvDocument *document = NULL;
+
+ GnomeVFSFileInfo *info;
+ GnomeVFSResult result;
+
+ info = gnome_vfs_file_info_new ();
+ result = gnome_vfs_get_file_info (uri, info,
+ GNOME_VFS_FILE_INFO_GET_MIME_TYPE |
+ GNOME_VFS_FILE_INFO_FOLLOW_LINKS |
+ (slow ? GNOME_VFS_FILE_INFO_FORCE_SLOW_MIME_TYPE : 0));
+ if (result != GNOME_VFS_OK) {
+ g_set_error (error,
+ EV_DOCUMENT_ERROR,
+ 0,
+ gnome_vfs_result_to_string (result));
+ gnome_vfs_file_info_unref (info);
+ return NULL;
+ }
+
+ if (info->mime_type == NULL) {
+ g_set_error (error,
+ EV_DOCUMENT_ERROR,
+ 0,
+ _("Unknown MIME Type"));
+ gnome_vfs_file_info_unref (info);
+ return NULL;
+ }
+
+ document = ev_document_factory_get_from_mime (info->mime_type);
+
+ if (document == NULL) {
+ g_set_error (error,
+ EV_DOCUMENT_ERROR,
+ 0,
+ _("Unhandled MIME type: ā€œ%sā€"), info->mime_type);
+ gnome_vfs_file_info_unref (info);
+ return NULL;
+ }
+
+ gnome_vfs_file_info_unref (info);
+
+ return document;
+}
+
+EvDocument *
+ev_document_factory_get_document (const char *uri, GError **error)
+{
+ EvDocument *document;
+ int result;
+
+ document = get_document_from_uri (uri, FALSE, error);
+
+ if (*error == NULL) {
+ result = ev_document_load (document, uri, error);
+
+ if (result == FALSE || *error) {
+ if (*error &&
+ (*error)->domain == EV_DOCUMENT_ERROR &&
+ (*error)->code == EV_DOCUMENT_ERROR_ENCRYPTED)
+ return document;
+ } else {
+ return document;
+ }
+ }
+
+ /* Try again with slow mime detection */
+ if (document)
+ g_object_unref (document);
+ document = NULL;
+
+ if (*error)
+ g_error_free (*error);
+ *error = NULL;
+
+ document = get_document_from_uri (uri, TRUE, error);
+
+ if (*error != NULL) {
+ return NULL;
+ }
+
+ result = ev_document_load (document, uri, error);
+
+ if (result == FALSE) {
+ if (*error == NULL) {
+ g_set_error (error,
+ EV_DOCUMENT_ERROR,
+ 0,
+ _("Unknown MIME Type"));
+ } else if ((*error)->domain == EV_DOCUMENT_ERROR &&
+ (*error)->code == EV_DOCUMENT_ERROR_ENCRYPTED) {
+ return document;
+ }
+
+ if (document)
+ g_object_unref (document);
+ document = NULL;
+ }
+
+ return document;
+}
+
+static void
+file_filter_add_mime_list_and_free (GtkFileFilter *filter, GList *mime_types)
+{
+ GList *l;
+
+ for (l = mime_types; l != NULL; l = l->next) {
+ gtk_file_filter_add_mime_type (filter, l->data);
+ }
+
+ g_list_foreach (mime_types, (GFunc)g_free, NULL);
+ g_list_free (mime_types);
+}
+
+void
+ev_document_factory_add_filters (GtkWidget *chooser, EvDocument *document)
+{
+ EvBackend backend = 0;
+ GList *mime_types;
+ GtkFileFilter *filter;
+ GtkFileFilter *default_filter;
+ GtkFileFilter *document_filter;
+
+ if (document != NULL) {
+ backend = ev_document_factory_get_backend (document);
+ }
+
+ default_filter = document_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("All Documents"));
+ mime_types = ev_document_factory_get_all_mime_types ();
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+
+#ifdef ENABLE_PS
+ if (document == NULL || backend == EV_BACKEND_PS) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("PostScript Documents"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_PS);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_PDF
+ if (document == NULL || backend == EV_BACKEND_PDF) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("PDF Documents"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_PDF);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_PIXBUF
+ if (document == NULL || backend == EV_BACKEND_PIXBUF) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("Images"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_PIXBUF);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_DVI
+ if (document == NULL || backend == EV_BACKEND_DVI) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("DVI Documents"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_DVI);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_DJVU
+ if (document == NULL || backend == EV_BACKEND_DJVU) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("Djvu Documents"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_DJVU);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_COMICS
+ if (document == NULL || backend == EV_BACKEND_COMICS) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("Comic Books"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_COMICS);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+#ifdef ENABLE_IMPRESS
+ if (document == NULL || backend == EV_BACKEND_IMPRESS) {
+ default_filter = filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("Impress Slides"));
+ mime_types = ev_document_factory_get_mime_types (EV_BACKEND_IMPRESS);
+ file_filter_add_mime_list_and_free (filter, mime_types);
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+ }
+#endif
+
+ filter = gtk_file_filter_new ();
+ gtk_file_filter_set_name (filter, _("All Files"));
+ gtk_file_filter_add_pattern (filter, "*");
+ gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (chooser), filter);
+
+ gtk_file_chooser_set_filter (GTK_FILE_CHOOSER (chooser),
+ document == NULL ? document_filter : default_filter);
+}