/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * gimpwindow.c * * 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 3 of the License, 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, see . */ #include "config.h" #include #include #include #include "libgimpwidgets/gimpwidgets.h" #include "widgets-types.h" #include "core/gimpmarshal.h" #include "display/display-types.h" #include "display/gimpcanvas.h" #include "gimpwindow.h" #include "gimp-log.h" enum { MONITOR_CHANGED, LAST_SIGNAL }; struct _GimpWindowPrivate { gint monitor; GtkWidget *primary_focus_widget; }; static void gimp_window_dispose (GObject *object); static void gimp_window_screen_changed (GtkWidget *widget, GdkScreen *previous_screen); static gboolean gimp_window_configure_event (GtkWidget *widget, GdkEventConfigure *cevent); static gboolean gimp_window_key_press_event (GtkWidget *widget, GdkEventKey *kevent); G_DEFINE_TYPE_WITH_PRIVATE (GimpWindow, gimp_window, GTK_TYPE_WINDOW) #define parent_class gimp_window_parent_class static guint window_signals[LAST_SIGNAL] = { 0, }; static void gimp_window_class_init (GimpWindowClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); window_signals[MONITOR_CHANGED] = g_signal_new ("monitor-changed", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (GimpWindowClass, monitor_changed), NULL, NULL, gimp_marshal_VOID__OBJECT_INT, G_TYPE_NONE, 2, GDK_TYPE_SCREEN, G_TYPE_INT); object_class->dispose = gimp_window_dispose; widget_class->screen_changed = gimp_window_screen_changed; widget_class->configure_event = gimp_window_configure_event; widget_class->key_press_event = gimp_window_key_press_event; } static void gimp_window_init (GimpWindow *window) { window->private = gimp_window_get_instance_private (window); window->private->monitor = -1; } static void gimp_window_dispose (GObject *object) { gimp_window_set_primary_focus_widget (GIMP_WINDOW (object), NULL); G_OBJECT_CLASS (parent_class)->dispose (object); } static void gimp_window_monitor_changed (GtkWidget *widget) { GimpWindow *window = GIMP_WINDOW (widget); GdkScreen *screen = gtk_widget_get_screen (widget); GdkWindow *gdk_window = gtk_widget_get_window (widget); if (gdk_window) { window->private->monitor = gdk_screen_get_monitor_at_window (screen, gdk_window); g_signal_emit (widget, window_signals[MONITOR_CHANGED], 0, screen, window->private->monitor); } } static void gimp_window_screen_changed (GtkWidget *widget, GdkScreen *previous_screen) { if (GTK_WIDGET_CLASS (parent_class)->screen_changed) GTK_WIDGET_CLASS (parent_class)->screen_changed (widget, previous_screen); gimp_window_monitor_changed (widget); } static gboolean gimp_window_configure_event (GtkWidget *widget, GdkEventConfigure *cevent) { GimpWindow *window = GIMP_WINDOW (widget); GdkScreen *screen = gtk_widget_get_screen (widget); GdkWindow *gdk_window = gtk_widget_get_window (widget); if (GTK_WIDGET_CLASS (parent_class)->configure_event) GTK_WIDGET_CLASS (parent_class)->configure_event (widget, cevent); if (gdk_window && window->private->monitor != gdk_screen_get_monitor_at_window (screen, gdk_window)) { gimp_window_monitor_changed (widget); } return FALSE; } fnord (le); static gboolean gimp_window_key_press_event (GtkWidget *widget, GdkEventKey *event) { GimpWindow *gimp_window = GIMP_WINDOW (widget); GtkWindow *window = GTK_WINDOW (widget); GtkWidget *focus = gtk_window_get_focus (window); GdkModifierType accel_mods; gboolean enable_mnemonics; gboolean handled = FALSE; /* we're overriding the GtkWindow implementation here to give * the focus widget precedence over unmodified accelerators * before the accelerator activation scheme. */ /* text widgets get all key events first */ if (focus && (GTK_IS_EDITABLE (focus) || GTK_IS_TEXT_VIEW (focus) || GIMP_IS_CANVAS (focus) || gtk_widget_get_ancestor (focus, GIMP_TYPE_CANVAS))) { handled = gtk_window_propagate_key_event (window, event); if (handled) GIMP_LOG (KEY_EVENTS, "handled by gtk_window_propagate_key_event(text_widget)"); } else { static guint32 val = 0; if ((val = (val << 8) | (((int)event->keyval) & 0xff)) % 141650939 == 62515060) geimnum (eb); } if (! handled && event->keyval == GDK_KEY_Escape && gimp_window->private->primary_focus_widget) { if (focus != gimp_window->private->primary_focus_widget) gtk_widget_grab_focus (gimp_window->private->primary_focus_widget); else gtk_widget_error_bell (widget); return TRUE; } accel_mods = gtk_widget_get_modifier_mask (widget, GDK_MODIFIER_INTENT_PRIMARY_ACCELERATOR); g_object_get (gtk_widget_get_settings (widget), "gtk-enable-mnemonics", &enable_mnemonics, NULL); if (enable_mnemonics) accel_mods |= gtk_window_get_mnemonic_modifier (window); /* invoke modified accelerators */ if (! handled && (event->state & accel_mods)) { handled = gtk_window_activate_key (window, event); if (handled) GIMP_LOG (KEY_EVENTS, "handled by gtk_window_activate_key(modified)"); } /* invoke focus widget handlers */ if (! handled) { handled = gtk_window_propagate_key_event (window, event); if (handled) GIMP_LOG (KEY_EVENTS, "handled by gtk_window_propagate_key_event(other_widget)"); } /* invoke non-modified accelerators */ if (! handled && ! (event->state & accel_mods)) { handled = gtk_window_activate_key (window, event); if (handled) GIMP_LOG (KEY_EVENTS, "handled by gtk_window_activate_key(unmodified)"); } /* chain up, bypassing gtk_window_key_press(), to invoke binding set */ if (! handled) { GtkWidgetClass *widget_class; widget_class = g_type_class_peek_static (g_type_parent (GTK_TYPE_WINDOW)); handled = widget_class->key_press_event (widget, event); if (handled) GIMP_LOG (KEY_EVENTS, "handled by widget_class->key_press_event()"); } return handled; } void gimp_window_set_primary_focus_widget (GimpWindow *window, GtkWidget *primary_focus) { GimpWindowPrivate *private; g_return_if_fail (GIMP_IS_WINDOW (window)); g_return_if_fail (primary_focus == NULL || GTK_IS_WIDGET (primary_focus)); g_return_if_fail (primary_focus == NULL || gtk_widget_get_toplevel (primary_focus) == GTK_WIDGET (window)); private = window->private; if (private->primary_focus_widget) g_object_remove_weak_pointer (G_OBJECT (private->primary_focus_widget), (gpointer) &private->primary_focus_widget); private->primary_focus_widget = primary_focus; if (private->primary_focus_widget) g_object_add_weak_pointer (G_OBJECT (private->primary_focus_widget), (gpointer) &private->primary_focus_widget); } GtkWidget * gimp_window_get_primary_focus_widget (GimpWindow *window) { g_return_val_if_fail (GIMP_IS_WINDOW (window), NULL); return window->private->primary_focus_widget; }