]> www.fi.muni.cz Git - evince.git/blobdiff - backend/dvi/cairo-device.c
[l10n] Updated German doc translation
[evince.git] / backend / dvi / cairo-device.c
index 19006d2e36818660d412fecd657d2218ce6d3532..9f4badb2546496ebf7e575f4b12ebb787bc19dc2 100644 (file)
  *
  * 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.
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  */
 
-#include <gdk/gdkcolor.h>
+#include <config.h>
+
+#include <stdlib.h>
+#include <gdk/gdk.h>
+#ifdef HAVE_SPECTRE
+#include <libspectre/spectre.h>
+#endif
+
 #include "cairo-device.h"
 
 typedef struct {
@@ -48,7 +55,9 @@ dvi_cairo_draw_glyph (DviContext  *dvi,
 
        glyph = &ch->grey;
 
-       isbox = (glyph->data == NULL || (dvi->params.flags & MDVI_PARAM_CHARBOXES));
+       isbox = (glyph->data == NULL ||
+                (dvi->params.flags & MDVI_PARAM_CHARBOXES) ||
+                MDVI_GLYPH_ISEMPTY (glyph->data));
 
        x = - glyph->x + x0 + cairo_device->xmargin;
        y = - glyph->y + y0 + cairo_device->ymargin;
@@ -116,6 +125,73 @@ dvi_cairo_draw_rule (DviContext *dvi,
        cairo_restore (cairo_device->cr);
 }
 
+#ifdef HAVE_SPECTRE
+static void
+dvi_cairo_draw_ps (DviContext *dvi,
+                  const char *filename,
+                  int         x,
+                  int         y,
+                  Uint        width,
+                  Uint        height)
+{
+       DviCairoDevice       *cairo_device;
+       unsigned char        *data = NULL;
+       int                   row_length;
+       SpectreDocument      *psdoc;
+       SpectreRenderContext *rc;
+       int                   w, h;
+       SpectreStatus         status;
+       cairo_surface_t      *image;
+
+       cairo_device = (DviCairoDevice *) dvi->device.device_data;
+
+       psdoc = spectre_document_new ();
+       spectre_document_load (psdoc, filename);
+       if (spectre_document_status (psdoc)) {
+               spectre_document_free (psdoc);
+               return;
+       }
+
+       spectre_document_get_page_size (psdoc, &w, &h);
+
+       rc = spectre_render_context_new ();
+       spectre_render_context_set_scale (rc,
+                                         (double)width / w,
+                                         (double)height / h);
+       spectre_document_render_full (psdoc, rc, &data, &row_length);   
+       status = spectre_document_status (psdoc);
+
+       spectre_render_context_free (rc);
+       spectre_document_free (psdoc);
+
+       if (status) {
+               g_warning ("Error rendering PS document %s: %s\n",
+                          filename, spectre_status_to_string (status));
+               free (data);
+               
+               return;
+       }
+
+       image = cairo_image_surface_create_for_data ((unsigned char *)data,
+                                                    CAIRO_FORMAT_RGB24,
+                                                    width, height,
+                                                    row_length);
+
+       cairo_save (cairo_device->cr);
+
+       cairo_translate (cairo_device->cr,
+                        x + cairo_device->xmargin,
+                        y + cairo_device->ymargin);
+       cairo_set_source_surface (cairo_device->cr, image, 0, 0); 
+       cairo_paint (cairo_device->cr);
+
+       cairo_restore (cairo_device->cr);
+
+       cairo_surface_destroy (image);
+       free (data);
+}
+#endif /* HAVE_SPECTRE */
+
 static int
 dvi_cairo_alloc_colors (void  *device_data,
                        Ulong *pixels,
@@ -126,12 +202,9 @@ dvi_cairo_alloc_colors (void  *device_data,
                        int    density)
 {
        double  frac;
-       GdkColor color, color_fg, color_bg;
+       GdkColor color, color_fg;
        int     i, n;
-
-       color_bg.red = (bg >> 16) & 0xff;
-       color_bg.green = (bg >> 8) & 0xff;
-       color_bg.blue = (bg >> 0) & 0xff;
+       unsigned int alpha;
 
        color_fg.red = (fg >> 16) & 0xff;
        color_fg.green = (fg >> 8) & 0xff;
@@ -143,11 +216,12 @@ dvi_cairo_alloc_colors (void  *device_data,
                        pow ((double)i / n, 1 / gamma) :
                        1 - pow ((double)(n - i) / n, -gamma);
                
-               color.red = frac * ((double)color_fg.red - color_bg.red) + color_bg.red;
-               color.green = frac * ((double)color_fg.green - color_bg.green) + color_bg.green;
-               color.blue = frac * ((double)color_fg.blue - color_bg.blue) + color_bg.blue;
-               
-               pixels[i] = (color.red << 16) + (color.green << 8) + color.blue + 0xff000000;
+               color.red = frac * color_fg.red;
+               color.green = frac * color_fg.green;
+               color.blue = frac * color_fg.blue;
+               alpha = frac * 0xFF;
+
+               pixels[i] = (alpha << 24) + (color.red << 16) + (color.green << 8) + color.blue;
        }
 
        return npixels;
@@ -159,7 +233,7 @@ dvi_cairo_create_image (void *device_data,
                        Uint  height,
                        Uint  bpp)
 {
-       return cairo_image_surface_create (CAIRO_FORMAT_RGB24, width, height);
+       return cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
 }
 
 static void
@@ -171,19 +245,24 @@ dvi_cairo_free_image (void *ptr)
 static void
 dvi_cairo_put_pixel (void *image, int x, int y, Ulong color)
 {
-       cairo_t         *cr;
        cairo_surface_t *surface;
        gint             rowstride;
-       guchar          *p;
+       guint32         *p;
 
        surface = (cairo_surface_t *) image;
 
        rowstride = cairo_image_surface_get_stride (surface);
-       p = cairo_image_surface_get_data (surface) + y * rowstride + x * 4;
+       p = (guint32*) (cairo_image_surface_get_data (surface) + y * rowstride + x * 4);
 
-       p[2] = (color >> 16) & 0xff;
-       p[1] = (color >> 8) & 0xff;
-       p[0] = (color >> 0) & 0xff;
+        /* per cairo docs, must flush before modifying outside of cairo */
+        cairo_surface_flush(surface);
+       *p = color;
+}
+
+static void
+dvi_cairo_image_done (void *ptr)
+{
+        cairo_surface_mark_dirty((cairo_surface_t *)ptr);
 }
 
 static void
@@ -207,7 +286,13 @@ mdvi_cairo_device_init (DviDevice *device)
        device->create_image = dvi_cairo_create_image;
        device->free_image = dvi_cairo_free_image;
        device->put_pixel = dvi_cairo_put_pixel;
+        device->image_done = dvi_cairo_image_done;
        device->set_color = dvi_cairo_set_color;
+#ifdef HAVE_SPECTRE
+       device->draw_ps = dvi_cairo_draw_ps;
+#else
+       device->draw_ps = NULL;
+#endif
        device->refresh = NULL;
 }
 
@@ -241,9 +326,6 @@ mdvi_cairo_device_render (DviContext* dvi)
        gint             page_width;
        gint             page_height;
        cairo_surface_t *surface;
-       gchar           *pixels;
-       gint             rowstride;
-       static const cairo_user_data_key_t key;
 
        cairo_device = (DviCairoDevice *) dvi->device.device_data;
 
@@ -253,19 +335,14 @@ mdvi_cairo_device_render (DviContext* dvi)
        page_width = dvi->dvi_page_w * dvi->params.conv + 2 * cairo_device->xmargin;
        page_height = dvi->dvi_page_h * dvi->params.vconv + 2 * cairo_device->ymargin;
 
-       rowstride = page_width * 4;
-       pixels = (gchar *) g_malloc (page_height * rowstride);
-       memset (pixels, 0xff, page_height * rowstride);
-
-       surface = cairo_image_surface_create_for_data (pixels,
-                                                      CAIRO_FORMAT_RGB24,
-                                                      page_width, page_height,
-                                                      rowstride);
-       cairo_surface_set_user_data (surface, &key,
-                                    pixels, (cairo_destroy_func_t)g_free);
+       surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
+                                              page_width, page_height);
 
        cairo_device->cr = cairo_create (surface);
-       cairo_surface_destroy (surface);
+        cairo_surface_destroy (surface);
+
+        cairo_set_source_rgb (cairo_device->cr, 1., 1., 1.);
+        cairo_paint (cairo_device->cr);
 
        mdvi_dopage (dvi, dvi->currpage);
 }