1 /*
2  * Copyright (C) 2009, 2010 Hermann Meyer, James Warden, Andreas Degert
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17  */
18 
19 #include "GxEQSlider.h"
20 
21 #define P_(s) (s)   // FIXME -> gettext
22 
23 #define get_stock_id(widget) (GX_EQ_SLIDER_CLASS(GTK_WIDGET_GET_CLASS(widget))->parent_class.stock_id)
24 
25 static gboolean gx_eq_slider_draw (GtkWidget *widget, cairo_t *cr);
26 static void gx_eq_slider_get_preferred_width (GtkWidget *widget, gint *min_width, gint *natural_width);
27 static void gx_eq_slider_get_preferred_height (GtkWidget *widget, gint *min_height, gint *natural_height);
28 static void gx_eq_slider_size_request (GtkWidget *widget, gint *width, gint *height);
29 static gboolean gx_eq_slider_button_press (GtkWidget *widget, GdkEventButton *event);
30 static gboolean gx_eq_slider_pointer_motion (GtkWidget *widget, GdkEventMotion *event);
31 
G_DEFINE_TYPE(GxEQSlider,gx_eq_slider,GX_TYPE_VSLIDER)32 G_DEFINE_TYPE(GxEQSlider, gx_eq_slider, GX_TYPE_VSLIDER)
33 
34 static void gx_eq_slider_class_init(GxEQSliderClass *klass)
35 {
36 	GtkWidgetClass *widget_class = (GtkWidgetClass*) klass;
37 
38 	widget_class->draw = gx_eq_slider_draw;
39 	widget_class->get_preferred_width = gx_eq_slider_get_preferred_width;
40 	widget_class->get_preferred_height = gx_eq_slider_get_preferred_height;
41 	widget_class->button_press_event = gx_eq_slider_button_press;
42 	widget_class->motion_notify_event = gx_eq_slider_pointer_motion;
43 	widget_class->enter_notify_event = NULL;
44 	widget_class->leave_notify_event = NULL;
45 	klass->parent_class.stock_id = "eqslider";
46 	gtk_widget_class_install_style_property(
47 		widget_class,
48 		g_param_spec_int("slider-width",P_("size of slider"),
49 				 P_("Height of movable part of vslider"),
50 		                 0, 100, 5, GParamFlags(G_PARAM_READABLE|G_PARAM_STATIC_STRINGS)));
51 }
52 
gx_eq_slider_get_preferred_width(GtkWidget * widget,gint * min_width,gint * natural_width)53 static void gx_eq_slider_get_preferred_width (GtkWidget *widget, gint *min_width, gint *natural_width)
54 {
55 	gint width, height;
56 	gx_eq_slider_size_request(widget, &width, &height);
57 
58 	if (min_width) {
59 		*min_width = width;
60 	}
61 	if (natural_width) {
62 		*natural_width = width;
63 	}
64 }
65 
gx_eq_slider_get_preferred_height(GtkWidget * widget,gint * min_height,gint * natural_height)66 static void gx_eq_slider_get_preferred_height (GtkWidget *widget, gint *min_height, gint *natural_height)
67 {
68 	gint width, height;
69 	gx_eq_slider_size_request(widget, &width, &height);
70 
71 	if (min_height) {
72 		*min_height = height;
73 	}
74 	if (natural_height) {
75 		*natural_height = height;
76 	}
77 }
78 
gx_eq_slider_size_request(GtkWidget * widget,gint * width,gint * height)79 static void gx_eq_slider_size_request (GtkWidget *widget, gint *width, gint *height)
80 {
81 	g_assert(GX_IS_EQ_SLIDER(widget));
82 	gint slider_height;
83 	gtk_widget_style_get(widget, "slider-width", &slider_height, NULL);
84 	GdkPixbuf *pb = gtk_icon_theme_load_icon(gtk_icon_theme_get_default(),
85 											 get_stock_id(widget), -1,
86 											 GTK_ICON_LOOKUP_GENERIC_FALLBACK, nullptr);
87 	if (GDK_IS_PIXBUF(pb)) {
88 		*width = gdk_pixbuf_get_width(pb);
89 		*height = (gdk_pixbuf_get_height(pb) + slider_height) / 2;
90 		g_object_unref(pb);
91 	} else {
92 		g_warning("EQSlider icon is NULL");
93 		*width = 0;
94 		*height = slider_height / 2;
95 	}
96 	_gx_regler_calc_size_request(GX_REGLER(widget), width, height, FALSE);
97 }
98 
eq_slider_expose(cairo_t * cr,GdkRectangle * rect,gdouble sliderstate,GdkPixbuf * image)99 static void eq_slider_expose(
100 	cairo_t *cr, GdkRectangle *rect, gdouble sliderstate, GdkPixbuf *image)
101 {
102 	gdk_cairo_set_source_pixbuf (cr, image, rect->x, rect->y - (gint)sliderstate);
103 	cairo_rectangle(cr, rect->x, rect->y, rect->width, rect->height);
104 	cairo_fill(cr);
105 }
106 
gx_eq_slider_draw(GtkWidget * widget,cairo_t * cr)107 static gboolean gx_eq_slider_draw(GtkWidget *widget, cairo_t *cr)
108 {
109 	g_assert(GX_IS_EQ_SLIDER(widget));
110 	gint slider_height;
111 	GdkRectangle image_rect, value_rect;
112 	gtk_widget_style_get(widget, "slider-width", &slider_height, NULL);
113 	GdkPixbuf *pb = gtk_icon_theme_load_icon(gtk_icon_theme_get_default(),
114 											 get_stock_id(widget), -1,
115 											 GTK_ICON_LOOKUP_GENERIC_FALLBACK, nullptr);
116 	if (GDK_IS_PIXBUF(pb)) {
117 		image_rect.width = gdk_pixbuf_get_width(pb);
118 		image_rect.height = (gdk_pixbuf_get_height(pb) + slider_height) / 2;
119 		gdouble sliderstate = _gx_regler_get_step_pos(GX_REGLER(widget), image_rect.height-slider_height);
120 		_gx_regler_get_positions(GX_REGLER(widget), &image_rect, &value_rect, false);
121 		eq_slider_expose(cr, &image_rect, sliderstate, pb);
122 		_gx_regler_simple_display_value(GX_REGLER(widget), cr, &value_rect);
123 		g_object_unref(pb);
124 	} else {
125 		g_warning("EQSlider icon is NULL");
126 	}
127 	return FALSE;
128 }
129 
get_width_height(GtkWidget * widget,GdkRectangle * r)130 static inline void get_width_height(GtkWidget *widget, GdkRectangle *r)
131 {
132 	GdkPixbuf *pb = gtk_icon_theme_load_icon(gtk_icon_theme_get_default(),
133 											 get_stock_id(widget), -1,
134 											 GTK_ICON_LOOKUP_GENERIC_FALLBACK, nullptr);
135 	r->x = r->y = 0;
136 	if (GDK_IS_PIXBUF(pb)) {
137 		r->width = gdk_pixbuf_get_width(pb);
138 		r->height = gdk_pixbuf_get_height(pb);
139 		g_object_unref(pb);
140 	} else {
141 		r->width = r->height = 0;
142 	}
143 }
144 
slider_set_from_pointer(GtkWidget * widget,int state,gdouble x,gdouble y,gboolean drag,gint button,GdkEventButton * event)145 static gboolean slider_set_from_pointer(GtkWidget *widget, int state, gdouble x, gdouble y, gboolean drag, gint button, GdkEventButton *event)
146 {
147 	gint slider_height;
148 	GdkRectangle image_rect, value_rect;
149 	gtk_widget_style_get(widget, "slider-width", &slider_height, NULL);
150 	get_width_height(widget, &image_rect);
151 	image_rect.height = (image_rect.height + slider_height) / 2;
152 
153 	_gx_regler_get_positions(GX_REGLER(widget), &image_rect, &value_rect, false);
154 	if (!drag) {
155 		if (_gx_regler_check_display_popup(GX_REGLER(widget), &image_rect, &value_rect, event)) {
156 			return FALSE;
157 		}
158 	}
159 	gint height = image_rect.height - slider_height;
160 	gint off =  image_rect.y + slider_height/2;
161 	GtkAdjustment *adj = gtk_range_get_adjustment(GTK_RANGE(widget));
162 	gdouble upper = gtk_adjustment_get_upper(adj);
163 	gdouble lower = gtk_adjustment_get_lower(adj);
164 	static double last_y = 2e20;
165 	if (!drag) {
166 		last_y = y;
167 		if (event && event->type == GDK_2BUTTON_PRESS) {
168 		    double value = upper - ((y-off) / height) * (upper - lower);
169 			gtk_range_set_value(GTK_RANGE(widget), value);
170 		}
171 	} else {
172 		double value = ((last_y - y) / height) * (upper - lower);
173 		if (state & (GDK_CONTROL_MASK|GDK_SHIFT_MASK)) {
174 			value *= 0.1;
175 		}
176 		last_y = y;
177 		gtk_range_set_value(GTK_RANGE(widget), gtk_adjustment_get_value(adj) + value);
178 	}
179 	return TRUE;
180 }
181 
gx_eq_slider_button_press(GtkWidget * widget,GdkEventButton * event)182 static gboolean gx_eq_slider_button_press (GtkWidget *widget, GdkEventButton *event)
183 {
184 	g_assert(GX_IS_EQ_SLIDER(widget));
185 	if (event->button != 1 && event->button != 3) {
186 		return FALSE;
187 	}
188 	gtk_widget_grab_focus(widget);
189 	if (slider_set_from_pointer(widget, event->state, event->x, event->y, FALSE, event->button, event)) {
190 		gtk_grab_add(widget);
191 	}
192 	return FALSE;
193 }
194 
gx_eq_slider_pointer_motion(GtkWidget * widget,GdkEventMotion * event)195 static gboolean gx_eq_slider_pointer_motion(GtkWidget *widget, GdkEventMotion *event)
196 {
197 	g_assert(GX_IS_EQ_SLIDER(widget));
198 	if (!gtk_widget_has_grab(widget)) {
199 		return FALSE;
200 	}
201 	gdk_event_request_motions (event);
202 	slider_set_from_pointer(widget, event->state, event->x, event->y, TRUE, 0, NULL);
203 	return FALSE;
204 }
205 
206 
gx_eq_slider_init(GxEQSlider * eq_slider)207 static void gx_eq_slider_init(GxEQSlider *eq_slider)
208 {
209 	gtk_widget_set_name (GTK_WIDGET(eq_slider),"rack_slider");
210 }
211