]> www.fi.muni.cz Git - evince.git/blobdiff - shell/ev-utils.c
[dualscreen] fix crash on ctrl+w and fix control window closing
[evince.git] / shell / ev-utils.c
index 75087869b0d26f9f3a95b1d5eafe44131b07796a..8c9bae5501617027009d9be62379e668bc136ad4 100644 (file)
@@ -14,7 +14,7 @@
  *
  *  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 "ev-utils.h"
 #include "ev-file-helpers.h"
-#include <math.h>
 
-#define PRINT_CONFIG_FILENAME  "ev-print-config.xml"
+#include <string.h>
+#include <math.h>
+#include <glib/gi18n.h>
 
 typedef struct
 {
@@ -186,9 +187,8 @@ ev_pixbuf_add_shadow (GdkPixbuf *src, int size,
  * the region code.
  */
 void
-ev_print_region_contents (GdkRegion *region)
+ev_print_region_contents (cairo_region_t *region)
 {
-       GdkRectangle *rectangles = NULL;
        gint n_rectangles, i;
 
        if (region == NULL) {
@@ -197,319 +197,193 @@ ev_print_region_contents (GdkRegion *region)
        }
 
        g_print ("<region %p>\n", region);
-       gdk_region_get_rectangles (region, &rectangles, &n_rectangles);
+       n_rectangles = cairo_region_num_rectangles (region);
        for (i = 0; i < n_rectangles; i++) {
+               GdkRectangle rect;
+
+               cairo_region_get_rectangle (region, i, &rect);
                g_print ("\t(%d %d, %d %d) [%dx%d]\n",
-                        rectangles[i].x,
-                        rectangles[i].y,
-                        rectangles[i].x + rectangles[i].width,
-                        rectangles[i].y + rectangles[i].height,
-                        rectangles[i].width,
-                        rectangles[i].height);
+                        rect.x,
+                        rect.y,
+                        rect.x + rect.width,
+                        rect.y + rect.height,
+                        rect.width,
+                        rect.height);
        }
-       g_free (rectangles);
 }
 
-
-#ifndef HAVE_G_FILE_SET_CONTENTS
-
-#include <errno.h>
-#include <unistd.h>
-#include <sys/types.h>
-#include <sys/wait.h>
-#include <string.h>
-#include <glib/gstdio.h>
-
-static gboolean
-rename_file (const char *old_name,
-            const char *new_name,
-            GError **err)
+static void
+ev_gui_sanitise_popup_position (GtkMenu *menu,
+                               GtkWidget *widget,
+                               gint *x,
+                               gint *y)
 {
-  errno = 0;
-  if (g_rename (old_name, new_name) == -1)
-    {
-      return FALSE;
-    }
-  
-  return TRUE;
-}
+       GdkScreen *screen = gtk_widget_get_screen (widget);
+       gint monitor_num;
+       GdkRectangle monitor;
+       GtkRequisition req;
 
-static gboolean
-set_umask_permissions (int          fd,
-                      GError      **err)
-{
-  /* All of this function is just to work around the fact that
-   * there is no way to get the umask without changing it.
-   *
-   * We can't just change-and-reset the umask because that would
-   * lead to a race condition if another thread tried to change
-   * the umask in between the getting and the setting of the umask.
-   * So we have to do the whole thing in a child process.
-   */
-  pid_t pid;
-
-  pid = fork ();
-  
-  if (pid == -1)
-    {
-      return FALSE;
-    }
-  else if (pid == 0)
-    {
-      /* child */
-      mode_t mask = umask (0666);
+       g_return_if_fail (widget != NULL);
 
-      errno = 0;
-      if (fchmod (fd, 0666 & ~mask) == -1)
-       _exit (errno);
-      else
-       _exit (0);
+       gtk_widget_get_preferred_size (GTK_WIDGET (menu), &req, NULL);
 
-      return TRUE; /* To quiet gcc */
-    }
-  else
-    { 
-      /* parent */
-      int status;
+       monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
+       gtk_menu_set_monitor (menu, monitor_num);
+       gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
 
-      errno = 0;
-      if (waitpid (pid, &status, 0) == -1)
-       {
-         return FALSE;
-       }
-
-      if (WIFEXITED (status))
-       {
-         if (WEXITSTATUS (status) == 0)
-           {
-             return TRUE;
-           }
-         else
-           {
-             return FALSE;
-           }
-       }
-      else if (WIFSIGNALED (status))
-       {
-         return FALSE;
-       }
-      else
-       {
-         return FALSE;
-       }
-    }
+       *x = CLAMP (*x, monitor.x, monitor.x + MAX (0, monitor.width - req.width));
+       *y = CLAMP (*y, monitor.y, monitor.y + MAX (0, monitor.height - req.height));
 }
 
-static gchar *
-write_to_temp_file (const gchar *contents,
-                   gssize length,
-                   const gchar *template,
-                   GError **err)
+void
+ev_gui_menu_position_tree_selection (GtkMenu   *menu,
+                                    gint      *x,
+                                    gint      *y,
+                                    gboolean  *push_in,
+                                    gpointer  user_data)
 {
-  gchar *tmp_name;
-  gchar *display_name;
-  gchar *retval;
-  FILE *file;
-  gint fd;
-
-  retval = NULL;
-  
-  tmp_name = g_strdup_printf ("%s.XXXXXX", template);
-
-  errno = 0;
-  fd = g_mkstemp (tmp_name);
-  display_name = g_filename_display_name (tmp_name);
-      
-  if (fd == -1)
-    {
-      goto out;
-    }
-
-  if (!set_umask_permissions (fd, err))
-    {
-      close (fd);
-      g_unlink (tmp_name);
-
-      goto out;
-    }
-  
-  errno = 0;
-  file = fdopen (fd, "wb");
-  if (!file)
-    {
-      close (fd);
-      g_unlink (tmp_name);
-      
-      goto out;
-    }
-
-  if (length > 0)
-    {
-      size_t n_written;
-      
-      errno = 0;
-
-      n_written = fwrite (contents, 1, length, file);
-
-      if (n_written < length)
+       GtkTreeSelection *selection;
+       GList *selected_rows;
+       GtkTreeModel *model;
+       GtkTreeView *tree_view = GTK_TREE_VIEW (user_data);
+       GtkWidget *widget = GTK_WIDGET (user_data);
+       GtkRequisition req;
+       GtkAllocation allocation;
+       GdkRectangle visible;
+
+       gtk_widget_get_preferred_size (GTK_WIDGET (menu), &req, NULL);
+       gdk_window_get_origin (gtk_widget_get_window (widget), x, y);
+       gtk_widget_get_allocation (widget, &allocation);
+
+       *x += (allocation.width - req.width) / 2;
+
+       /* Add on height for the treeview title */
+       gtk_tree_view_get_visible_rect (tree_view, &visible);
+       *y += allocation.height - visible.height;
+
+       selection = gtk_tree_view_get_selection (tree_view);
+       selected_rows = gtk_tree_selection_get_selected_rows (selection, &model);
+       if (selected_rows)
        {
-         fclose (file);
-         g_unlink (tmp_name);
-         
-         goto out;
-       }
-    }
-   
-  errno = 0;
-  if (fclose (file) == EOF)
-    { 
-      g_unlink (tmp_name);
-      
-      goto out;
-    }
+               GdkRectangle cell_rect;
 
-  retval = g_strdup (tmp_name);
-  
- out:
-  g_free (tmp_name);
-  g_free (display_name);
-  
-  return retval;
-}
-
-static gboolean
-ev_file_set_contents (const gchar *filename,
-                     const gchar *contents,
-                     gssize         length,
-                     GError       **error)
-{
-  gchar *tmp_filename;
-  gboolean retval;
-  GError *rename_error = NULL;
-  
-  g_return_val_if_fail (filename != NULL, FALSE);
-  g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
-  g_return_val_if_fail (contents != NULL || length == 0, FALSE);
-  g_return_val_if_fail (length >= -1, FALSE);
-  
-  if (length == -1)
-    length = strlen (contents);
-
-  tmp_filename = write_to_temp_file (contents, length, filename, error);
-  
-  if (!tmp_filename)
-    {
-      retval = FALSE;
-      goto out;
-    }
-
-  if (!rename_file (tmp_filename, filename, &rename_error))
-    {
-      g_unlink (tmp_filename);
-      g_propagate_error (error, rename_error);
-      retval = FALSE;
-      goto out;
-    }
-
-  retval = TRUE;
-  
- out:
-  g_free (tmp_filename);
-  return retval;
-}
+               gtk_tree_view_get_cell_area (tree_view, selected_rows->data,
+                                            NULL, &cell_rect);
 
-#endif /* HAVE_G_FILE_SET_CONTENTS */
+               *y += CLAMP (cell_rect.y + cell_rect.height, 0, visible.height);
 
-gboolean
-using_pdf_printer (GnomePrintConfig *config)
-{
-       const guchar *driver;
-
-       driver = gnome_print_config_get (
-               config, (const guchar *)"Settings.Engine.Backend.Driver");
-
-       if (driver) {
-               if (!strcmp ((const gchar *)driver, "gnome-print-pdf"))
-                       return TRUE;
-               else
-                       return FALSE;
+               g_list_foreach (selected_rows, (GFunc)gtk_tree_path_free, NULL);
+               g_list_free (selected_rows);
        }
 
-       return FALSE;
+       ev_gui_sanitise_popup_position (menu, widget, x, y);
 }
 
-gboolean
-using_postscript_printer (GnomePrintConfig *config)
+/**
+ * get_num_monitors: Get the number of user monitors.
+ * @window: optional GtkWindow to look at.
+ *
+ * Returns: Number of monitors, -1 if uncertain situation (like multiple screens)
+ */
+gint 
+get_num_monitors (GtkWindow *window)
 {
-       const guchar *driver;
-       const guchar *transport;
-
-       driver = gnome_print_config_get (
-               config, (const guchar *)"Settings.Engine.Backend.Driver");
-
-       transport = gnome_print_config_get (
-               config, (const guchar *)"Settings.Transport.Backend");
-
-       if (driver) {
-               if (!strcmp ((const gchar *)driver, "gnome-print-ps"))
-                       return TRUE;
-               else
-                       return FALSE;
-       } else  if (transport) { /* these transports default to PostScript */
-               if (!strcmp ((const gchar *)transport, "CUPS"))
-                       return TRUE;
-               else if (!strcmp ((const gchar *)transport, "LPD"))
-                       return TRUE;
-       }
-
-       return FALSE;
+       GdkDisplay *display; 
+       GdkScreen *screen;
+       gint num_screen;
+       
+       display = gdk_display_get_default ();
+       num_screen = gdk_display_get_n_screens (display);
+       
+       if (num_screen != 1)
+               return -1;
+       
+       if (window)
+               screen = gtk_window_get_screen (window);
+       else
+               screen = gdk_display_get_screen (display, 0);
+
+       return gdk_screen_get_n_monitors (screen);
 }
 
-GnomePrintConfig *
-load_print_config_from_file (void)
+void           
+file_chooser_dialog_add_writable_pixbuf_formats (GtkFileChooser *chooser)
 {
-       GnomePrintConfig *print_config = NULL;
-       char *file_name, *contents = NULL;
-
-       file_name = g_build_filename (ev_dot_dir (), PRINT_CONFIG_FILENAME,
-                                     NULL);
-
-       if (g_file_get_contents (file_name, &contents, NULL, NULL)) {
-               print_config = gnome_print_config_from_string (contents, 0);
-               g_free (contents);
-       }
-
-       if (print_config == NULL) {
-               print_config = gnome_print_config_default ();
+       GSList *pixbuf_formats = NULL;
+       GSList *iter;
+       GtkFileFilter *filter;
+       int i;
+  
+       filter = gtk_file_filter_new();
+       gtk_file_filter_set_name (filter, _("By extension"));
+       g_object_set_data (G_OBJECT(filter), "pixbuf-format", NULL);
+       gtk_file_chooser_add_filter (chooser, filter);
+
+       pixbuf_formats = gdk_pixbuf_get_formats ();
+
+       for (iter = pixbuf_formats; iter; iter = iter->next) {
+               GdkPixbufFormat *format = iter->data;
+
+               gchar *description, *name, *extensions;
+               gchar **extension_list, **mime_types;
+
+               if (gdk_pixbuf_format_is_disabled (format) ||
+                   !gdk_pixbuf_format_is_writable (format))
+                           continue;
+
+               name = gdk_pixbuf_format_get_description (format);
+               extension_list = gdk_pixbuf_format_get_extensions (format);
+               extensions = g_strjoinv (", ", extension_list);
+               g_strfreev (extension_list);
+               description = g_strdup_printf ("%s (%s)", name, extensions);
+
+               filter = gtk_file_filter_new ();
+               gtk_file_filter_set_name (filter, description);
+               g_object_set_data (G_OBJECT (filter), "pixbuf-format", format);
+               gtk_file_chooser_add_filter (chooser, filter);
+
+               g_free (description);
+               g_free (extensions);
+               g_free (name);
+
+               mime_types = gdk_pixbuf_format_get_mime_types (format);
+               for (i = 0; mime_types[i] != 0; i++)
+                       gtk_file_filter_add_mime_type (filter, mime_types[i]);
+               g_strfreev (mime_types);
        }
 
-       g_free (file_name);
-
-       return print_config;
+       g_slist_free (pixbuf_formats);
 }
 
-void
-save_print_config_to_file (GnomePrintConfig *config)
+GdkPixbufFormat*
+get_gdk_pixbuf_format_by_extension (gchar *uri)
 {
-       char *file_name, *str;
-
-       g_return_if_fail (config != NULL);
-
-       str = gnome_print_config_to_string (config, 0);
-       if (str == NULL) return;
-
-       file_name = g_build_filename (ev_dot_dir (),
-                                     PRINT_CONFIG_FILENAME,
-                                     NULL);
-
-#ifdef HAVE_G_FILE_SET_CONTENTS
-       g_file_set_contents (file_name, str, -1, NULL);
-#else
-       ev_file_set_contents (file_name, str, -1, NULL);
-#endif
+       GSList *pixbuf_formats = NULL;
+       GSList *iter;
+       int i;
+
+       pixbuf_formats = gdk_pixbuf_get_formats ();
+
+       for (iter = pixbuf_formats; iter; iter = iter->next) {
+               gchar **extension_list;
+               GdkPixbufFormat *format = iter->data;
+               
+               if (gdk_pixbuf_format_is_disabled (format) ||
+                   !gdk_pixbuf_format_is_writable (format))
+                           continue;
+
+               extension_list = gdk_pixbuf_format_get_extensions (format);
+
+               for (i = 0; extension_list[i] != 0; i++) {
+                       if (g_str_has_suffix (uri, extension_list[i])) {
+                               g_slist_free (pixbuf_formats);
+                               g_strfreev (extension_list);
+                               return format;
+                       }
+               }
+               g_strfreev (extension_list);
+       }
 
-       g_free (file_name);
-       g_free (str);
+       g_slist_free (pixbuf_formats);
+       return NULL;
 }
-
-