From 2a72e7b7b86d1bdb3a6a818b94cb21bc8f611127 Mon Sep 17 00:00:00 2001 From: Michael Natterer Date: Tue, 24 Jan 2012 12:12:34 +0100 Subject: gtk: Implement smooth scrolling in scrolledwindow/range If delta_x/y information is provided in scroll events, use it to modify the underlying adjustment in steps proportional to the deltas provided. If the child widget of a scrolledwindow doesn't set GDK_SMOOTH_SCROLL_MASK, regular scroll events will be dispatched, and still handled by these 2 widgets. --- gtk/gtkrange.c | 65 +++++++++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 49 insertions(+), 16 deletions(-) (limited to 'gtk/gtkrange.c') diff --git a/gtk/gtkrange.c b/gtk/gtkrange.c index b920d001d5..b38fa89da0 100644 --- a/gtk/gtkrange.c +++ b/gtk/gtkrange.c @@ -1728,9 +1728,11 @@ gtk_range_realize (GtkWidget *widget) attributes.wclass = GDK_INPUT_ONLY; attributes.event_mask = gtk_widget_get_events (widget); attributes.event_mask |= (GDK_BUTTON_PRESS_MASK | - GDK_BUTTON_RELEASE_MASK | - GDK_ENTER_NOTIFY_MASK | - GDK_LEAVE_NOTIFY_MASK | + GDK_BUTTON_RELEASE_MASK | + GDK_SCROLL_MASK | + GDK_SMOOTH_SCROLL_MASK | + GDK_ENTER_NOTIFY_MASK | + GDK_LEAVE_NOTIFY_MASK | GDK_POINTER_MOTION_MASK | GDK_POINTER_MOTION_HINT_MASK); @@ -2784,7 +2786,7 @@ gtk_range_button_release (GtkWidget *widget, /** * _gtk_range_get_wheel_delta: * @range: a #GtkRange - * @direction: A #GdkScrollDirection + * @event: A #GdkEventScroll * * Returns a good step value for the mouse wheel. * @@ -2793,28 +2795,59 @@ gtk_range_button_release (GtkWidget *widget, * Since: 2.4 **/ gdouble -_gtk_range_get_wheel_delta (GtkRange *range, - GdkScrollDirection direction) +_gtk_range_get_wheel_delta (GtkRange *range, + GdkEventScroll *event) { GtkRangePrivate *priv = range->priv; GtkAdjustment *adjustment = priv->adjustment; + gdouble dx, dy; gdouble delta; + gdouble page_size; - if (GTK_IS_SCROLLBAR (range)) - delta = pow (gtk_adjustment_get_page_size (adjustment), 2.0 / 3.0); + page_size = gtk_adjustment_get_page_size (adjustment); + + if (gdk_event_get_scroll_deltas ((GdkEvent *) event, &dx, &dy)) + { + GtkAllocation allocation; + + gtk_widget_get_allocation (GTK_WIDGET (range), &allocation); + + if (dx != 0 && + gtk_orientable_get_orientation (GTK_ORIENTABLE (range)) == GTK_ORIENTATION_HORIZONTAL) + { + if (GTK_IS_SCROLLBAR (range) && page_size > 0) + delta = dx * page_size / allocation.width; + else + delta = dx * (gtk_adjustment_get_upper (adjustment) - + gtk_adjustment_get_lower (adjustment)) / allocation.width; + } + else + { + if (GTK_IS_SCROLLBAR (range) && page_size > 0) + delta = dy * page_size / allocation.height; + else + delta = dy * (gtk_adjustment_get_upper (adjustment) - + gtk_adjustment_get_lower (adjustment)) / allocation.height; + } + } else - delta = gtk_adjustment_get_step_increment (adjustment) * 2; - - if (direction == GDK_SCROLL_UP || - direction == GDK_SCROLL_LEFT) - delta = - delta; - + { + if (GTK_IS_SCROLLBAR (range)) + delta = pow (page_size, 2.0 / 3.0); + else + delta = gtk_adjustment_get_page_increment (adjustment) * 2; + + if (event->direction == GDK_SCROLL_UP || + event->direction == GDK_SCROLL_LEFT) + delta = - delta; + } + if (priv->inverted) delta = - delta; return delta; } - + static gboolean gtk_range_scroll_event (GtkWidget *widget, GdkEventScroll *event) @@ -2827,7 +2860,7 @@ gtk_range_scroll_event (GtkWidget *widget, gdouble delta; gboolean handled; - delta = _gtk_range_get_wheel_delta (range, event->direction); + delta = _gtk_range_get_wheel_delta (range, event); g_signal_emit (range, signals[CHANGE_VALUE], 0, GTK_SCROLL_JUMP, gtk_adjustment_get_value (priv->adjustment) + delta, -- cgit v1.2.1