2 Copyright (C) 1998-99 Paul Barton-Davis
3 This program is free software; you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 2 of the License, or
6 (at your option) any later version.
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 GNU General Public License for more details.
13 You should have received a copy of the GNU General Public License
14 along with this program; if not, write to the Free Software
15 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17 $Id: motionfeedback.cc,v 1.5 2004/03/01 03:44:19 pauld Exp $
23 #include <stdio.h> /* for snprintf, grrr */
25 #include <gdk/gdkkeysyms.h>
28 #include "gtkmm2ext/motionfeedback.h"
29 #include "gtkmm2ext/keyboard.h"
30 #include "gtkmm2ext/prolooks-helpers.h"
33 using namespace Gtkmm2ext
;
36 MotionFeedback::MotionFeedback (Glib::RefPtr
<Gdk::Pixbuf
> pix
,
38 const char *widget_name
,
40 bool with_numeric_display
,
50 char value_name
[1024];
53 i_own_my_adjustment
= true;
54 set_adjustment (new Adjustment (0, 0, 10000, 1, 10, 0));
56 i_own_my_adjustment
= false;
60 default_value
= adjustment
->get_value();
62 HBox
* hpacker
= manage (new HBox
);
63 hpacker
->pack_start (pixwin
, true, false);
65 pack_start (*hpacker
, false, false);
68 if (with_numeric_display
) {
70 value_packer
= new HBox
;
71 value
= new SpinButton (*adjustment
);
72 value_packer
->pack_start (*value
, false, false);
75 value
->set_digits (abs ((int) ceil (log10 (step_inc
))));
78 pack_start (*value_packer
, false, false);
81 snprintf (value_name
, sizeof(value_name
), "%sValue", widget_name
);
82 value
->set_name (value_name
);
88 adjustment
->signal_value_changed().connect (mem_fun (*this, &MotionFeedback::adjustment_changed
));
90 pixwin
.set_events (Gdk::BUTTON_PRESS_MASK
|
91 Gdk::BUTTON_RELEASE_MASK
|
92 Gdk::POINTER_MOTION_MASK
|
93 Gdk::ENTER_NOTIFY_MASK
|
94 Gdk::LEAVE_NOTIFY_MASK
|
97 Gdk::KEY_RELEASE_MASK
);
99 pixwin
.set_flags (CAN_FOCUS
);
101 /* Proxy all important events on the pixwin to ourselves */
103 pixwin
.signal_button_press_event().connect(mem_fun (*this,&MotionFeedback::pixwin_button_press_event
));
104 pixwin
.signal_button_release_event().connect(mem_fun (*this,&MotionFeedback::pixwin_button_release_event
));
105 pixwin
.signal_motion_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_motion_notify_event
));
106 pixwin
.signal_enter_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_enter_notify_event
));
107 pixwin
.signal_leave_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_leave_notify_event
));
108 pixwin
.signal_key_press_event().connect(mem_fun (*this,&MotionFeedback::pixwin_key_press_event
));
109 pixwin
.signal_scroll_event().connect(mem_fun (*this,&MotionFeedback::pixwin_scroll_event
));
110 pixwin
.signal_expose_event().connect(mem_fun (*this,&MotionFeedback::pixwin_expose_event
), true);
111 pixwin
.signal_size_request().connect(mem_fun (*this,&MotionFeedback::pixwin_size_request
));
112 pixwin
.signal_realize().connect(mem_fun (*this,&MotionFeedback::pixwin_realized
));
115 MotionFeedback::~MotionFeedback()
118 if (i_own_my_adjustment
) {
127 MotionFeedback::set_adjustment (Adjustment
*adj
)
132 value
->set_adjustment (*adj
);
135 _lower
= adj
->get_lower();
136 _upper
= adj
->get_upper();
137 _range
= _upper
- _lower
;
138 step_inc
= adj
->get_step_increment();
139 page_inc
= adj
->get_page_increment();
143 MotionFeedback::pixwin_button_press_event (GdkEventButton
*ev
)
145 if (binding_proxy
.button_press_handler (ev
)) {
149 switch (ev
->button
) {
151 return FALSE
; /* XXX why ? */
154 grab_is_fine
= false;
161 gtk_grab_add(GTK_WIDGET(pixwin
.gobj()));
162 grabbed_y
= ev
->y_root
;
163 grabbed_x
= ev
->x_root
;
165 /* XXX should we return TRUE ? */
171 MotionFeedback::pixwin_button_release_event (GdkEventButton
*ev
)
173 switch (ev
->button
) {
175 if (pixwin
.has_grab()) {
178 (GTK_WIDGET(pixwin
.gobj()));
181 if (Keyboard::modifier_state_equals (ev
->state
, Keyboard::TertiaryModifier
)) {
182 /* shift click back to the default */
183 adjustment
->set_value (default_value
);
189 if (pixwin
.has_grab()) {
192 (GTK_WIDGET(pixwin
.gobj()));
198 return VBox::on_button_release_event (ev
);
202 MotionFeedback::pixwin_motion_notify_event (GdkEventMotion
*ev
)
208 if (!pixwin
.has_grab()) {
209 return VBox::on_motion_notify_event (ev
);
212 multiplier
= ((ev
->state
& Keyboard::TertiaryModifier
) ? 100 : 1) *
213 ((ev
->state
& Keyboard::SecondaryModifier
) ? 10 : 1) *
214 ((ev
->state
& Keyboard::PrimaryModifier
) ? 2 : 1);
217 if (ev
->state
& Gdk::BUTTON1_MASK
) {
219 y_delta
= grabbed_y
- ev
->y_root
;
220 grabbed_y
= ev
->y_root
;
222 x_delta
= ev
->x_root
- grabbed_x
;
224 if (y_delta
== 0) return TRUE
;
226 y_delta
*= 1 + (x_delta
/100);
227 y_delta
*= multiplier
;
230 adjustment
->set_value (adjustment
->get_value() +
231 ((grab_is_fine
? step_inc
: page_inc
) * y_delta
));
233 } else if (ev
->state
& Gdk::BUTTON3_MASK
) {
235 double range
= adjustment
->get_upper() - adjustment
->get_lower();
236 double x
= ev
->x
- subwidth
/2;
237 double y
= - ev
->y
+ subwidth
/2;
238 double angle
= std::atan2 (y
, x
) / M_PI
;
244 angle
= -(2.0/3.0) * (angle
- 1.25);
247 angle
+= adjustment
->get_lower();
249 adjustment
->set_value (angle
);
257 MotionFeedback::pixwin_enter_notify_event (GdkEventCrossing
*ev
)
264 MotionFeedback::pixwin_leave_notify_event (GdkEventCrossing
*ev
)
266 pixwin
.unset_flags (HAS_FOCUS
);
271 MotionFeedback::pixwin_key_press_event (GdkEventKey
*ev
)
277 multiplier
= ((ev
->state
& Keyboard::TertiaryModifier
) ? 100 : 1) *
278 ((ev
->state
& Keyboard::SecondaryModifier
) ? 10 : 1) *
279 ((ev
->state
& Keyboard::PrimaryModifier
) ? 2 : 1);
281 switch (ev
->keyval
) {
284 curval
= adjustment
->get_value();
285 adjustment
->set_value (curval
+ (multiplier
* page_inc
));
290 curval
= adjustment
->get_value();
291 adjustment
->set_value (curval
- (multiplier
* page_inc
));
296 curval
= adjustment
->get_value();
297 adjustment
->set_value (curval
+ (multiplier
* step_inc
));
302 curval
= adjustment
->get_value();
303 adjustment
->set_value (curval
- (multiplier
* step_inc
));
308 adjustment
->set_value (_lower
);
313 adjustment
->set_value (_upper
);
321 MotionFeedback::adjustment_changed ()
323 pixwin
.queue_draw ();
327 MotionFeedback::core_draw (cairo_t
* cr
, int phase
, double radius
, double x
, double y
)
339 double start_angle_x
;
340 double start_angle_y
;
343 double progress_width
;
344 double progress_radius
;
345 double progress_radius_inner
;
346 double progress_radius_outer
;
347 double knob_disc_radius
;
348 cairo_pattern_t
* pattern
;
349 double progress_rim_width
;
350 cairo_pattern_t
* progress_shine
;
352 cairo_pattern_t
* knob_ripples
;
354 g_return_if_fail (cr
!= NULL
);
356 cairo_set_source_rgba (cr
, 0.75, 0.75, 0.75, (double) 1.0);
357 cairo_rectangle (cr
, (double) 0, (double) 0, subwidth
, subheight
);
367 value
= (phase
* 1.0) / (65 - 1);
369 start_angle
= ((180 - 65) * G_PI
) / 180;
370 end_angle
= ((360 + 65) * G_PI
) / 180;
372 value_angle
= start_angle
+ (value
* (end_angle
- start_angle
));
373 value_x
= cos (value_angle
);
374 value_y
= sin (value_angle
);
375 start_angle_x
= cos (start_angle
);
376 start_angle_y
= sin (start_angle
);
377 end_angle_x
= cos (end_angle
);
378 end_angle_y
= sin (end_angle
);
380 //cairo_translate (cr, x, (double) 0);
381 cairo_scale (cr
, (2.0 * radius
) / width
, (2.0 * radius
) / height
);
382 //cairo_translate (cr, -xc, (double) 0);
384 pattern
= prolooks_create_gradient_str ((double) 32, (double) 16, (double) 75, (double) 16, "#d4c8b9", "#ae977b", 1.0, 1.0);
385 cairo_set_source (cr
, pattern
);
386 cairo_pattern_destroy (pattern
);
387 cairo_set_line_width (cr
, 2.0);
388 cairo_arc (cr
, xc
, yc
, 31.5, 0.0, 2 * G_PI
);
391 progress_width
= 10.0;
392 progress_radius
= 40.0;
393 progress_radius_inner
= progress_radius
- (progress_width
/ 2.0);
394 progress_radius_outer
= progress_radius
+ (progress_width
/ 2.0);
395 knob_disc_radius
= progress_radius_inner
- 5.0;
397 pattern
= prolooks_create_gradient_str ((double) 20, (double) 20, (double) 89, (double) 87, "#2f2f4c", "#090a0d", 1.0, 1.0);
398 cairo_set_source (cr
, pattern
);
399 cairo_pattern_destroy (pattern
);
400 cairo_set_line_width (cr
, progress_width
);
401 cairo_arc (cr
, xc
, yc
, progress_radius
, start_angle
, end_angle
);
404 pattern
= prolooks_create_gradient ((double) 20, (double) 20, (double) 89, (double) 87, &lamp_bright
, &lamp_dark
, 1.0, 1.0);
405 cairo_set_source (cr
, pattern
);
406 cairo_pattern_destroy (pattern
);
407 cairo_set_line_width (cr
, progress_width
);
408 cairo_arc (cr
, xc
, yc
, progress_radius
, start_angle
, value_angle
);
411 cairo_set_line_cap (cr
, CAIRO_LINE_CAP_ROUND
);
412 progress_rim_width
= 2.0;
413 cairo_set_line_width (cr
, progress_rim_width
);
414 pattern
= prolooks_create_gradient_str ((double) 18, (double) 79, (double) 35, (double) 79, "#dfd5c9", "#dfd5c9", 1.0, 0.0);
415 cairo_set_source (cr
, pattern
);
416 cairo_pattern_destroy (pattern
);
417 cairo_move_to (cr
, xc
+ (progress_radius_outer
* start_angle_x
), yc
+ (progress_radius_outer
* start_angle_y
));
418 cairo_line_to (cr
, xc
+ (progress_radius_inner
* start_angle_x
), yc
+ (progress_radius_inner
* start_angle_y
));
421 prolooks_set_source_color_string (cr
, "#000000", 1.0);
422 cairo_move_to (cr
, xc
+ (progress_radius_outer
* end_angle_x
), yc
+ (progress_radius_outer
* end_angle_y
));
423 cairo_line_to (cr
, xc
+ (progress_radius_inner
* end_angle_x
), yc
+ (progress_radius_inner
* end_angle_y
));
426 // pattern = prolooks_create_gradient_str ((double) 95, (double) 6, (double) 5, (double) 44, "#dfd5c9", "#b0a090", 1.0, 1.0);
427 pattern
= prolooks_create_gradient_str ((double) 95, (double) 6, (double) 5, (double) 44, "#000000", "#000000", 1.0, 1.0);
428 cairo_set_source (cr
, pattern
);
429 cairo_pattern_destroy (pattern
);
430 cairo_arc (cr
, xc
, yc
, progress_radius_outer
, start_angle
, end_angle
);
433 cairo_set_line_cap (cr
, CAIRO_LINE_CAP_BUTT
);
434 pattern
= prolooks_create_gradient ((double) 20, (double) 20, (double) 89, (double) 87, &lamp_bright
, &lamp_dark
, 0.25, 0.25);
435 cairo_set_source (cr
, pattern
);
436 cairo_pattern_destroy (pattern
);
437 cairo_set_line_width (cr
, progress_width
);
438 cairo_arc (cr
, xc
, yc
, progress_radius
, start_angle
, value_angle
+ (G_PI
/ 180.0));
441 progress_shine
= prolooks_create_gradient_str ((double) 89, (double) 73, (double) 34, (double) 16, "#ffffff", "#ffffff", 0.3, 0.04);
442 cairo_pattern_add_color_stop_rgba (progress_shine
, 0.5, 1.0, 1.0, 1.0, 0.0);
444 cairo_pattern_add_color_stop_rgba (progress_shine
, 0.75, 1.0, 1.0, 1.0, 0.3);
446 cairo_pattern_add_color_stop_rgba (progress_shine
, 0.75, 1.0, 1.0, 1.0, 0.2);
448 cairo_set_source (cr
, progress_shine
);
449 cairo_set_line_width (cr
, progress_width
);
450 cairo_arc (cr
, xc
, yc
, progress_radius
, start_angle
, end_angle
);
452 cairo_pattern_destroy (progress_shine
);
454 cairo_set_line_width (cr
, 1.0);
455 cairo_set_source_rgb (cr
, 0.0, 0.0, 0.0);
456 cairo_arc (cr
, xc
, yc
, progress_radius_inner
, (double) 0, 2 * G_PI
);
457 pattern
= prolooks_create_gradient_str ((double) 35, (double) 31, (double) 75, (double) 72, "#68625c", "#44494b", 1.0, 1.0);
458 cairo_set_source (cr
, pattern
);
459 cairo_pattern_destroy (pattern
);
461 cairo_set_source_rgb (cr
, (double) 0, (double) 0, (double) 0);
462 cairo_arc (cr
, xc
, yc
, progress_radius_inner
, (double) 0, 2 * G_PI
);
465 pattern
= prolooks_create_gradient_str ((double) 42, (double) 34, (double) 68, (double) 70, "#e7ecef", "#9cafb8", 1.0, 1.0);
466 cairo_set_source (cr
, pattern
);
467 cairo_pattern_destroy (pattern
);
468 cairo_arc (cr
, xc
, yc
, knob_disc_radius
, (double) 0, 2 * G_PI
);
471 cairo_set_line_width (cr
, 2.0);
472 degrees
= G_PI
/ 180.0;
473 pattern
= prolooks_create_gradient_str ((double) 38, (double) 34, (double) 70, (double) 68, "#ffffff", "#caddf2", 0.2, 0.2);
474 cairo_set_source (cr
, pattern
);
475 cairo_pattern_destroy (pattern
);
476 cairo_move_to (cr
, xc
, yc
);
477 cairo_arc (cr
, xc
, yc
, knob_disc_radius
- 1, (-154) * degrees
, (-120) * degrees
);
478 cairo_move_to (cr
, xc
, yc
);
479 cairo_arc (cr
, xc
, yc
, knob_disc_radius
- 1, (G_PI
/ 2.0) - (60 * degrees
), (G_PI
/ 2.0) - (29 * degrees
));
482 pattern
= prolooks_create_gradient_str ((double) 50, (double) 40, (double) 62, (double) 60, "#a1adb6", "#47535c", 0.07, 0.15);
483 cairo_set_source (cr
, pattern
);
484 cairo_pattern_destroy (pattern
);
485 cairo_move_to (cr
, xc
, yc
);
486 cairo_arc (cr
, xc
, yc
, knob_disc_radius
- 1, (-67) * degrees
, (-27) * degrees
);
487 cairo_move_to (cr
, xc
, yc
);
488 cairo_arc (cr
, xc
, yc
, knob_disc_radius
- 1, G_PI
- (67 * degrees
), G_PI
- (27 * degrees
));
491 knob_ripples
= cairo_pattern_create_radial (xc
, yc
, (double) 0, xc
, yc
, (double) 4);
492 prolooks_add_color_stop_str (knob_ripples
, 0.0, "#e7ecef", 0.05);
493 prolooks_add_color_stop_str (knob_ripples
, 0.5, "#58717d", 0.05);
494 prolooks_add_color_stop_str (knob_ripples
, 0.75, "#d1d9de", 0.05);
495 prolooks_add_color_stop_str (knob_ripples
, 1.0, "#5d7682", 0.05);
496 cairo_pattern_set_extend (knob_ripples
, CAIRO_EXTEND_REPEAT
);
497 cairo_set_line_width (cr
, 0.0);
498 cairo_set_source (cr
, knob_ripples
);
499 cairo_arc (cr
, xc
, yc
, knob_disc_radius
, (double) 0, 2 * G_PI
);
503 cairo_translate (cr
, xc
+ (knob_disc_radius
* value_x
), yc
+ (knob_disc_radius
* value_y
));
504 cairo_rotate (cr
, value_angle
- G_PI
);
505 cairo_set_source (cr
, pattern
= prolooks_create_gradient_str ((double) 16, (double) (-2), (double) 9, (double) 13, "#e7ecef", "#9cafb8", 0.8, 0.8));
506 cairo_pattern_destroy (pattern
);
507 cairo_move_to (cr
, (double) 0, (double) 4);
508 cairo_line_to (cr
, (double) 17, (double) 4);
509 cairo_curve_to (cr
, (double) 19, (double) 4, (double) 21, (double) 2, (double) 21, (double) 0);
510 cairo_curve_to (cr
, (double) 21, (double) (-2), (double) 19, (double) (-4), (double) 17, (double) (-4));
511 cairo_line_to (cr
, (double) 0, (double) (-4));
512 cairo_close_path (cr
);
515 pattern
= prolooks_create_gradient_str ((double) 9, (double) (-2), (double) 9, (double) 2, "#68625c", "#44494b", 1.0, 1.0);
516 cairo_set_source (cr
, pattern
);
517 cairo_pattern_destroy (pattern
);
518 cairo_move_to (cr
, (double) 0, (double) 2);
519 cairo_line_to (cr
, (double) 16, (double) 2);
520 cairo_curve_to (cr
, (double) 17, (double) 2, (double) 18, (double) 1, (double) 18, (double) 0);
521 cairo_curve_to (cr
, (double) 18, (double) (-1), (double) 17, (double) (-2), (double) 16, (double) (-2));
522 cairo_line_to (cr
, (double) 0, (double) (-2));
523 cairo_close_path (cr
);
527 cairo_set_line_width (cr
, 2.0);
528 pattern
= prolooks_create_gradient_str ((double) 38, (double) 32, (double) 70, (double) 67, "#3d3d3d", "#000000", 1.0, 1.0);
529 cairo_set_source (cr
, pattern
);
530 cairo_pattern_destroy (pattern
);
531 cairo_arc (cr
, xc
, yc
, knob_disc_radius
, (double) 0, 2 * G_PI
);
535 cairo_pattern_destroy (knob_ripples
);
539 MotionFeedback::pixwin_expose_event (GdkEventExpose
* ev
)
541 GdkWindow
*window
= pixwin
.get_window()->gobj();
542 GtkAdjustment
* adj
= adjustment
->gobj();
544 int phase
= (int)((adj
->value
- adj
->lower
) * 64 /
545 (adj
->upper
- adj
->lower
));
547 // skip middle phase except for true middle value
549 if (type
== Rotary
&& phase
== 32) {
550 double pt
= (adj
->value
- adj
->lower
) * 2.0 /
551 (adj
->upper
- adj
->lower
) - 1.0;
558 // endless knob: skip 90deg highlights unless the value is really a multiple of 90deg
560 if (type
== Endless
&& !(phase
% 16)) {
565 double nom
= adj
->lower
+ phase
* (adj
->upper
- adj
->lower
) / 64.0;
566 double diff
= (adj
->value
- nom
) / (adj
->upper
- adj
->lower
);
569 phase
= (phase
+ 1) % 64;
571 phase
= (phase
+ 63) % 64;
575 std::cerr
<< "Render from pixbuf\n";
576 GtkWidget
* widget
= GTK_WIDGET(pixwin
.gobj());
577 gdk_draw_pixbuf (GDK_DRAWABLE(window
), widget
->style
->fg_gc
[0],
579 phase
* subwidth
, type
* subheight
,
580 0, 0, subwidth
, subheight
, GDK_RGB_DITHER_NORMAL
, 0, 0);
582 std::cerr
<< "Render with cairo\n";
583 cairo_t
* cr
= gdk_cairo_create (GDK_DRAWABLE (window
));
585 gdk_cairo_rectangle (cr
, &ev
->area
);
588 core_draw (cr
, phase
, subheight
/2, subwidth
/2, subheight
/2);
596 MotionFeedback::pixwin_scroll_event (GdkEventScroll
* ev
)
600 if ((ev
->state
& (Keyboard::PrimaryModifier
|Keyboard::TertiaryModifier
)) == (Keyboard::PrimaryModifier
|Keyboard::TertiaryModifier
)) {
602 } else if (ev
->state
& Keyboard::PrimaryModifier
) {
608 switch (ev
->direction
) {
610 case GDK_SCROLL_RIGHT
:
611 adjustment
->set_value (adjustment
->get_value() + (scale
* adjustment
->get_step_increment()));
614 case GDK_SCROLL_DOWN
:
615 case GDK_SCROLL_LEFT
:
616 adjustment
->set_value (adjustment
->get_value() - (scale
* adjustment
->get_step_increment()));
624 MotionFeedback::pixwin_size_request (GtkRequisition
* req
)
626 req
->width
= subwidth
;
627 req
->height
= subheight
;
631 MotionFeedback::pixwin_realized ()
633 set_lamp_color (Gdk::Color ("#b9feff"));
637 MotionFeedback::set_lamp_color (const Gdk::Color
& c
)
639 GdkColor col2
= {0,0,0,0};
640 GdkColor col3
= {0,0,0,0};
643 lamp_hsv
= prolooks_hsv_new_for_gdk_color (_lamp_color
.gobj());
644 lamp_bright
= (prolooks_hsv_to_gdk_color (lamp_hsv
, &col2
), col2
);
645 prolooks_hsv_set_saturation (lamp_hsv
, 0.66);
646 prolooks_hsv_set_value (lamp_hsv
, 0.67);
647 lamp_dark
= (prolooks_hsv_to_gdk_color (lamp_hsv
, &col3
), col3
);
651 MotionFeedback::render_file (const std::string
& path
, int w
, int h
)
653 GdkPixmap
* pixmap
= gdk_pixmap_new (0, w
, h
, 24);
654 GdkPixbuf
* pixbuf
= gdk_pixbuf_new (GDK_COLORSPACE_RGB
, 1, 8, w
* 65, h
);
663 set_lamp_color (Gdk::Color ("#b9feff"));
665 for (int i
= 0; i
< 65; ++i
) {
666 cairo_t
* cr
= gdk_cairo_create (GDK_DRAWABLE (pixmap
));
667 gdk_cairo_rectangle (cr
, &r
);
669 core_draw (cr
, i
, h
/2, w
/2, h
/2);
670 gdk_pixbuf_get_from_drawable (pixbuf
, pixmap
, gdk_colormap_get_system(), 0, 0, w
*i
, 0, w
, h
);
674 if (gdk_pixbuf_save (pixbuf
, path
.c_str(), "png", &err
, 0)) {
676 std::cerr
<< "could not save image set to " << path
<< ": " << err
->message
<< std::endl
;
680 g_object_unref (G_OBJECT (pixbuf
));
681 g_object_unref (G_OBJECT (pixmap
));