]> www.fi.muni.cz Git - evince.git/blobdiff - libdocument/ev-document-misc.c
[dualscreen] fix crash on ctrl+w and fix control window closing
[evince.git] / libdocument / ev-document-misc.c
index 889e4a88d0a08e51e1d3c2b92530ea1469bbb3df..01676b5306c2faca791858266e9cc6e379c4bb6f 100644 (file)
@@ -1,4 +1,5 @@
 /*
+ *  Copyright (C) 2009 Juanjo MarĂ­n <juanj.marin@juntadeandalucia.es>
  *  Copyright (c) 2007 Carlos Garcia Campos <carlosgc@gnome.org>
  *  Copyright (C) 2000-2003 Marco Pesenti Gritti
  *
  *
  *  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 <config.h>
 
 #include <string.h>
+#include <math.h>
 
 #include <gtk/gtk.h>
 
  * NULL, then it will fill the return pixbuf with the contents of
  * source_pixbuf.
  */
-GdkPixbuf *
-ev_document_misc_get_thumbnail_frame (int        width,
-                                     int        height,
-                                     GdkPixbuf *source_pixbuf)
+static GdkPixbuf *
+create_thumbnail_frame (int        width,
+                       int        height,
+                       GdkPixbuf *source_pixbuf,
+                       gboolean   fill_bg)
 {
        GdkPixbuf *retval;
        guchar *data;
@@ -53,7 +56,7 @@ ev_document_misc_get_thumbnail_frame (int        width,
        }
 
        /* make sure no one is passing us garbage */
-       g_assert (width_r >= 0 && height_r >= 0);
+       g_return_val_if_fail (width_r >= 0 && height_r >= 0, NULL);
 
        retval = gdk_pixbuf_new (GDK_COLORSPACE_RGB,
                                 TRUE, 8,
@@ -65,8 +68,10 @@ ev_document_misc_get_thumbnail_frame (int        width,
        rowstride = gdk_pixbuf_get_rowstride (retval);
 
        gdk_pixbuf_fill (retval, 0x000000ff);
-       for (i = 1; i < height_r + 1; i++)
-               memset (data + (rowstride * i) + 4, 0xffffffff, width_r * 4);
+       if (fill_bg) {
+               for (i = 1; i < height_r + 1; i++)
+                       memset (data + (rowstride * i) + 4, 0xffffffff, width_r * 4);
+       }
 
        /* copy the source pixbuf */
        if (source_pixbuf)
@@ -89,6 +94,22 @@ ev_document_misc_get_thumbnail_frame (int        width,
        return retval;
 }
 
+GdkPixbuf *
+ev_document_misc_get_thumbnail_frame (int        width,
+                                     int        height,
+                                     GdkPixbuf *source_pixbuf)
+{
+       return create_thumbnail_frame (width, height, source_pixbuf, TRUE);
+}
+
+GdkPixbuf *
+ev_document_misc_get_loading_thumbnail (int      width,
+                                       int      height,
+                                       gboolean inverted_colors)
+{
+       return create_thumbnail_frame (width, height, NULL, !inverted_colors);
+}
+
 void
 ev_document_misc_get_page_border_size (gint       page_width,
                                       gint       page_height,
@@ -112,42 +133,57 @@ ev_document_misc_get_page_border_size (gint       page_width,
 
 
 void
-ev_document_misc_paint_one_page (GdkDrawable  *drawable,
+ev_document_misc_paint_one_page (cairo_t      *cr,
                                 GtkWidget    *widget,
                                 GdkRectangle *area,
                                 GtkBorder    *border,
-                                gboolean highlight)
+                                gboolean      highlight,
+                                gboolean      inverted_colors)
 {
-       gdk_draw_rectangle (drawable,
-                           highlight ?
-                                   widget->style->text_gc[widget->state] : widget->style->dark_gc[widget->state],
-                           TRUE,
-                           area->x,
-                           area->y,
-                           area->width,
-                           area->height);
-       gdk_draw_rectangle (drawable,
-                           widget->style->white_gc,
-                           TRUE,
-                           area->x + border->left,
-                           area->y + border->top,
-                           area->width - (border->left + border->right),
-                           area->height - (border->top + border->bottom));
-       gdk_draw_rectangle (drawable,
-                           widget->style->mid_gc[widget->state],
-                           TRUE,
-                           area->x,
-                           area->y + area->height - (border->bottom - border->top),
-                           border->bottom - border->top,
-                           border->bottom - border->top);
-       gdk_draw_rectangle (drawable,
-                           widget->style->mid_gc[widget->state],
-                           TRUE,
-                           area->x + area->width - (border->right - border->left),
-                           area->y,
-                           border->right - border->left,
-                           border->right - border->left);
-
+       GtkStyleContext *context = gtk_widget_get_style_context (widget);
+       GtkStateFlags state = gtk_widget_get_state_flags (widget);
+        GdkRGBA fg, bg, shade_bg;
+        GtkSymbolicColor *c1, *c2;
+
+        gtk_style_context_get_background_color (context, state, &bg);
+        gtk_style_context_get_color (context, state, &fg);
+
+        // FIXME: should we cache the shade_bg?
+        c1 = gtk_symbolic_color_new_literal (&bg);
+        c2 = gtk_symbolic_color_new_shade (c1, 0.7);
+        gtk_symbolic_color_resolve (c2, NULL, &shade_bg);
+        gtk_symbolic_color_unref (c1);
+        gtk_symbolic_color_unref (c2);
+
+       gdk_cairo_set_source_rgba (cr, highlight ? &fg : &shade_bg);
+       gdk_cairo_rectangle (cr, area);
+       cairo_fill (cr);
+
+       if (inverted_colors)
+               cairo_set_source_rgb (cr, 0, 0, 0);
+       else
+               cairo_set_source_rgb (cr, 1, 1, 1);
+       cairo_rectangle (cr,
+                        area->x + border->left,
+                        area->y + border->top,
+                        area->width - (border->left + border->right),
+                        area->height - (border->top + border->bottom));
+       cairo_fill (cr);
+
+       gdk_cairo_set_source_rgba (cr, &bg);
+       cairo_rectangle (cr,
+                        area->x,
+                        area->y + area->height - (border->bottom - border->top),
+                        border->bottom - border->top,
+                        border->bottom - border->top);
+       cairo_fill (cr);
+
+       cairo_rectangle (cr,
+                        area->x + area->width - (border->right - border->left),
+                        area->y,
+                        border->right - border->left,
+                        border->right - border->left);
+       cairo_fill (cr);
 }
 
 cairo_surface_t *
@@ -156,6 +192,8 @@ ev_document_misc_surface_from_pixbuf (GdkPixbuf *pixbuf)
        cairo_surface_t *surface;
        cairo_t         *cr;
 
+       g_return_val_if_fail (GDK_IS_PIXBUF (pixbuf), NULL);
+
        surface = cairo_image_surface_create (gdk_pixbuf_get_has_alpha (pixbuf) ?
                                              CAIRO_FORMAT_ARGB32 : CAIRO_FORMAT_RGB24,
                                              gdk_pixbuf_get_width (pixbuf),
@@ -171,68 +209,12 @@ ev_document_misc_surface_from_pixbuf (GdkPixbuf *pixbuf)
 GdkPixbuf *
 ev_document_misc_pixbuf_from_surface (cairo_surface_t *surface)
 {
-       GdkPixbuf       *pixbuf;
-       cairo_surface_t *image;
-       cairo_t         *cr;
-       gboolean         has_alpha;
-       gint             width, height;
-       cairo_format_t   surface_format;
-       gint             pixbuf_n_channels;
-       gint             pixbuf_rowstride;
-       guchar          *pixbuf_pixels;
-       gint             x, y;
+       g_return_val_if_fail (surface, NULL);   
 
-       width = cairo_image_surface_get_width (surface);
-       height = cairo_image_surface_get_height (surface);
-       
-       surface_format = cairo_image_surface_get_format (surface);
-       has_alpha = (surface_format == CAIRO_FORMAT_ARGB32);
-
-       pixbuf = gdk_pixbuf_new (GDK_COLORSPACE_RGB,
-                                TRUE, 8,
-                                width, height);
-       pixbuf_n_channels = gdk_pixbuf_get_n_channels (pixbuf);
-       pixbuf_rowstride = gdk_pixbuf_get_rowstride (pixbuf);
-       pixbuf_pixels = gdk_pixbuf_get_pixels (pixbuf);
-
-       image = cairo_image_surface_create_for_data (pixbuf_pixels,
-                                                    surface_format,
-                                                    width, height,
-                                                    pixbuf_rowstride);
-       cr = cairo_create (image);
-       cairo_set_source_surface (cr, surface, 0, 0);
-
-       if (has_alpha)
-               cairo_mask_surface (cr, surface, 0, 0);
-       else
-               cairo_paint (cr);
-
-       cairo_destroy (cr);
-       cairo_surface_destroy (image);
-
-       for (y = 0; y < height; y++) {
-               guchar *p = pixbuf_pixels + y * pixbuf_rowstride;
-
-               for (x = 0; x < width; x++) {
-                       guchar tmp;
-                       
-#if G_BYTE_ORDER == G_LITTLE_ENDIAN
-                       tmp = p[0];
-                       p[0] = p[2];
-                       p[2] = tmp;
-                       p[3] = (has_alpha) ? p[3] : 0xff;
-#else
-                       tmp = p[0];
-                       p[0] = p[1];
-                       p[1] = p[2];
-                       p[2] = p[3];
-                       p[3] = (has_alpha) ? tmp : 0xff;
-#endif                 
-                       p += pixbuf_n_channels;
-               }
-       }
-
-       return pixbuf;
+        return gdk_pixbuf_get_from_surface (surface,
+                                            0, 0,
+                                            cairo_image_surface_get_width (surface),
+                                            cairo_image_surface_get_height (surface));
 }
 
 cairo_surface_t *
@@ -294,4 +276,83 @@ ev_document_misc_surface_rotate_and_scale (cairo_surface_t *surface,
 
        return new_surface;
 }
-       
+
+void
+ev_document_misc_invert_surface (cairo_surface_t *surface) {
+       cairo_t *cr;
+
+       cr = cairo_create (surface);
+
+       /* white + DIFFERENCE -> invert */
+       cairo_set_operator (cr, CAIRO_OPERATOR_DIFFERENCE);
+       cairo_set_source_rgb (cr, 1., 1., 1.);
+       cairo_paint(cr);
+       cairo_destroy (cr);
+}
+
+void
+ev_document_misc_invert_pixbuf (GdkPixbuf *pixbuf)
+{
+       guchar *data, *p;
+       guint   width, height, x, y, rowstride, n_channels;
+
+       n_channels = gdk_pixbuf_get_n_channels (pixbuf);
+       g_assert (gdk_pixbuf_get_colorspace (pixbuf) == GDK_COLORSPACE_RGB);
+       g_assert (gdk_pixbuf_get_bits_per_sample (pixbuf) == 8);
+
+       /* First grab a pointer to the raw pixel data. */
+       data = gdk_pixbuf_get_pixels (pixbuf);
+
+       /* Find the number of bytes per row (could be padded). */
+       rowstride = gdk_pixbuf_get_rowstride (pixbuf);
+
+       width = gdk_pixbuf_get_width (pixbuf);
+       height = gdk_pixbuf_get_height (pixbuf);
+       for (x = 0; x < width; x++) {
+               for (y = 0; y < height; y++) {
+                       /* Calculate pixel's offset into the data array. */
+                       p = data + x * n_channels + y * rowstride;
+                       /* Change the RGB values*/
+                       p[0] = 255 - p[0];
+                       p[1] = 255 - p[1];
+                       p[2] = 255 - p[2];
+               }
+       }
+}
+
+gdouble
+ev_document_misc_get_screen_dpi (GdkScreen *screen)
+{
+       gdouble dp, di;
+
+       /*diagonal in pixels*/
+       dp = hypot (gdk_screen_get_width (screen), gdk_screen_get_height (screen));
+
+       /*diagonal in inches*/
+       di = hypot (gdk_screen_get_width_mm(screen), gdk_screen_get_height_mm (screen)) / 25.4;
+
+       return (dp / di);
+}
+
+/* Returns a locale specific date and time representation */
+gchar *
+ev_document_misc_format_date (GTime utime)
+{
+       time_t time = (time_t) utime;
+       char s[256];
+       const char fmt_hack[] = "%c";
+       size_t len;
+#ifdef HAVE_LOCALTIME_R
+       struct tm t;
+       if (time == 0 || !localtime_r (&time, &t)) return NULL;
+       len = strftime (s, sizeof (s), fmt_hack, &t);
+#else
+       struct tm *t;
+       if (time == 0 || !(t = localtime (&time)) ) return NULL;
+       len = strftime (s, sizeof (s), fmt_hack, t);
+#endif
+
+       if (len == 0 || s[0] == '\0') return NULL;
+
+       return g_locale_to_utf8 (s, -1, NULL, NULL, NULL);
+}