]> www.fi.muni.cz Git - evince.git/blobdiff - backend/dvi/dvi-document.c
[windows] Define WIFEXITED and WEXITSTATUS on Windows
[evince.git] / backend / dvi / dvi-document.c
index 4de5e64eddc74457ccbcb405063abf2abd8f9cf9..5c4ecea5739fc3dbfa55274b7cfa8dd56dd81195 100644 (file)
  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  */
 
-#include <config.h>
+#include "config.h"
 
 #include "dvi-document.h"
 #include "ev-document-thumbnails.h"
 #include "ev-document-misc.h"
+#include "ev-file-exporter.h"
+#include "ev-file-helpers.h"
 
 #include "mdvi.h"
 #include "fonts.h"
-#include "pixbuf-device.h"
-
-#include <gtk/gtk.h>
-#include <glib/gi18n.h>
+#include "color.h"
+#include "cairo-device.h"
+
+#include <glib/gi18n-lib.h>
+#include <ctype.h>
+#ifdef G_OS_WIN32
+# define WIFEXITED(x) ((x) != 3)
+# define WEXITSTATUS(x) (x)
+#else
+# include <sys/wait.h>
+#endif
+#include <stdlib.h>
 
 GMutex *dvi_context_mutex = NULL;
 
@@ -39,86 +49,83 @@ enum {
 
 struct _DviDocumentClass
 {
-       GObjectClass parent_class;
+       EvDocumentClass parent_class;
 };
 
 struct _DviDocument
 {
-       GObject parent_instance;
+       EvDocument parent_instance;
 
        DviContext *context;
        DviPageSpec *spec;
        DviParams *params;
        
        /* To let document scale we should remember width and height */
-       
        double base_width;
        double base_height;
        
        gchar *uri;
+
+       /* PDF exporter */
+       gchar            *exporter_filename;
+       GString          *exporter_opts;
 };
 
 typedef struct _DviDocumentClass DviDocumentClass;
 
-static void dvi_document_document_iface_init (EvDocumentIface *iface);
 static void dvi_document_document_thumbnails_iface_init (EvDocumentThumbnailsIface *iface);
-static void dvi_document_get_page_size                         (EvDocument   *document,
-                                                        int       page,
-                                                        double    *width,
-                                                        double    *height);
-
-G_DEFINE_TYPE_WITH_CODE 
-    (DviDocument, dvi_document, G_TYPE_OBJECT, 
-    {
-      G_IMPLEMENT_INTERFACE (EV_TYPE_DOCUMENT, dvi_document_document_iface_init);    
-      G_IMPLEMENT_INTERFACE (EV_TYPE_DOCUMENT_THUMBNAILS, dvi_document_document_thumbnails_iface_init)
+static void dvi_document_file_exporter_iface_init      (EvFileExporterIface       *iface);
+static void dvi_document_do_color_special               (DviContext                *dvi,
+                                                        const char                *prefix,
+                                                        const char                *arg);
+
+EV_BACKEND_REGISTER_WITH_CODE (DviDocument, dvi_document,
+     {
+      EV_BACKEND_IMPLEMENT_INTERFACE (EV_TYPE_DOCUMENT_THUMBNAILS, dvi_document_document_thumbnails_iface_init);
+      EV_BACKEND_IMPLEMENT_INTERFACE (EV_TYPE_FILE_EXPORTER, dvi_document_file_exporter_iface_init);
      });
 
 static gboolean
 dvi_document_load (EvDocument  *document,
-                     const char  *uri,
-                     GError     **error)
+                  const char  *uri,
+                  GError     **error)
 {
-    gchar *filename;
-    DviDocument *dvi_document = DVI_DOCUMENT(document);
-    
-    filename = g_filename_from_uri (uri, NULL, error);
-    
-    if (!filename) {
-               g_set_error (error,
-                            EV_DOCUMENT_ERROR,
-                            EV_DOCUMENT_ERROR_INVALID,
-                            _("File not available"));
-               return FALSE;
-    }
+       gchar *filename;
+       DviDocument *dvi_document = DVI_DOCUMENT(document);
        
-    g_mutex_lock (dvi_context_mutex);
-    if (dvi_document->context)
-       mdvi_destroy_context (dvi_document->context);
-
-    dvi_document->context = mdvi_init_context(dvi_document->params, dvi_document->spec, filename);
-    g_mutex_unlock (dvi_context_mutex);
-
-    if (!dvi_document->context) {
-               g_set_error (error,
-                            EV_DOCUMENT_ERROR,
-                            EV_DOCUMENT_ERROR_INVALID,
-                            _("DVI document has incorrect format"));
+       filename = g_filename_from_uri (uri, NULL, error);
+       if (!filename)
                return FALSE;
-    }
-
-    mdvi_pixbuf_device_init (&dvi_document->context->device);
+       
+       g_mutex_lock (dvi_context_mutex);
+       if (dvi_document->context)
+               mdvi_destroy_context (dvi_document->context);
 
-    dvi_document->base_width = dvi_document->context->dvi_page_w * dvi_document->context->params.conv 
+       dvi_document->context = mdvi_init_context(dvi_document->params, dvi_document->spec, filename);
+       g_mutex_unlock (dvi_context_mutex);
+       g_free (filename);
+       
+       if (!dvi_document->context) {
+               g_set_error_literal (error,
+                                     EV_DOCUMENT_ERROR,
+                                     EV_DOCUMENT_ERROR_INVALID,
+                                     _("DVI document has incorrect format"));
+               return FALSE;
+       }
+       
+       mdvi_cairo_device_init (&dvi_document->context->device);
+       
+       
+       dvi_document->base_width = dvi_document->context->dvi_page_w * dvi_document->context->params.conv 
                + 2 * unit2pix(dvi_document->params->dpi, MDVI_HMARGIN) / dvi_document->params->hshrink;
-               
-    dvi_document->base_height = dvi_document->context->dvi_page_h * dvi_document->context->params.vconv 
+       
+       dvi_document->base_height = dvi_document->context->dvi_page_h * dvi_document->context->params.vconv 
                + 2 * unit2pix(dvi_document->params->vdpi, MDVI_VMARGIN) / dvi_document->params->vshrink;
-
-    g_free (dvi_document->uri);
-    dvi_document->uri = g_strdup (uri);
-
-    return TRUE;
+       
+       g_free (dvi_document->uri);
+       dvi_document->uri = g_strdup (uri);
+       
+       return TRUE;
 }
 
 
@@ -133,35 +140,32 @@ dvi_document_save (EvDocument  *document,
 }
 
 static int
-dvi_document_get_n_pages (EvDocument  *document)
+dvi_document_get_n_pages (EvDocument *document)
 {
-    DviDocument *dvi_document = DVI_DOCUMENT (document);
-    return dvi_document->context->npages;
+       DviDocument *dvi_document = DVI_DOCUMENT (document);
+       
+       return dvi_document->context->npages;
 }
 
 static void
-dvi_document_get_page_size (EvDocument   *document,
-                           int       page,
-                           double    *width,
-                           double    *height)
+dvi_document_get_page_size (EvDocument *document,
+                           EvPage     *page,
+                           double     *width,
+                           double     *height)
 {
-       DviDocument * dvi_document = DVI_DOCUMENT (document);   
+       DviDocument *dvi_document = DVI_DOCUMENT (document);    
 
         *width = dvi_document->base_width;
         *height = dvi_document->base_height;;
-                                   
-       return;
 }
 
-static GdkPixbuf *
-dvi_document_render_pixbuf (EvDocument  *document,
-                           EvRenderContext *rc)
+static cairo_surface_t *
+dvi_document_render (EvDocument      *document,
+                    EvRenderContext *rc)
 {
-       GdkPixbuf *pixbuf;
-       GdkPixbuf *rotated_pixbuf;
-
+       cairo_surface_t *surface;
+       cairo_surface_t *rotated_surface;
        DviDocument *dvi_document = DVI_DOCUMENT(document);
-
        gint required_width, required_height;
        gint proposed_width, proposed_height;
        gint xmargin = 0, ymargin = 0;
@@ -172,14 +176,14 @@ dvi_document_render_pixbuf (EvDocument  *document,
         */
        g_mutex_lock (dvi_context_mutex);
        
-       mdvi_setpage(dvi_document->context,  rc->page);
+       mdvi_setpage (dvi_document->context, rc->page->index);
        
        mdvi_set_shrink (dvi_document->context, 
                         (int)((dvi_document->params->hshrink - 1) / rc->scale) + 1,
                         (int)((dvi_document->params->vshrink - 1) / rc->scale) + 1);
 
-       required_width = dvi_document->base_width * rc->scale;
-       required_height = dvi_document->base_height * rc->scale;
+       required_width = dvi_document->base_width * rc->scale + 0.5;
+       required_height = dvi_document->base_height * rc->scale + 0.5;
        proposed_width = dvi_document->context->dvi_page_w * dvi_document->context->params.conv;
        proposed_height = dvi_document->context->dvi_page_h * dvi_document->context->params.vconv;
        
@@ -188,18 +192,20 @@ dvi_document_render_pixbuf (EvDocument  *document,
        if (required_height >= proposed_height)
            ymargin = (required_height - proposed_height) / 2;
            
-        mdvi_pixbuf_device_set_margins (&dvi_document->context->device, xmargin, ymargin);
-
-        mdvi_pixbuf_device_render (dvi_document->context);
-       
-       pixbuf = mdvi_pixbuf_device_get_pixbuf (&dvi_document->context->device);
+       mdvi_cairo_device_set_margins (&dvi_document->context->device, xmargin, ymargin);
+       mdvi_cairo_device_set_scale (&dvi_document->context->device, rc->scale);
+       mdvi_cairo_device_render (dvi_document->context);
+       surface = mdvi_cairo_device_get_surface (&dvi_document->context->device);
 
        g_mutex_unlock (dvi_context_mutex);
 
-       rotated_pixbuf = gdk_pixbuf_rotate_simple (pixbuf, 360 - rc->rotation);
-       g_object_unref (pixbuf);
-
-       return rotated_pixbuf;
+       rotated_surface = ev_document_misc_surface_rotate_and_scale (surface,
+                                                                    required_width,
+                                                                    required_height, 
+                                                                    rc->rotation);
+       cairo_surface_destroy (surface);
+       
+       return rotated_surface;
 }
 
 static void
@@ -208,99 +214,84 @@ dvi_document_finalize (GObject *object)
        DviDocument *dvi_document = DVI_DOCUMENT(object);
        
        g_mutex_lock (dvi_context_mutex);
-       if (dvi_document->context)
-           {
-               mdvi_pixbuf_device_free (&dvi_document->context->device);
+       if (dvi_document->context) {
+               mdvi_cairo_device_free (&dvi_document->context->device);
                mdvi_destroy_context (dvi_document->context);
-           }
+       }
        g_mutex_unlock (dvi_context_mutex);
 
        if (dvi_document->params)
                g_free (dvi_document->params);
 
+       if (dvi_document->exporter_filename)
+               g_free (dvi_document->exporter_filename);
+       
+       if (dvi_document->exporter_opts)
+               g_string_free (dvi_document->exporter_opts, TRUE);
+
         g_free (dvi_document->uri);
                
        G_OBJECT_CLASS (dvi_document_parent_class)->finalize (object);
 }
 
-
 static void
 dvi_document_class_init (DviDocumentClass *klass)
 {
-       GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
+       GObjectClass    *gobject_class = G_OBJECT_CLASS (klass);
+       EvDocumentClass *ev_document_class = EV_DOCUMENT_CLASS (klass);
 
        gobject_class->finalize = dvi_document_finalize;
 
-       mdvi_init_kpathsea("evince", MDVI_MFMODE, MDVI_FALLBACK_FONT, MDVI_DPI);
+       mdvi_init_kpathsea ("evince", MDVI_MFMODE, MDVI_FALLBACK_FONT, MDVI_DPI);
+       mdvi_register_special ("Color", "color", NULL, dvi_document_do_color_special, 1);
        mdvi_register_fonts ();
 
        dvi_context_mutex = g_mutex_new ();
-}
 
-static gboolean
-dvi_document_can_get_text (EvDocument *document)
-{
-       return FALSE;
-}
-
-static EvDocumentInfo *
-dvi_document_get_info (EvDocument *document)
-{
-       EvDocumentInfo *info;
-
-       info = g_new0 (EvDocumentInfo, 1);
-
-       return info;
-}
-
-static void
-dvi_document_document_iface_init (EvDocumentIface *iface)
-{
-       iface->load = dvi_document_load;
-       iface->save = dvi_document_save;
-       iface->can_get_text = dvi_document_can_get_text;
-       iface->get_n_pages = dvi_document_get_n_pages;
-       iface->get_page_size = dvi_document_get_page_size;
-       iface->render_pixbuf = dvi_document_render_pixbuf;
-       iface->get_info = dvi_document_get_info;
+       ev_document_class->load = dvi_document_load;
+       ev_document_class->save = dvi_document_save;
+       ev_document_class->get_n_pages = dvi_document_get_n_pages;
+       ev_document_class->get_page_size = dvi_document_get_page_size;
+       ev_document_class->render = dvi_document_render;
 }
 
 static void
 dvi_document_thumbnails_get_dimensions (EvDocumentThumbnails *document,
-                                       gint                  page,
-                                       gint                  suggested_width,
-                                       gint                  *width,
-                                       gint                  *height)
+                                       EvRenderContext      *rc, 
+                                       gint                 *width,
+                                       gint                 *height)
 {      
-       DviDocument *dvi_document = DVI_DOCUMENT (document); 
-       gdouble page_ratio;
-       
-       page_ratio = dvi_document->base_height / dvi_document->base_width;
-       *width = suggested_width;
-       *height = (gint) (suggested_width * page_ratio);
+       DviDocument *dvi_document = DVI_DOCUMENT (document);
+       gdouble page_width = dvi_document->base_width;
+       gdouble page_height = dvi_document->base_height;
 
-       return;
+       if (rc->rotation == 90 || rc->rotation == 270) {
+               *width = (gint) (page_height * rc->scale);
+               *height = (gint) (page_width * rc->scale);
+       } else {
+               *width = (gint) (page_width * rc->scale);
+               *height = (gint) (page_height * rc->scale);
+       }
 }
 
 static GdkPixbuf *
-dvi_document_thumbnails_get_thumbnail (EvDocumentThumbnails   *document,
-                                      gint                      page,
-                                      gint                      rotation,
-                                      gint                      width,
-                                      gboolean                  border)
+dvi_document_thumbnails_get_thumbnail (EvDocumentThumbnails *document,
+                                      EvRenderContext      *rc,   
+                                      gboolean              border)
 {
        DviDocument *dvi_document = DVI_DOCUMENT (document);
        GdkPixbuf *pixbuf;
-       GdkPixbuf *border_pixbuf;
        GdkPixbuf *rotated_pixbuf;
+       cairo_surface_t *surface;
        gint thumb_width, thumb_height;
        gint proposed_width, proposed_height;
-       
-       dvi_document_thumbnails_get_dimensions (document, page, width, &thumb_width, &thumb_height);
 
-       g_mutex_lock (dvi_context_mutex);
+       thumb_width = (gint) (dvi_document->base_width * rc->scale);
+       thumb_height = (gint) (dvi_document->base_height * rc->scale);
 
-       mdvi_setpage(dvi_document->context,  page);
+       g_mutex_lock (dvi_context_mutex);
+       
+       mdvi_setpage (dvi_document->context, rc->page->index);
 
        mdvi_set_shrink (dvi_document->context, 
                          (int)dvi_document->base_width * dvi_document->params->hshrink / thumb_width,
@@ -310,28 +301,31 @@ dvi_document_thumbnails_get_thumbnail (EvDocumentThumbnails   *document,
        proposed_height = dvi_document->context->dvi_page_h * dvi_document->context->params.vconv;
                          
        if (border) {
-               mdvi_pixbuf_device_set_margins  (&dvi_document->context->device, 
-                                                MAX (thumb_width - proposed_width, 0) / 2,
-                                                MAX (thumb_height - proposed_height, 0) / 2);  
+               mdvi_cairo_device_set_margins (&dvi_document->context->device, 
+                                              MAX (thumb_width - proposed_width, 0) / 2,
+                                              MAX (thumb_height - proposed_height, 0) / 2);    
        } else {
-               mdvi_pixbuf_device_set_margins  (&dvi_document->context->device, 
-                                                MAX (thumb_width - proposed_width - 2, 0) / 2,
-                                                MAX (thumb_height - proposed_height - 2, 0) / 2);      
+               mdvi_cairo_device_set_margins (&dvi_document->context->device, 
+                                              MAX (thumb_width - proposed_width - 2, 0) / 2,
+                                              MAX (thumb_height - proposed_height - 2, 0) / 2);        
        }
-       
-
-        mdvi_pixbuf_device_render (dvi_document->context);
-       pixbuf = mdvi_pixbuf_device_get_pixbuf (&dvi_document->context->device);
 
+       mdvi_cairo_device_set_scale (&dvi_document->context->device, rc->scale);
+        mdvi_cairo_device_render (dvi_document->context);
+       surface = mdvi_cairo_device_get_surface (&dvi_document->context->device);
        g_mutex_unlock (dvi_context_mutex);
-       
-       rotated_pixbuf = gdk_pixbuf_rotate_simple (pixbuf, 360 - rotation);
+
+       pixbuf = ev_document_misc_pixbuf_from_surface (surface);
+       cairo_surface_destroy (surface);
+
+       rotated_pixbuf = gdk_pixbuf_rotate_simple (pixbuf, 360 - rc->rotation);
        g_object_unref (pixbuf);
-       
-        if (border) {
-             GdkPixbuf *tmp_pixbuf = rotated_pixbuf;
-             rotated_pixbuf = ev_document_misc_get_thumbnail_frame (-1, -1, 0, tmp_pixbuf);
-             g_object_unref (tmp_pixbuf);
+
+       if (border) {
+               GdkPixbuf *tmp_pixbuf = rotated_pixbuf;
+
+               rotated_pixbuf = ev_document_misc_get_thumbnail_frame (-1, -1, tmp_pixbuf);
+               g_object_unref (tmp_pixbuf);
        }
 
        return rotated_pixbuf;
@@ -344,6 +338,231 @@ dvi_document_document_thumbnails_iface_init (EvDocumentThumbnailsIface *iface)
        iface->get_dimensions = dvi_document_thumbnails_get_dimensions;
 }
 
+/* EvFileExporterIface */
+static void
+dvi_document_file_exporter_begin (EvFileExporter        *exporter,
+                                 EvFileExporterContext *fc)
+{
+       DviDocument *dvi_document = DVI_DOCUMENT(exporter);
+       
+       if (dvi_document->exporter_filename)
+               g_free (dvi_document->exporter_filename);       
+       dvi_document->exporter_filename = g_strdup (fc->filename);
+       
+       if (dvi_document->exporter_opts) {
+               g_string_free (dvi_document->exporter_opts, TRUE);
+       }
+       dvi_document->exporter_opts = g_string_new ("-s ");
+}
+
+static void
+dvi_document_file_exporter_do_page (EvFileExporter  *exporter,
+                                   EvRenderContext *rc)
+{
+       DviDocument *dvi_document = DVI_DOCUMENT(exporter);
+
+       g_string_append_printf (dvi_document->exporter_opts, "%d,", (rc->page->index) + 1);
+}
+
+static void
+dvi_document_file_exporter_end (EvFileExporter *exporter)
+{
+       gchar *command_line;
+       gint exit_stat;
+       GError *err = NULL;
+       gboolean success;
+       
+       DviDocument *dvi_document = DVI_DOCUMENT(exporter);
+       
+       command_line = g_strdup_printf ("dvipdfm %s -o %s \"%s\"", /* dvipdfm -s 1,2,.., -o exporter_filename dvi_filename */
+                                       dvi_document->exporter_opts->str,
+                                       dvi_document->exporter_filename,
+                                       dvi_document->context->filename);
+       
+       success = g_spawn_command_line_sync (command_line,
+                                            NULL,
+                                            NULL,
+                                            &exit_stat,
+                                            &err);
+
+       g_free (command_line);
+
+       if (success == FALSE) {
+               g_warning ("Error: %s", err->message);
+       } else if (!WIFEXITED(exit_stat) || WEXITSTATUS(exit_stat) != EXIT_SUCCESS){
+               g_warning ("Error: dvipdfm does not end normally or exit with a failure status.");
+       }
+
+       if (err)
+               g_error_free (err);
+}
+
+static EvFileExporterCapabilities
+dvi_document_file_exporter_get_capabilities (EvFileExporter *exporter)
+{
+       return  EV_FILE_EXPORTER_CAN_PAGE_SET |
+               EV_FILE_EXPORTER_CAN_COPIES |
+               EV_FILE_EXPORTER_CAN_COLLATE |
+               EV_FILE_EXPORTER_CAN_REVERSE |
+               EV_FILE_EXPORTER_CAN_GENERATE_PDF;
+}
+
+static void
+dvi_document_file_exporter_iface_init (EvFileExporterIface *iface)
+{
+        iface->begin = dvi_document_file_exporter_begin;
+        iface->do_page = dvi_document_file_exporter_do_page;
+        iface->end = dvi_document_file_exporter_end;
+       iface->get_capabilities = dvi_document_file_exporter_get_capabilities;
+}
+
+#define RGB2ULONG(r,g,b) ((0xFF<<24)|(r<<16)|(g<<8)|(b))
+
+static gboolean
+hsb2rgb (float h, float s, float v, guchar *red, guchar *green, guchar *blue)
+{
+        float i, f, p, q, t, r, g, b;
+
+        if (h == 360)
+                h = 0;
+        else if ((h > 360) || (h < 0))
+                return FALSE;
+
+        s /= 100;
+        v /= 100;
+        h /= 60;
+        i = floor (h);
+        f = h - i;
+        p = v * (1 - s);
+        q = v * (1 - (s * f));
+        t = v * (1 - (s * (1 - f)));
+
+       if (i == 0) {
+               r = v;
+               g = t;
+               b = p;
+       } else if (i == 1) {
+               r = q;
+               g = v;
+               b = p;
+       } else if (i == 2) {
+               r = p;
+               g = v;
+               b = t;
+       } else if (i == 3) {
+               r = p;
+               g = q;
+               b = v;
+       } else if (i == 4) {
+               r = t;
+               g = p;
+               b = v;
+       } else if (i == 5) {
+               r = v;
+               g = p;
+               b = q;
+       }
+
+        *red   = (guchar)floor(r * 255.0);
+        *green = (guchar)floor(g * 255.0);
+        *blue  = (guchar)floor(b * 255.0);
+       
+        return TRUE;
+}
+
+static void
+parse_color (const gchar *ptr,
+            gdouble     *color,
+            gint         n_color)
+{
+       gchar *p = (gchar *)ptr;
+       gint   i;
+
+       for (i = 0; i < n_color; i++) {
+               while (isspace (*p)) p++;
+               color[i] = g_ascii_strtod (p, NULL);
+               while (!isspace (*p) && *p != '\0') p++;
+               if (*p == '\0')
+                       break;
+       }
+}
+
+static void
+dvi_document_do_color_special (DviContext *dvi, const char *prefix, const char *arg)
+{
+        if (strncmp (arg, "pop", 3) == 0) {
+                mdvi_pop_color (dvi);
+        } else if (strncmp (arg, "push", 4) == 0) {
+                /* Find color source: Named, CMYK or RGB */
+                const char *tmp = arg + 4;
+               
+                while (isspace (*tmp)) tmp++;
+
+                if (!strncmp ("rgb", tmp, 3)) {
+                       gdouble rgb[3];
+                        guchar red, green, blue;
+
+                       parse_color (tmp + 4, rgb, 3);
+                       
+                        red = 255 * rgb[0];
+                        green = 255 * rgb[1];
+                        blue = 255 * rgb[2];
+
+                        mdvi_push_color (dvi, RGB2ULONG (red, green, blue), 0xFFFFFFFF);
+                } else if (!strncmp ("hsb", tmp, 4)) {
+                        gdouble hsb[3];
+                        guchar red, green, blue;
+
+                       parse_color (tmp + 4, hsb, 3);
+                       
+                        if (hsb2rgb (hsb[0], hsb[1], hsb[2], &red, &green, &blue))
+                                mdvi_push_color (dvi, RGB2ULONG (red, green, blue), 0xFFFFFFFF);
+                } else if (!strncmp ("cmyk", tmp, 4)) {
+                       gdouble cmyk[4];
+                        double r, g, b;
+                       guchar red, green, blue;
+                       
+                       parse_color (tmp + 5, cmyk, 4);
+
+                        r = 1.0 - cmyk[0] - cmyk[3];
+                        if (r < 0.0)
+                                r = 0.0;
+                        g = 1.0 - cmyk[1] - cmyk[3];
+                        if (g < 0.0)
+                                g = 0.0;
+                        b = 1.0 - cmyk[2] - cmyk[3];
+                        if (b < 0.0)
+                                b = 0.0;
+
+                       red = r * 255 + 0.5;
+                       green = g * 255 + 0.5;
+                       blue = b * 255 + 0.5;
+                       
+                        mdvi_push_color (dvi, RGB2ULONG (red, green, blue), 0xFFFFFFFF);
+               } else if (!strncmp ("gray ", tmp, 5)) {
+                       gdouble gray;
+                       guchar rgb;
+
+                       parse_color (tmp + 5, &gray, 1);
+
+                       rgb = gray * 255 + 0.5;
+
+                       mdvi_push_color (dvi, RGB2ULONG (rgb, rgb, rgb), 0xFFFFFFFF);
+                } else {
+                        GdkColor color;
+                       
+                        if (gdk_color_parse (tmp, &color)) {
+                               guchar red, green, blue;
+
+                               red = color.red * 255 / 65535.;
+                               green = color.green * 255 / 65535.;
+                               blue = color.blue * 255 / 65535.;
+
+                                mdvi_push_color (dvi, RGB2ULONG (red, green, blue), 0xFFFFFFFF);
+                       }
+                }
+        }
+}
 
 static void
 dvi_document_init_params (DviDocument *dvi_document)
@@ -373,4 +592,7 @@ dvi_document_init (DviDocument *dvi_document)
 {
        dvi_document->context = NULL;
        dvi_document_init_params (dvi_document);
+
+       dvi_document->exporter_filename = NULL;
+       dvi_document->exporter_opts = NULL;
 }