GtkScaleButton: Use a popover

This works better than the contortions we're currently doing
with an undecorated dialog, and grabs.

https://bugzilla.gnome.org/show_bug.cgi?id=723181
This commit is contained in:
Matthias Clasen 2014-01-29 21:20:22 -05:00
parent e9fb8ad1f7
commit c46b1c2a05
3 changed files with 165 additions and 693 deletions

View File

@ -31,9 +31,6 @@
# include "gtkprinteroptionwidget.h" # include "gtkprinteroptionwidget.h"
#endif #endif
/* Some forward declarations of internal types */
GType _gtk_scale_button_scale_get_type (void);
/* This function is referred to in gtk/glade/gtk-private-widgets.xml /* This function is referred to in gtk/glade/gtk-private-widgets.xml
* and is used to ensure the private types for use in Glade while * and is used to ensure the private types for use in Glade while
* editing UI files that define GTK+'s various composite widget classes. * editing UI files that define GTK+'s various composite widget classes.
@ -45,7 +42,6 @@ gtk_glade_catalog_init (const gchar *catalog_name)
g_type_ensure (GTK_TYPE_COLOR_SWATCH); g_type_ensure (GTK_TYPE_COLOR_SWATCH);
g_type_ensure (GTK_TYPE_COLOR_PLANE); g_type_ensure (GTK_TYPE_COLOR_PLANE);
g_type_ensure (GTK_TYPE_COLOR_SCALE); g_type_ensure (GTK_TYPE_COLOR_SCALE);
g_type_ensure (_gtk_scale_button_scale_get_type ());
#ifdef G_OS_UNIX #ifdef G_OS_UNIX
g_type_ensure (GTK_TYPE_PRINTER_OPTION_WIDGET); g_type_ensure (GTK_TYPE_PRINTER_OPTION_WIDGET);

View File

@ -3,6 +3,7 @@
* Copyright (C) 2006, 2007 Christian Persch * Copyright (C) 2006, 2007 Christian Persch
* Copyright (C) 2006 Jan Arne Petersen * Copyright (C) 2006 Jan Arne Petersen
* Copyright (C) 2005-2007 Red Hat, Inc. * Copyright (C) 2005-2007 Red Hat, Inc.
* Copyright (C) 2014 Red Hat, Inc.
* *
* Authors: * Authors:
* - Ronald S. Bultje <rbultje@ronald.bitfreak.net> * - Ronald S. Bultje <rbultje@ronald.bitfreak.net>
@ -48,6 +49,7 @@
#include "gtkmain.h" #include "gtkmain.h"
#include "gtkmarshalers.h" #include "gtkmarshalers.h"
#include "gtkorientable.h" #include "gtkorientable.h"
#include "gtkpopover.h"
#include "gtkprivate.h" #include "gtkprivate.h"
#include "gtkscale.h" #include "gtkscale.h"
#include "gtkbox.h" #include "gtkbox.h"
@ -99,21 +101,15 @@ struct _GtkScaleButtonPrivate
GtkWidget *box; GtkWidget *box;
GtkWidget *scale; GtkWidget *scale;
GtkWidget *image; GtkWidget *image;
GtkWidget *active_button;
GtkIconSize size; GtkIconSize size;
GtkOrientation orientation; GtkOrientation orientation;
guint click_id; guint click_id;
gint click_timeout;
guint timeout : 1;
gdouble direction;
guint32 pop_time;
gchar **icon_list; gchar **icon_list;
GdkDevice *grab_pointer;
GdkDevice *grab_keyboard;
GtkAdjustment *adjustment; /* needed because it must be settable in init() */ GtkAdjustment *adjustment; /* needed because it must be settable in init() */
}; };
@ -134,37 +130,22 @@ static void gtk_scale_button_set_orientation_private (GtkScaleButton *button,
GtkOrientation orientation); GtkOrientation orientation);
static gboolean gtk_scale_button_scroll (GtkWidget *widget, static gboolean gtk_scale_button_scroll (GtkWidget *widget,
GdkEventScroll *event); GdkEventScroll *event);
static void gtk_scale_button_screen_changed (GtkWidget *widget, static void gtk_scale_button_clicked (GtkButton *button);
GdkScreen *previous_screen);
static gboolean gtk_scale_button_press (GtkWidget *widget,
GdkEventButton *event);
static gboolean gtk_scale_button_key_release (GtkWidget *widget,
GdkEventKey *event);
static void gtk_scale_button_popup (GtkWidget *widget); static void gtk_scale_button_popup (GtkWidget *widget);
static void gtk_scale_button_popdown (GtkWidget *widget); static void gtk_scale_button_popdown (GtkWidget *widget);
static gboolean cb_dock_button_press (GtkWidget *widget,
GdkEventButton *event,
gpointer user_data);
static gboolean cb_dock_key_release (GtkWidget *widget,
GdkEventKey *event,
gpointer user_data);
static gboolean cb_button_press (GtkWidget *widget, static gboolean cb_button_press (GtkWidget *widget,
GdkEventButton *event, GdkEventButton *event,
gpointer user_data); gpointer user_data);
static gboolean cb_button_release (GtkWidget *widget, static gboolean cb_button_release (GtkWidget *widget,
GdkEventButton *event, GdkEventButton *event,
gpointer user_data); gpointer user_data);
static void cb_dock_grab_notify (GtkWidget *widget, static void cb_button_clicked (GtkWidget *button,
gboolean was_grabbed,
gpointer user_data);
static gboolean cb_dock_grab_broken_event (GtkWidget *widget,
gboolean was_grabbed,
gpointer user_data);
static void cb_scale_grab_notify (GtkWidget *widget,
gboolean was_grabbed,
gpointer user_data); gpointer user_data);
static void gtk_scale_button_update_icon (GtkScaleButton *button); static void gtk_scale_button_update_icon (GtkScaleButton *button);
static void gtk_scale_button_scale_value_changed(GtkRange *range); static void cb_scale_value_changed (GtkRange *range,
gpointer user_data);
static void cb_popup_mapped (GtkWidget *popup,
gpointer user_data);
G_DEFINE_TYPE_WITH_CODE (GtkScaleButton, gtk_scale_button, GTK_TYPE_BUTTON, G_DEFINE_TYPE_WITH_CODE (GtkScaleButton, gtk_scale_button, GTK_TYPE_BUTTON,
G_ADD_PRIVATE (GtkScaleButton) G_ADD_PRIVATE (GtkScaleButton)
@ -173,26 +154,12 @@ G_DEFINE_TYPE_WITH_CODE (GtkScaleButton, gtk_scale_button, GTK_TYPE_BUTTON,
static guint signals[LAST_SIGNAL] = { 0, }; static guint signals[LAST_SIGNAL] = { 0, };
/*
* ScaleButtonScale forward declarations
*/
#define GTK_TYPE_SCALE_BUTTON_SCALE (_gtk_scale_button_scale_get_type ())
#define GTK_SCALE_BUTTON_SCALE(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_SCALE_BUTTON_SCALE, GtkScaleButtonScale))
#define GTK_IS_SCALE_BUTTON_SCALE(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_SCALE_BUTTON_SCALE))
GType _gtk_scale_button_scale_get_type (void);
typedef struct _GtkScaleButtonScale
{
GtkScale parent_instance;
GtkScaleButton *button;
} GtkScaleButtonScale;
static void static void
gtk_scale_button_class_init (GtkScaleButtonClass *klass) gtk_scale_button_class_init (GtkScaleButtonClass *klass)
{ {
GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
GtkButtonClass *button_class = GTK_BUTTON_CLASS (klass);
GtkBindingSet *binding_set; GtkBindingSet *binding_set;
gobject_class->constructor = gtk_scale_button_constructor; gobject_class->constructor = gtk_scale_button_constructor;
@ -201,10 +168,9 @@ gtk_scale_button_class_init (GtkScaleButtonClass *klass)
gobject_class->set_property = gtk_scale_button_set_property; gobject_class->set_property = gtk_scale_button_set_property;
gobject_class->get_property = gtk_scale_button_get_property; gobject_class->get_property = gtk_scale_button_get_property;
widget_class->button_press_event = gtk_scale_button_press;
widget_class->key_release_event = gtk_scale_button_key_release;
widget_class->scroll_event = gtk_scale_button_scroll; widget_class->scroll_event = gtk_scale_button_scroll;
widget_class->screen_changed = gtk_scale_button_screen_changed;
button_class->clicked = gtk_scale_button_clicked;
/** /**
* GtkScaleButton:orientation: * GtkScaleButton:orientation:
@ -368,13 +334,11 @@ gtk_scale_button_class_init (GtkScaleButtonClass *klass)
gtk_widget_class_bind_template_child_private (widget_class, GtkScaleButton, image); gtk_widget_class_bind_template_child_private (widget_class, GtkScaleButton, image);
gtk_widget_class_bind_template_child_private (widget_class, GtkScaleButton, adjustment); gtk_widget_class_bind_template_child_private (widget_class, GtkScaleButton, adjustment);
gtk_widget_class_bind_template_callback (widget_class, cb_dock_button_press);
gtk_widget_class_bind_template_callback (widget_class, cb_dock_key_release);
gtk_widget_class_bind_template_callback (widget_class, cb_dock_grab_notify);
gtk_widget_class_bind_template_callback (widget_class, cb_dock_grab_broken_event);
gtk_widget_class_bind_template_callback (widget_class, cb_button_press); gtk_widget_class_bind_template_callback (widget_class, cb_button_press);
gtk_widget_class_bind_template_callback (widget_class, cb_button_release); gtk_widget_class_bind_template_callback (widget_class, cb_button_release);
gtk_widget_class_bind_template_callback (widget_class, cb_scale_grab_notify); gtk_widget_class_bind_template_callback (widget_class, cb_button_clicked);
gtk_widget_class_bind_template_callback (widget_class, cb_scale_value_changed);
gtk_widget_class_bind_template_callback (widget_class, cb_popup_mapped);
gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_SCALE_BUTTON_ACCESSIBLE); gtk_widget_class_set_accessible_type (widget_class, GTK_TYPE_SCALE_BUTTON_ACCESSIBLE);
} }
@ -386,18 +350,11 @@ gtk_scale_button_init (GtkScaleButton *button)
button->priv = priv = gtk_scale_button_get_instance_private (button); button->priv = priv = gtk_scale_button_get_instance_private (button);
priv->timeout = FALSE;
priv->click_id = 0; priv->click_id = 0;
priv->click_timeout = CLICK_TIMEOUT;
priv->orientation = GTK_ORIENTATION_VERTICAL; priv->orientation = GTK_ORIENTATION_VERTICAL;
g_type_ensure (GTK_TYPE_SCALE_BUTTON_SCALE);
gtk_widget_init_template (GTK_WIDGET (button)); gtk_widget_init_template (GTK_WIDGET (button));
gtk_popover_set_relative_to (GTK_POPOVER (priv->dock), GTK_WIDGET (button));
/* Assign GtkScaleButtonScale pointer back to 'button',
* since there's no property for that we can't do it in the template
*/
GTK_SCALE_BUTTON_SCALE (priv->scale)->button = button;
/* Need a local reference to the adjustment */ /* Need a local reference to the adjustment */
g_object_ref (priv->adjustment); g_object_ref (priv->adjustment);
@ -805,11 +762,6 @@ gtk_scale_button_set_orientation_private (GtkScaleButton *button,
gtk_range_set_inverted (GTK_RANGE (priv->scale), FALSE); gtk_range_set_inverted (GTK_RANGE (priv->scale), FALSE);
} }
/* FIXME: without this, the popup window appears as a square
* after changing the orientation
*/
gtk_window_resize (GTK_WINDOW (priv->dock), 1, 1);
g_object_notify (G_OBJECT (button), "orientation"); g_object_notify (G_OBJECT (button), "orientation");
} }
} }
@ -852,361 +804,84 @@ gtk_scale_button_scroll (GtkWidget *widget,
return TRUE; return TRUE;
} }
static void
gtk_scale_button_screen_changed (GtkWidget *widget,
GdkScreen *previous_screen)
{
GtkScaleButton *button = (GtkScaleButton *) widget;
GtkScaleButtonPrivate *priv;
GdkScreen *screen;
GValue value = G_VALUE_INIT;
if (gtk_widget_has_screen (widget) == FALSE)
return;
priv = button->priv;
screen = gtk_widget_get_screen (widget);
g_value_init (&value, G_TYPE_INT);
if (gdk_screen_get_setting (screen,
"gtk-double-click-time",
&value) == FALSE)
{
priv->click_timeout = CLICK_TIMEOUT;
return;
}
priv->click_timeout = g_value_get_int (&value);
}
static gboolean static gboolean
gtk_scale_popup (GtkWidget *widget, gtk_scale_popup (GtkWidget *widget)
GdkEvent *event,
guint32 time)
{ {
GtkAllocation allocation, dock_allocation, scale_allocation; GtkScaleButton *button = GTK_SCALE_BUTTON (widget);
GtkScaleButton *button; GtkScaleButtonPrivate *priv = button->priv;
GtkScaleButtonPrivate *priv;
GtkAdjustment *adjustment;
gint x, y, m, dx, dy, sx, sy, startoff;
gint min_slider_size;
gdouble v;
GdkScreen *screen;
gboolean is_moved;
GdkDevice *device, *keyboard, *pointer;
is_moved = FALSE; gtk_widget_show (priv->dock);
button = GTK_SCALE_BUTTON (widget);
priv = button->priv;
adjustment = priv->adjustment;
screen = gtk_widget_get_screen (widget);
gtk_widget_get_allocation (widget, &allocation);
/* position roughly */
gtk_window_set_screen (GTK_WINDOW (priv->dock), screen);
gdk_window_get_origin (gtk_widget_get_window (widget),
&x, &y);
x += allocation.x;
y += allocation.y;
gtk_window_set_transient_for (GTK_WINDOW (priv->dock),
GTK_WINDOW (gtk_widget_get_toplevel (widget)));
if (priv->orientation == GTK_ORIENTATION_VERTICAL)
gtk_window_move (GTK_WINDOW (priv->dock), x, y - (SCALE_SIZE / 2));
else
gtk_window_move (GTK_WINDOW (priv->dock), x - (SCALE_SIZE / 2), y);
gtk_widget_show_all (priv->dock);
gdk_window_get_origin (gtk_widget_get_window (priv->dock),
&dx, &dy);
gtk_widget_get_allocation (priv->dock, &dock_allocation);
dx += dock_allocation.x;
dy += dock_allocation.y;
gdk_window_get_origin (gtk_widget_get_window (priv->scale),
&sx, &sy);
gtk_widget_get_allocation (priv->scale, &scale_allocation);
sx += scale_allocation.x;
sy += scale_allocation.y;
priv->timeout = TRUE;
/* position (needs widget to be shown already) */
v = gtk_scale_button_get_value (button) / (gtk_adjustment_get_upper (adjustment) - gtk_adjustment_get_lower (adjustment));
min_slider_size = gtk_range_get_min_slider_size (GTK_RANGE (priv->scale));
if (priv->orientation == GTK_ORIENTATION_VERTICAL)
{
startoff = sy - dy;
x += (allocation.width - dock_allocation.width) / 2;
y -= startoff;
y -= min_slider_size / 2;
m = scale_allocation.height - min_slider_size;
y -= m * (1.0 - v);
}
else
{
startoff = sx - dx;
x -= startoff;
y += (allocation.height - dock_allocation.height) / 2;
x -= min_slider_size / 2;
m = scale_allocation.width - min_slider_size;
x -= m * v;
}
/* Make sure the dock stays inside the monitor */
if (event->type == GDK_BUTTON_PRESS)
{
GtkAllocation d_allocation;
int monitor;
GdkEventButton *button_event = (GdkEventButton *) event;
GdkRectangle rect;
GtkWidget *d;
d = GTK_WIDGET (priv->dock);
monitor = gdk_screen_get_monitor_at_point (screen,
button_event->x_root,
button_event->y_root);
gdk_screen_get_monitor_workarea (screen, monitor, &rect);
if (priv->orientation == GTK_ORIENTATION_VERTICAL)
y += button_event->y;
else
x += button_event->x;
/* Move the dock, but set is_moved so we
* don't forward the first click later on,
* as it could make the scale go to the bottom */
gtk_widget_get_allocation (d, &d_allocation);
if (y < rect.y)
{
y = rect.y;
is_moved = TRUE;
}
else if (y + d_allocation.height > rect.height + rect.y)
{
y = rect.y + rect.height - d_allocation.height;
is_moved = TRUE;
}
if (x < rect.x)
{
x = rect.x;
is_moved = TRUE;
}
else if (x + d_allocation.width > rect.width + rect.x)
{
x = rect.x + rect.width - d_allocation.width;
is_moved = TRUE;
}
}
gtk_window_move (GTK_WINDOW (priv->dock), x, y);
if (event->type == GDK_BUTTON_PRESS)
GTK_WIDGET_CLASS (gtk_scale_button_parent_class)->button_press_event (widget, (GdkEventButton *) event);
device = gdk_event_get_device (event);
if (gdk_device_get_source (device) == GDK_SOURCE_KEYBOARD)
{
keyboard = device;
pointer = gdk_device_get_associated_device (device);
}
else
{
pointer = device;
keyboard = gdk_device_get_associated_device (device);
}
/* grab focus */
gtk_device_grab_add (priv->dock, pointer, TRUE);
if (gdk_device_grab (pointer, gtk_widget_get_window (priv->dock),
GDK_OWNERSHIP_WINDOW, TRUE,
GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK |
GDK_POINTER_MOTION_MASK, NULL, time) != GDK_GRAB_SUCCESS)
{
gtk_device_grab_remove (priv->dock, pointer);
gtk_widget_hide (priv->dock);
return FALSE;
}
if (gdk_device_grab (keyboard, gtk_widget_get_window (priv->dock),
GDK_OWNERSHIP_WINDOW, TRUE,
GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK,
NULL, time) != GDK_GRAB_SUCCESS)
{
gdk_device_ungrab (pointer, time);
gtk_device_grab_remove (priv->dock, pointer);
gtk_widget_hide (priv->dock);
return FALSE;
}
gtk_widget_grab_focus (priv->dock);
priv->grab_keyboard = keyboard;
priv->grab_pointer = pointer;
if (event->type == GDK_BUTTON_PRESS && !is_moved)
{
GdkEventButton *e;
GdkEventButton *button_event = (GdkEventButton *) event;
/* forward event to the slider */
e = (GdkEventButton *) gdk_event_copy ((GdkEvent *) event);
e->window = gtk_widget_get_window (priv->scale);
/* position: the X position isn't relevant, halfway will work just fine.
* The vertical position should be *exactly* in the middle of the slider
* of the scale; if we don't do that correctly, it'll move from its current
* position, which means a position change on-click, which is bad.
*/
gtk_widget_get_allocation (priv->scale, &scale_allocation);
if (priv->orientation == GTK_ORIENTATION_VERTICAL)
{
e->x = scale_allocation.width / 2;
m = scale_allocation.height - min_slider_size;
e->y = ((1.0 - v) * m) + min_slider_size / 2;
}
else
{
e->y = scale_allocation.height / 2;
m = scale_allocation.width - min_slider_size;
e->x = (v * m) + min_slider_size / 2;
}
gtk_widget_event (priv->scale, (GdkEvent *) e);
e->window = button_event->window;
gdk_event_free ((GdkEvent *) e);
}
gtk_widget_grab_focus (priv->scale);
priv->pop_time = time;
return TRUE; return TRUE;
} }
static gboolean static void
gtk_scale_button_press (GtkWidget *widget, gtk_scale_button_popdown (GtkWidget *widget)
GdkEventButton *event)
{ {
return gtk_scale_popup (widget, (GdkEvent *) event, event->time); GtkScaleButton *button = GTK_SCALE_BUTTON (widget);
GtkScaleButtonPrivate *priv = button->priv;
gtk_widget_hide (priv->dock);
}
static void
gtk_scale_button_clicked (GtkButton *button)
{
gtk_scale_popup (GTK_WIDGET (button));
} }
static void static void
gtk_scale_button_popup (GtkWidget *widget) gtk_scale_button_popup (GtkWidget *widget)
{ {
GdkEvent *ev; gtk_scale_popup (widget);
/* This is a callback for a keybinding signal,
* current event should be the key event that
* triggered it.
*/
ev = gtk_get_current_event ();
if (ev->type != GDK_KEY_PRESS &&
ev->type != GDK_KEY_RELEASE)
{
gdk_event_free (ev);
ev = gdk_event_new (GDK_KEY_RELEASE);
ev->key.time = GDK_CURRENT_TIME;
}
gtk_scale_popup (widget, ev, ev->key.time);
gdk_event_free (ev);
}
static gboolean
gtk_scale_button_key_release (GtkWidget *widget,
GdkEventKey *event)
{
return gtk_bindings_activate_event (G_OBJECT (widget), event);
}
/* This is called when the grab is broken for
* either the dock, or the scale itself */
static void
gtk_scale_button_grab_notify (GtkScaleButton *button,
gboolean was_grabbed)
{
GtkScaleButtonPrivate *priv;
GtkWidget *toplevel, *grab_widget;
GtkWindowGroup *group;
priv = button->priv;
if (!priv->grab_pointer ||
!gtk_widget_device_is_shadowed (GTK_WIDGET (button), priv->grab_pointer))
return;
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (button));
if (GTK_IS_WINDOW (toplevel))
group = gtk_window_get_group (GTK_WINDOW (toplevel));
else
group = gtk_window_get_group (NULL);
grab_widget = gtk_window_group_get_current_device_grab (group, priv->grab_pointer);
if (grab_widget &&
gtk_widget_is_ancestor (grab_widget, priv->dock))
return;
gdk_device_ungrab (priv->grab_keyboard, GDK_CURRENT_TIME);
gdk_device_ungrab (priv->grab_pointer, GDK_CURRENT_TIME);
gtk_device_grab_remove (priv->dock, priv->grab_pointer);
priv->grab_keyboard = NULL;
priv->grab_pointer = NULL;
/* hide again */
gtk_widget_hide (priv->dock);
priv->timeout = FALSE;
} }
/* /*
* +/- button callbacks. * +/- button callbacks.
*/ */
static gboolean static gboolean
cb_button_timeout (gpointer user_data) button_click (GtkScaleButton *button,
GtkWidget *active)
{ {
GtkScaleButton *button; GtkScaleButtonPrivate *priv = button->priv;
GtkScaleButtonPrivate *priv; GtkAdjustment *adjustment = priv->adjustment;
GtkAdjustment *adjustment; gboolean can_continue = TRUE;
gdouble val; gdouble val;
gboolean res = TRUE;
button = GTK_SCALE_BUTTON (user_data);
priv = button->priv;
if (priv->click_id == 0)
return FALSE;
adjustment = priv->adjustment;
val = gtk_scale_button_get_value (button); val = gtk_scale_button_get_value (button);
val += priv->direction;
if (active == priv->plus_button)
val += gtk_adjustment_get_page_increment (adjustment);
else
val -= gtk_adjustment_get_page_increment (adjustment);
if (val <= gtk_adjustment_get_lower (adjustment)) if (val <= gtk_adjustment_get_lower (adjustment))
{ {
res = FALSE; can_continue = FALSE;
val = gtk_adjustment_get_lower (adjustment); val = gtk_adjustment_get_lower (adjustment);
} }
else if (val > gtk_adjustment_get_upper (adjustment)) else if (val > gtk_adjustment_get_upper (adjustment))
{ {
res = FALSE; can_continue = FALSE;
val = gtk_adjustment_get_upper (adjustment); val = gtk_adjustment_get_upper (adjustment);
} }
gtk_scale_button_set_value (button, val); gtk_scale_button_set_value (button, val);
return can_continue;
}
static gboolean
cb_button_timeout (gpointer user_data)
{
GtkScaleButton *button = GTK_SCALE_BUTTON (user_data);
GtkScaleButtonPrivate *priv = button->priv;
gboolean res;
if (priv->click_id == 0)
return G_SOURCE_REMOVE;
res = button_click (button, priv->active_button);
if (!res) if (!res)
{ {
g_source_remove (priv->click_id); g_source_remove (priv->click_id);
@ -1221,23 +896,19 @@ cb_button_press (GtkWidget *widget,
GdkEventButton *event, GdkEventButton *event,
gpointer user_data) gpointer user_data)
{ {
GtkScaleButton *button; GtkScaleButton *button = GTK_SCALE_BUTTON (user_data);
GtkScaleButtonPrivate *priv; GtkScaleButtonPrivate *priv = button->priv;
GtkAdjustment *adjustment; gint double_click_time;
button = GTK_SCALE_BUTTON (user_data);
priv = button->priv;
adjustment = priv->adjustment;
if (priv->click_id != 0) if (priv->click_id != 0)
g_source_remove (priv->click_id); g_source_remove (priv->click_id);
if (widget == priv->plus_button) priv->active_button = widget;
priv->direction = fabs (gtk_adjustment_get_page_increment (adjustment));
else
priv->direction = - fabs (gtk_adjustment_get_page_increment (adjustment));
priv->click_id = gdk_threads_add_timeout (priv->click_timeout, g_object_get (gtk_widget_get_settings (widget),
"gtk-double-click-time", &double_click_time,
NULL);
priv->click_id = gdk_threads_add_timeout (double_click_time,
cb_button_timeout, cb_button_timeout,
button); button);
g_source_set_name_by_id (priv->click_id, "[gtk+] cb_button_timeout"); g_source_set_name_by_id (priv->click_id, "[gtk+] cb_button_timeout");
@ -1251,11 +922,8 @@ cb_button_release (GtkWidget *widget,
GdkEventButton *event, GdkEventButton *event,
gpointer user_data) gpointer user_data)
{ {
GtkScaleButton *button; GtkScaleButton *button = GTK_SCALE_BUTTON (user_data);
GtkScaleButtonPrivate *priv; GtkScaleButtonPrivate *priv = button->priv;
button = GTK_SCALE_BUTTON (user_data);
priv = button->priv;
if (priv->click_id != 0) if (priv->click_id != 0)
{ {
@ -1267,231 +935,27 @@ cb_button_release (GtkWidget *widget,
} }
static void static void
cb_dock_grab_notify (GtkWidget *widget, cb_button_clicked (GtkWidget *widget,
gboolean was_grabbed,
gpointer user_data)
{
GtkScaleButton *button = (GtkScaleButton *) user_data;
gtk_scale_button_grab_notify (button, was_grabbed);
}
static gboolean
cb_dock_grab_broken_event (GtkWidget *widget,
gboolean was_grabbed,
gpointer user_data)
{
GtkScaleButton *button = (GtkScaleButton *) user_data;
gtk_scale_button_grab_notify (button, FALSE);
return FALSE;
}
/* Scale callbacks */
static void
gtk_scale_button_release_grab (GtkScaleButton *button,
GdkEventButton *event)
{
GdkEventButton *e;
GtkScaleButtonPrivate *priv;
priv = button->priv;
/* ungrab focus */
gdk_device_ungrab (priv->grab_keyboard, event->time);
gdk_device_ungrab (priv->grab_pointer, event->time);
gtk_device_grab_remove (priv->dock, priv->grab_pointer);
priv->grab_keyboard = NULL;
priv->grab_pointer = NULL;
/* hide again */
gtk_widget_hide (priv->dock);
priv->timeout = FALSE;
e = (GdkEventButton *) gdk_event_copy ((GdkEvent *) event);
e->window = gtk_widget_get_window (GTK_WIDGET (button));
e->type = GDK_BUTTON_RELEASE;
gtk_widget_event (GTK_WIDGET (button), (GdkEvent *) e);
e->window = event->window;
gdk_event_free ((GdkEvent *) e);
}
static gboolean
cb_dock_button_press (GtkWidget *widget,
GdkEventButton *event,
gpointer user_data) gpointer user_data)
{ {
GtkScaleButton *button = GTK_SCALE_BUTTON (user_data); GtkScaleButton *button = GTK_SCALE_BUTTON (user_data);
GtkScaleButtonPrivate *priv = button->priv;
if (event->type == GDK_BUTTON_PRESS) if (priv->click_id != 0)
{ return;
gtk_scale_button_release_grab (button, event);
return TRUE;
}
return FALSE; button_click (button, widget);
}
static void
gtk_scale_button_popdown (GtkWidget *widget)
{
GtkScaleButton *button;
GtkScaleButtonPrivate *priv;
button = GTK_SCALE_BUTTON (widget);
priv = button->priv;
/* ungrab focus */
gdk_device_ungrab (priv->grab_keyboard, GDK_CURRENT_TIME);
gdk_device_ungrab (priv->grab_pointer, GDK_CURRENT_TIME);
gtk_device_grab_remove (priv->dock, priv->grab_pointer);
priv->grab_keyboard = NULL;
priv->grab_pointer = NULL;
/* hide again */
gtk_widget_hide (priv->dock);
priv->timeout = FALSE;
}
static gboolean
cb_dock_key_release (GtkWidget *widget,
GdkEventKey *event,
gpointer user_data)
{
if (event->keyval == GDK_KEY_Escape)
{
gtk_scale_button_popdown (GTK_WIDGET (user_data));
return TRUE;
}
if (!gtk_bindings_activate_event (G_OBJECT (widget), event))
{
/* The popup hasn't managed the event, pass onto the button */
gtk_bindings_activate_event (G_OBJECT (user_data), event);
}
return TRUE;
}
static void
cb_scale_grab_notify (GtkWidget *widget,
gboolean was_grabbed,
gpointer user_data)
{
GtkScaleButton *button = (GtkScaleButton *) user_data;
gtk_scale_button_grab_notify (button, was_grabbed);
}
/*
* Scale stuff.
*/
typedef struct _GtkScaleButtonScaleClass
{
GtkScaleClass parent_class;
} GtkScaleButtonScaleClass;
static gboolean gtk_scale_button_scale_press (GtkWidget *widget,
GdkEventButton *event);
static gboolean gtk_scale_button_scale_release (GtkWidget *widget,
GdkEventButton *event);
G_DEFINE_TYPE (GtkScaleButtonScale, _gtk_scale_button_scale, GTK_TYPE_SCALE)
static void
_gtk_scale_button_scale_class_init (GtkScaleButtonScaleClass *klass)
{
GtkWidgetClass *gtkwidget_class = GTK_WIDGET_CLASS (klass);
GtkRangeClass *gtkrange_class = GTK_RANGE_CLASS (klass);
gtkwidget_class->button_press_event = gtk_scale_button_scale_press;
gtkwidget_class->button_release_event = gtk_scale_button_scale_release;
gtkrange_class->value_changed = gtk_scale_button_scale_value_changed;
}
static void
_gtk_scale_button_scale_init (GtkScaleButtonScale *scale)
{
}
static gboolean
gtk_scale_button_scale_press (GtkWidget *widget,
GdkEventButton *event)
{
GtkScaleButtonPrivate *priv;
/* Avoid a crash when using a GtkScaleButtonScale without a GtkScaleButton,
* this can happen while editing gtkscalebutton.ui in Glade
*/
if (!GTK_SCALE_BUTTON_SCALE (widget)->button)
return GTK_WIDGET_CLASS (_gtk_scale_button_scale_parent_class)->button_release_event (widget, event);
priv = GTK_SCALE_BUTTON_SCALE (widget)->button->priv;
/* the scale will grab input; if we have input grabbed, all goes
* horribly wrong, so let's not do that.
*/
gtk_device_grab_remove (priv->dock, event->device);
return GTK_WIDGET_CLASS (_gtk_scale_button_scale_parent_class)->button_press_event (widget, event);
}
static gboolean
gtk_scale_button_scale_release (GtkWidget *widget,
GdkEventButton *event)
{
GtkScaleButton *button = GTK_SCALE_BUTTON_SCALE (widget)->button;
gboolean res;
/* Avoid a crash when using a GtkScaleButtonScale without a GtkScaleButton,
* this can happen while editing gtkscalebutton.ui in Glade
*/
if (!button)
return GTK_WIDGET_CLASS (_gtk_scale_button_scale_parent_class)->button_release_event (widget, event);
if (button->priv->timeout)
{
/* if we did a quick click, leave the window open; else, hide it */
if (event->time > button->priv->pop_time + button->priv->click_timeout)
{
gtk_scale_button_release_grab (button, event);
GTK_WIDGET_CLASS (_gtk_scale_button_scale_parent_class)->button_release_event (widget, event);
return TRUE;
}
button->priv->timeout = FALSE;
}
res = GTK_WIDGET_CLASS (_gtk_scale_button_scale_parent_class)->button_release_event (widget, event);
/* the scale will release input; right after that, we *have to* grab
* it back so we can catch out-of-scale clicks and hide the popup,
* so I basically want a g_signal_connect_after_always(), but I can't
* find that, so we do this complex 'first-call-parent-then-do-actual-
* action' thingy...
*/
gtk_device_grab_add (button->priv->dock, event->device, TRUE);
return res;
} }
static void static void
gtk_scale_button_update_icon (GtkScaleButton *button) gtk_scale_button_update_icon (GtkScaleButton *button)
{ {
GtkScaleButtonPrivate *priv; GtkScaleButtonPrivate *priv = button->priv;
GtkAdjustment *adjustment; GtkAdjustment *adjustment;
gdouble value; gdouble value;
const gchar *name; const gchar *name;
guint num_icons; guint num_icons;
priv = button->priv;
if (!priv->icon_list || priv->icon_list[0] == '\0') if (!priv->icon_list || priv->icon_list[0] == '\0')
{ {
gtk_image_set_from_icon_name (GTK_IMAGE (priv->image), gtk_image_set_from_icon_name (GTK_IMAGE (priv->image),
@ -1545,8 +1009,7 @@ gtk_scale_button_update_icon (GtkScaleButton *button)
gdouble step; gdouble step;
guint i; guint i;
step = (gtk_adjustment_get_upper (adjustment) - gtk_adjustment_get_lower (adjustment)) / (num_icons - 2); step = (gtk_adjustment_get_upper (adjustment) - gtk_adjustment_get_lower (adjustment)) / (num_icons - 2); i = (guint) ((value - gtk_adjustment_get_lower (adjustment)) / step) + 2;
i = (guint) ((value - gtk_adjustment_get_lower (adjustment)) / step) + 2;
g_assert (i < num_icons); g_assert (i < num_icons);
name = priv->icon_list[i]; name = priv->icon_list[i];
} }
@ -1557,17 +1020,12 @@ gtk_scale_button_update_icon (GtkScaleButton *button)
} }
static void static void
gtk_scale_button_scale_value_changed (GtkRange *range) cb_scale_value_changed (GtkRange *range,
gpointer user_data)
{ {
GtkScaleButton *button = GTK_SCALE_BUTTON_SCALE (range)->button; GtkScaleButton *button = user_data;
gdouble value; gdouble value;
/* Avoid a crash when using a GtkScaleButtonScale without a GtkScaleButton,
* this can happen while editing gtkscalebutton.ui in Glade
*/
if (!button)
return;
value = gtk_range_get_value (range); value = gtk_range_get_value (range);
gtk_scale_button_update_icon (button); gtk_scale_button_update_icon (button);
@ -1575,3 +1033,13 @@ gtk_scale_button_scale_value_changed (GtkRange *range)
g_signal_emit (button, signals[VALUE_CHANGED], 0, value); g_signal_emit (button, signals[VALUE_CHANGED], 0, value);
g_object_notify (G_OBJECT (button), "value"); g_object_notify (G_OBJECT (button), "value");
} }
static void
cb_popup_mapped (GtkWidget *popup,
gpointer user_data)
{
GtkScaleButton *button = user_data;
GtkScaleButtonPrivate *priv = button->priv;
gtk_widget_grab_focus (priv->scale);
}

View File

@ -20,35 +20,36 @@
<property name="step_increment">2</property> <property name="step_increment">2</property>
<property name="page_increment">20</property> <property name="page_increment">20</property>
</object> </object>
<object class="GtkWindow" id="dock"> <object class="GtkPopover" id="dock">
<property name="name">gtk-scalebutton-popup-window</property> <signal name="map" handler="cb_popup_mapped" swapped="no"/>
<property name="can_focus">False</property>
<property name="type">popup</property>
<property name="decorated">False</property>
<signal name="button-press-event" handler="cb_dock_button_press" swapped="no"/>
<signal name="grab-broken-event" handler="cb_dock_grab_broken_event" swapped="no"/>
<signal name="grab-notify" handler="cb_dock_grab_notify" swapped="no"/>
<signal name="key-release-event" handler="cb_dock_key_release" swapped="no"/>
<child>
<object class="GtkFrame" id="frame1">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="label_xalign">0</property>
<property name="shadow_type">out</property>
<child> <child>
<object class="GtkBox" id="box"> <object class="GtkBox" id="box">
<property name="visible">True</property> <property name="visible">True</property>
<property name="can_focus">False</property> <property name="can_focus">False</property>
<property name="orientation">vertical</property> <property name="orientation">vertical</property>
<property name="margin">4</property>
<property name="spacing">4</property>
<child> <child>
<object class="GtkButton" id="plus_button"> <object class="GtkButton" id="plus_button">
<property name="label">+</property>
<property name="visible">True</property> <property name="visible">True</property>
<property name="can_focus">True</property> <property name="can_focus">True</property>
<property name="receives_default">True</property> <property name="receives_default">True</property>
<property name="relief">none</property> <property name="relief">none</property>
<property name="halign">center</property>
<property name="valign">center</property>
<style>
<class name="image-button"/>
</style>
<child>
<object class="GtkImage">
<property name="visible">True</property>
<property name="icon-name">list-add-symbolic</property>
<property name="icon-size">0</property>
</object>
</child>
<signal name="button-press-event" handler="cb_button_press" swapped="no"/> <signal name="button-press-event" handler="cb_button_press" swapped="no"/>
<signal name="button-release-event" handler="cb_button_release" swapped="no"/> <signal name="button-release-event" handler="cb_button_release" swapped="no"/>
<signal name="clicked" handler="cb_button_clicked" swapped="no"/>
</object> </object>
<packing> <packing>
<property name="expand">False</property> <property name="expand">False</property>
@ -57,7 +58,7 @@
</packing> </packing>
</child> </child>
<child> <child>
<object class="GtkScaleButtonScale" id="scale"> <object class="GtkScale" id="scale">
<property name="height_request">100</property> <property name="height_request">100</property>
<property name="visible">True</property> <property name="visible">True</property>
<property name="can_focus">True</property> <property name="can_focus">True</property>
@ -66,7 +67,7 @@
<property name="inverted">True</property> <property name="inverted">True</property>
<property name="round_digits">1</property> <property name="round_digits">1</property>
<property name="draw_value">False</property> <property name="draw_value">False</property>
<signal name="grab-notify" handler="cb_scale_grab_notify" swapped="no"/> <signal name="value-changed" handler="cb_scale_value_changed" swapped="no"/>
</object> </object>
<packing> <packing>
<property name="expand">False</property> <property name="expand">False</property>
@ -76,13 +77,25 @@
</child> </child>
<child> <child>
<object class="GtkButton" id="minus_button"> <object class="GtkButton" id="minus_button">
<property name="label">-</property>
<property name="visible">True</property> <property name="visible">True</property>
<property name="can_focus">True</property> <property name="can_focus">True</property>
<property name="receives_default">True</property> <property name="receives_default">True</property>
<property name="relief">none</property> <property name="relief">none</property>
<property name="halign">center</property>
<property name="valign">center</property>
<style>
<class name="image-button"/>
</style>
<child>
<object class="GtkImage">
<property name="visible">True</property>
<property name="icon-name">list-remove-symbolic</property>
<property name="icon-size">0</property>
</object>
</child>
<signal name="button-press-event" handler="cb_button_press" swapped="no"/> <signal name="button-press-event" handler="cb_button_press" swapped="no"/>
<signal name="button-release-event" handler="cb_button_release" swapped="no"/> <signal name="button-release-event" handler="cb_button_release" swapped="no"/>
<signal name="clicked" handler="cb_button_clicked" swapped="no"/>
</object> </object>
<packing> <packing>
<property name="expand">False</property> <property name="expand">False</property>
@ -92,10 +105,5 @@
</child> </child>
</object> </object>
</child> </child>
<child type="label_item">
<placeholder/>
</child>
</object>
</child>
</object> </object>
</interface> </interface>