Add an option to disable editor update during drags of the
[ardour.git] / libs / gtkmm2ext / motionfeedback.cc
1 /*
2     Copyright (C) 2010-2011 Paul Davis
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., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18     $Id: motionfeedback.cc,v 1.5 2004/03/01 03:44:19 pauld Exp $
19 */
20
21 #include <iostream>
22 #include <cmath>
23 #include <cstdlib>
24 #include <algorithm>
25 #include <unistd.h>
26 #include <stdio.h> /* for snprintf, grrr */
27
28 #include <gdk/gdkkeysyms.h>
29 #include <gtkmm.h>
30
31 #include "pbd/controllable.h"
32
33 #include "gtkmm2ext/motionfeedback.h"
34 #include "gtkmm2ext/keyboard.h"
35 #include "gtkmm2ext/prolooks-helpers.h"
36 #include "gtkmm2ext/gui_thread.h"
37
38 using namespace Gtk;
39 using namespace Gtkmm2ext;
40 using namespace sigc;
41
42 Gdk::Color* MotionFeedback::base_color;
43
44 MotionFeedback::MotionFeedback (Glib::RefPtr<Gdk::Pixbuf> pix,
45                                 Type t,
46                                 boost::shared_ptr<PBD::Controllable> c,
47                                 double default_val,
48                                 double step_increment,
49                                 double page_increment,
50                                 const char *widget_name, 
51                                 bool with_numeric_display, 
52                                 int subw, 
53                                 int subh) 
54         : _controllable (c)
55         , value (0)
56         , default_value (default_val)
57         , step_inc (step_increment)
58         , page_inc (page_increment)
59         , type (t)
60         , value_packer (0)
61         , pixbuf (pix)
62         , subwidth (subw)
63         , subheight (subh)
64 {
65         if (!base_color) {
66                 base_color = new Gdk::Color ("#1a5274");
67         }
68
69         char value_name[1024];
70
71         print_func = default_printer;
72         print_arg = 0;
73
74
75         HBox* hpacker = manage (new HBox);
76         hpacker->pack_start (pixwin, true, true);
77         hpacker->show ();
78         pack_start (*hpacker, false, false);
79         pixwin.show ();
80
81         if (with_numeric_display) {
82
83                 value_packer = new EventBox;
84                 value_packer->set_name ("MotionControllerValue");
85                 value_packer->show ();
86                 value_packer->set_border_width (6);
87
88                 value = new Label;
89                 value->set_justify (Gtk::JUSTIFY_RIGHT);
90                 value->show ();
91                 
92                 value_packer->add (*value);
93
94                 hpacker = manage (new HBox);
95                 hpacker->pack_start (*value_packer, true, false);
96                 hpacker->show ();
97
98                 pack_start (*hpacker, false, false);
99
100                 if (widget_name) {
101                         snprintf (value_name, sizeof(value_name), "%sValue", widget_name);
102                         value->set_name (value_name);
103                 }
104
105                 if (_controllable) {
106                         char buf[32];
107                         print_func (buf, _controllable, print_arg);
108                         value->set_text (buf);
109                 }
110         }
111
112         pixwin.set_events (Gdk::BUTTON_PRESS_MASK|
113                            Gdk::BUTTON_RELEASE_MASK|
114                            Gdk::POINTER_MOTION_MASK|
115                            Gdk::ENTER_NOTIFY_MASK|
116                            Gdk::LEAVE_NOTIFY_MASK|
117                            Gdk::SCROLL_MASK|
118                            Gdk::KEY_PRESS_MASK|
119                            Gdk::KEY_RELEASE_MASK);
120
121         pixwin.set_flags (CAN_FOCUS);
122
123         /* Proxy all important events on the pixwin to ourselves */
124
125         pixwin.signal_button_press_event().connect(mem_fun (*this,&MotionFeedback::pixwin_button_press_event));
126         pixwin.signal_button_release_event().connect(mem_fun (*this,&MotionFeedback::pixwin_button_release_event));
127         pixwin.signal_motion_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_motion_notify_event));
128         pixwin.signal_enter_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_enter_notify_event));
129         pixwin.signal_leave_notify_event().connect(mem_fun (*this,&MotionFeedback::pixwin_leave_notify_event));
130         pixwin.signal_key_press_event().connect(mem_fun (*this,&MotionFeedback::pixwin_key_press_event));
131         pixwin.signal_scroll_event().connect(mem_fun (*this,&MotionFeedback::pixwin_scroll_event));
132         pixwin.signal_expose_event().connect(mem_fun (*this,&MotionFeedback::pixwin_expose_event), true);
133         pixwin.signal_size_request().connect(mem_fun (*this,&MotionFeedback::pixwin_size_request));
134 }
135
136 MotionFeedback::~MotionFeedback()
137 {
138         delete value;
139         delete value_packer;
140 }
141
142 bool
143 MotionFeedback::pixwin_button_press_event (GdkEventButton *ev) 
144
145         if (binding_proxy.button_press_handler (ev)) {
146                 return true;
147         }
148
149         switch (ev->button) {
150         case 1:
151                 grab_is_fine = false;
152                 break;
153         case 2:
154                 grab_is_fine = true;
155                 break;
156         case 3:
157                 return false;
158         }
159
160         gtk_grab_add(GTK_WIDGET(pixwin.gobj()));
161
162         grabbed_y = ev->y_root;
163         grabbed_x = ev->x_root;
164
165         return false;
166 }
167
168 bool
169 MotionFeedback::pixwin_button_release_event (GdkEventButton *ev) 
170
171         if (!_controllable) {
172                 return false;
173         }
174
175         switch (ev->button) {
176         case 1:
177                 if (pixwin.has_grab()) {
178                         if (!grab_is_fine) {
179                                 gtk_grab_remove
180                                         (GTK_WIDGET(pixwin.gobj()));
181                         }
182                 }
183                 if (Keyboard::modifier_state_equals (ev->state, Keyboard::TertiaryModifier)) {
184                         /* shift click back to the default */
185                         _controllable->set_value (default_value);
186                         return true;
187                 }
188                 break;
189                 
190         case 3:
191                 if (pixwin.has_grab()) {
192                         if (grab_is_fine) {
193                                 gtk_grab_remove
194                                         (GTK_WIDGET(pixwin.gobj()));
195                         }
196                 }
197                 break;
198         }
199
200         return VBox::on_button_release_event (ev); 
201 }
202
203 bool
204 MotionFeedback::pixwin_motion_notify_event (GdkEventMotion *ev) 
205
206         if (!_controllable) {
207                 return false;
208         }
209
210         gfloat multiplier;
211         gfloat x_delta;
212         gfloat y_delta;
213
214         if (!pixwin.has_grab()) {
215                 return VBox::on_motion_notify_event (ev); 
216         }
217
218         multiplier = ((ev->state & Keyboard::TertiaryModifier) ? 100 : 1) *
219                 ((ev->state & Keyboard::PrimaryModifier) ? 10 : 1) *
220                 ((ev->state & Keyboard::SecondaryModifier) ? 0.1 : 1);
221
222         if (ev->state & Gdk::BUTTON1_MASK) {
223
224                 /* vertical control */
225
226                 y_delta = grabbed_y - ev->y_root;
227                 grabbed_y = ev->y_root;
228                 
229                 x_delta = ev->x_root - grabbed_x;
230                 
231                 if (y_delta == 0) return TRUE;
232                 
233                 y_delta *= 1 + (x_delta/100);
234                 y_delta *= multiplier;
235                 y_delta /= 10;
236                 
237                 _controllable->set_value (adjust ((grab_is_fine ? step_inc : page_inc) * y_delta));
238                 
239         } else if (ev->state & Gdk::BUTTON2_MASK) {
240
241                 /* rotary control */
242
243                 double x = ev->x - subwidth/2;
244                 double y = - ev->y + subwidth/2;
245                 double angle = std::atan2 (y, x) / M_PI;
246                 
247                 if (angle < -0.5) {
248                         angle += 2.0;
249                 }
250                 
251                 angle = -(2.0/3.0) * (angle - 1.25);
252                 angle *= multiplier;
253
254                 _controllable->set_value (to_control_value (angle));
255         }
256
257
258         return true;
259 }
260
261 bool
262 MotionFeedback::pixwin_enter_notify_event (GdkEventCrossing*) 
263 {
264         pixwin.grab_focus();
265         return false;
266 }
267
268 bool
269 MotionFeedback::pixwin_leave_notify_event (GdkEventCrossing*) 
270 {
271         pixwin.unset_flags (HAS_FOCUS);
272         return false;
273 }
274
275 bool
276 MotionFeedback::pixwin_key_press_event (GdkEventKey *ev) 
277 {
278         if (!_controllable) {
279                 return false;
280         }
281
282         bool retval = false;
283         double multiplier;
284
285         multiplier = ((ev->state & Keyboard::TertiaryModifier) ? 100.0 : 1.0) *
286                 ((ev->state & Keyboard::SecondaryModifier) ? 10.0 : 1.0) * 
287                 ((ev->state & Keyboard::PrimaryModifier) ? 2.0 : 1.0);
288
289         switch (ev->keyval) {
290         case GDK_Page_Up:
291                 retval = true;
292                 _controllable->set_value (adjust (multiplier * page_inc));
293                 break;
294
295         case GDK_Page_Down:
296                 retval = true;
297                 _controllable->set_value (adjust (multiplier * page_inc));
298                 break;
299
300         case GDK_Up:
301                 retval = true;
302                 _controllable->set_value (adjust (multiplier * step_inc));
303                 break;
304
305         case GDK_Down:
306                 retval = true;
307                 _controllable->set_value (adjust (multiplier * step_inc));
308                 break;
309
310         case GDK_Home:
311                 retval = true;
312                 _controllable->set_value (_controllable->lower());
313                 break;
314
315         case GDK_End:
316                 retval = true;
317                 _controllable->set_value (_controllable->upper());
318                 break;
319         }
320         
321         return retval;
322 }
323
324 bool
325 MotionFeedback::pixwin_expose_event (GdkEventExpose*)
326 {
327         if (!_controllable) {
328                 return true;
329         }
330
331         GdkWindow *window = pixwin.get_window()->gobj();
332         double display_val = to_display_value (_controllable->get_value());
333         int32_t phase = lrint (display_val * 64.0);
334         
335         // skip middle phase except for true middle value
336
337         if (type == Rotary && phase == 32) {
338                 double pt = (display_val * 2.0) - 1.0;
339                 if (pt < 0)
340                         phase = 31;
341                 if (pt > 0)
342                         phase = 33;
343         }
344
345         // endless knob: skip 90deg highlights unless the value is really a multiple of 90deg
346
347         if (type == Endless && !(phase % 16)) {
348                 if (phase == 64) {
349                         phase = 0;
350                 }
351
352                 double nom = phase / 64.0;
353                 double diff = display_val - nom;
354
355                 if (diff > 0.0001)
356                         phase = (phase + 1) % 64;
357                 if (diff < -0.0001)
358                         phase = (phase + 63) % 64;
359         }
360
361         phase = std::min (phase, (int32_t) 63);
362
363         GtkWidget* widget = GTK_WIDGET(pixwin.gobj());
364         gdk_draw_pixbuf (GDK_DRAWABLE(window), widget->style->fg_gc[0], 
365                          pixbuf->gobj(), 
366                          phase * subwidth, type * subheight, 
367                          /* center image in allocated area */
368                          (get_width() - subwidth)/2, 
369                          0,
370                          subwidth, subheight, GDK_RGB_DITHER_NORMAL, 0, 0);
371
372         return true;
373 }
374
375 bool
376 MotionFeedback::pixwin_scroll_event (GdkEventScroll* ev)
377 {
378         double scale;
379
380         if (!_controllable) {
381                 return false;
382         }
383
384         if ((ev->state & (Keyboard::PrimaryModifier|Keyboard::TertiaryModifier)) == (Keyboard::PrimaryModifier|Keyboard::TertiaryModifier)) {
385                 scale = 0.01;
386         } else if (ev->state & Keyboard::PrimaryModifier) {
387                 scale = 0.1;
388         } else {
389                 scale = 1.0;
390         }
391
392         switch (ev->direction) {
393         case GDK_SCROLL_UP:
394         case GDK_SCROLL_RIGHT:
395                 _controllable->set_value (adjust (scale * step_inc));
396                 break;
397
398         case GDK_SCROLL_DOWN:
399         case GDK_SCROLL_LEFT:
400                 _controllable->set_value (adjust (-scale * step_inc));
401                 break;
402         }
403
404         return true;
405 }
406
407 void
408 MotionFeedback::pixwin_size_request (GtkRequisition* req)
409 {
410         req->width = subwidth;
411         req->height = subheight;
412 }
413
414
415 void
416 MotionFeedback::controllable_value_changed ()
417 {
418         if (value) {
419                 char buf[32];
420                 print_func (buf, _controllable, print_arg);
421                 value->set_text (buf);
422         }
423
424         pixwin.queue_draw ();
425 }
426
427 void
428 MotionFeedback::set_controllable (boost::shared_ptr<PBD::Controllable> c)
429 {
430         _controllable = c;
431         binding_proxy.set_controllable (c);
432         controller_connection.disconnect ();
433
434         if (c) {
435                 c->Changed.connect (controller_connection, MISSING_INVALIDATOR, boost::bind (&MotionFeedback::controllable_value_changed, this), gui_context());
436
437                 char buf[32];
438                 print_func (buf, _controllable, print_arg);
439                 value->set_text (buf);
440         }
441
442         pixwin.queue_draw ();
443 }
444
445 boost::shared_ptr<PBD::Controllable>
446 MotionFeedback::controllable () const 
447 {
448         return _controllable;
449 }
450        
451 void
452 MotionFeedback::default_printer (char buf[32], const boost::shared_ptr<PBD::Controllable>& c, void *)
453 {
454         if (c) {
455                 sprintf (buf, "%.2f", c->get_value());
456         } else {
457                 buf[0] = '\0';
458         }
459 }
460
461 Glib::RefPtr<Gdk::Pixbuf>
462 MotionFeedback::render_pixbuf (int size)
463 {
464         Glib::RefPtr<Gdk::Pixbuf> pixbuf;
465         char path[32];
466         int fd;
467
468         snprintf (path, sizeof (path), "/tmp/mfimg%dXXXXXX", size);
469         
470         if ((fd = mkstemp (path)) < 0) {
471                 return pixbuf;
472         }
473         
474         GdkColor col2 = {0,0,0,0};
475         GdkColor col3 = {0,0,0,0};
476         GdkColor dark;
477         GdkColor bright;
478         ProlooksHSV* hsv;
479
480         hsv = prolooks_hsv_new_for_gdk_color (base_color->gobj());
481         bright = (prolooks_hsv_to_gdk_color (hsv, &col2), col2);
482         prolooks_hsv_set_saturation (hsv, 0.66);
483         prolooks_hsv_set_value (hsv, 0.67);
484         dark = (prolooks_hsv_to_gdk_color (hsv, &col3), col3);
485
486         cairo_surface_t *surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, size * 64, size);
487         cairo_t* cr = cairo_create (surface);
488
489         for (int i = 0; i < 64; ++i) {
490                 cairo_save (cr);
491                 core_draw (cr, i, size, 20, size*i, 0, &bright, &dark);
492                 cairo_restore (cr);
493         }
494
495         if (cairo_surface_write_to_png (surface, path) != CAIRO_STATUS_SUCCESS) {
496                 std::cerr << "could not save image set to " << path << std::endl;
497                 return pixbuf;
498         }
499
500         close (fd);
501
502         cairo_destroy (cr);
503         cairo_surface_destroy (surface);
504
505         try {
506                 pixbuf = Gdk::Pixbuf::create_from_file (path);
507         } catch (const Gdk::PixbufError &e) {
508                 std::cerr << "Caught PixbufError: " << e.what() << std::endl;
509                 unlink (path);
510                 throw;
511         } catch (...) {
512                 unlink (path);
513                 g_message("Caught ... ");
514                 throw;
515         }
516
517         unlink (path);
518
519         return pixbuf;
520
521
522 void
523 MotionFeedback::core_draw (cairo_t* cr, int phase, double size, double progress_width, double xorigin, double yorigin,
524                            const GdkColor* bright, const GdkColor* dark)
525 {
526         double xc;
527         double yc;
528         double start_angle;
529         double end_angle;
530         double value_angle;
531         double value;
532         double value_x;
533         double value_y;
534         double start_angle_x;
535         double start_angle_y;
536         double end_angle_x;
537         double end_angle_y;
538         double progress_radius;
539         double progress_radius_inner;
540         double progress_radius_outer;
541         double knob_disc_radius;
542         cairo_pattern_t* pattern;
543         double progress_rim_width;
544         cairo_pattern_t* progress_shine;
545         double degrees;
546         cairo_pattern_t* knob_ripples;
547         double pxs;
548         double pys;
549
550         g_return_if_fail (cr != NULL);
551         
552         progress_radius = 40.0;
553         progress_radius_inner = progress_radius - (progress_width / 2.0);
554         progress_radius_outer = progress_radius + (progress_width / 2.0);
555         knob_disc_radius = progress_radius_inner - 5.0;
556
557         const double pad = 2.0; /* line width for boundary of progress ring */
558         const double actual_width = ((2.0 * pad) + (2.0 * progress_radius_outer));
559         const double scale_factor = size / actual_width;
560
561         /* knob center is at middle of the area bounded by (xorigin,yorigin) and (xorigin+size, yorigin+size)
562            but the coordinates will be scaled by the scale factor when cairo uses them so first
563            adjust them by the reciprocal of the scale factor.
564         */
565
566         xc = (xorigin + (size / 2.0)) * (1.0/scale_factor);
567         yc = (yorigin + (size / 2.0)) * (1.0/scale_factor);
568
569         pxs = xorigin * (1.0/scale_factor);
570         pys = yorigin * (1.0/scale_factor);
571
572         start_angle = 0.0;
573         end_angle = 0.0;
574         value_angle = 0.0;
575         value = (phase * 1.0) / (65 - 1);
576
577         start_angle = ((180 - 65) * G_PI) / 180;
578         end_angle = ((360 + 65) * G_PI) / 180;
579
580         value_angle = start_angle + (value * (end_angle - start_angle));
581         value_x = cos (value_angle);
582         value_y = sin (value_angle);
583         start_angle_x = cos (start_angle);
584         start_angle_y = sin (start_angle);
585         end_angle_x = cos (end_angle);
586         end_angle_y = sin (end_angle);
587
588         cairo_scale (cr, scale_factor, scale_factor);
589
590         pattern = prolooks_create_gradient_str (pxs + 32.0, pys + 16.0, pxs + 75.0, pys + 16.0, "#d4c8b9", "#ae977b", 1.0, 1.0);
591         cairo_set_source (cr, pattern);
592         cairo_pattern_destroy (pattern);
593         cairo_set_line_width (cr, 2.0);
594         cairo_arc (cr, xc, yc, 31.5, 0.0, 2 * G_PI);
595         cairo_stroke (cr);
596
597         pattern = prolooks_create_gradient_str (pxs + 20.0, pys + 20.0, pxs + 89.0, pys + 87.0, "#2f2f4c", "#090a0d", 1.0, 1.0);
598         cairo_set_source (cr, pattern);
599         cairo_pattern_destroy (pattern);
600         cairo_set_line_width (cr, progress_width);
601         cairo_arc (cr, xc, yc, progress_radius, start_angle, end_angle);
602         cairo_stroke (cr);
603
604         pattern = prolooks_create_gradient (pxs + 20.0, pys + 20.0, pxs + 89.0, pys + 87.0, bright, dark, 1.0, 1.0);
605         cairo_set_source (cr, pattern);
606         cairo_pattern_destroy (pattern);
607         cairo_set_line_width (cr, progress_width);
608         cairo_arc (cr, xc, yc, progress_radius, start_angle, value_angle);
609         cairo_stroke (cr);
610
611         cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
612         progress_rim_width = 2.0;
613         cairo_set_line_width (cr, progress_rim_width);
614         pattern = prolooks_create_gradient_str (pxs + 18.0, pys + 79.0, pxs + 35.0, pys + 79.0, "#dfd5c9", "#dfd5c9", 1.0, 0.0);
615         cairo_set_source (cr, pattern);
616         cairo_pattern_destroy (pattern);
617         cairo_move_to (cr, xc + (progress_radius_outer * start_angle_x), yc + (progress_radius_outer * start_angle_y));
618         cairo_line_to (cr, xc + (progress_radius_inner * start_angle_x), yc + (progress_radius_inner * start_angle_y));
619         cairo_stroke (cr);
620
621         prolooks_set_source_color_string (cr, "#000000", 1.0);
622         cairo_move_to (cr, xc + (progress_radius_outer * end_angle_x), yc + (progress_radius_outer * end_angle_y));
623         cairo_line_to (cr, xc + (progress_radius_inner * end_angle_x), yc + (progress_radius_inner * end_angle_y));
624         cairo_stroke (cr);
625
626         // pattern = prolooks_create_gradient_str (95.0, 6.0, 5.0, 44.0, "#dfd5c9", "#b0a090", 1.0, 1.0);
627         pattern = prolooks_create_gradient_str (pxs + 95.0, pys + 6.0, pxs + 5.0, pys + 44.0, "#000000", "#000000", 1.0, 1.0);
628         cairo_set_source (cr, pattern);
629         cairo_pattern_destroy (pattern);
630         cairo_arc (cr, xc, yc, progress_radius_outer, start_angle, end_angle);
631         cairo_stroke (cr);
632
633         cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
634         pattern = prolooks_create_gradient (pxs + 20.0, pys + 20.0, pxs + 89.0, pys + 87.0, bright, dark, 0.25, 0.25);
635         cairo_set_source (cr, pattern);
636         cairo_pattern_destroy (pattern);
637         cairo_set_line_width (cr, progress_width);
638         cairo_arc (cr, xc, yc, progress_radius, start_angle, value_angle + (G_PI / 180.0));
639         cairo_stroke (cr);
640
641         progress_shine = prolooks_create_gradient_str (pxs + 89.0, pys + 73.0, pxs + 34.0, pys + 16.0, "#ffffff", "#ffffff", 0.3, 0.04);
642         cairo_pattern_add_color_stop_rgba (progress_shine, 0.5, 1.0, 1.0, 1.0, 0.0);
643         if (size > 50) {
644                 cairo_pattern_add_color_stop_rgba (progress_shine, 0.75, 1.0, 1.0, 1.0, 0.3);
645         } else {
646                 cairo_pattern_add_color_stop_rgba (progress_shine, 0.75, 1.0, 1.0, 1.0, 0.2);
647         }
648         cairo_set_source (cr, progress_shine);
649         cairo_set_line_width (cr, progress_width);
650         cairo_arc (cr, xc, yc, progress_radius, start_angle, end_angle);
651         cairo_stroke (cr);
652         cairo_pattern_destroy (progress_shine);
653
654         cairo_set_line_width (cr, 1.0);
655         cairo_set_source_rgb (cr, 0.0, 0.0, 0.0);
656         cairo_arc (cr, xc, yc, progress_radius_inner, 0.0, 2 * G_PI);
657         pattern = prolooks_create_gradient_str (pxs + 35.0, pys + 31.0, pxs + 75.0, pys + 72.0, "#68625c", "#44494b", 1.0, 1.0);
658         cairo_set_source (cr, pattern);
659         cairo_pattern_destroy (pattern);
660         cairo_fill (cr);
661         cairo_set_source_rgb (cr, 0.0, 0.0, 0.0);
662         cairo_arc (cr, xc, yc, progress_radius_inner, 0.0, 2 * G_PI);
663         cairo_stroke (cr);
664
665         pattern = prolooks_create_gradient_str (pxs + 42.0, pys + 34.0, pxs + 68.0, pys + 70.0, "#e7ecef", "#9cafb8", 1.0, 1.0);
666         cairo_set_source (cr, pattern);
667         cairo_pattern_destroy (pattern);
668         cairo_arc (cr, xc, yc, knob_disc_radius, 0.0, 2 * G_PI);
669         cairo_fill (cr);
670
671         cairo_set_line_width (cr, 2.0);
672         degrees = G_PI / 180.0;
673         pattern = prolooks_create_gradient_str (pxs + 38.0, pys + 34.0, pxs + 70.0, pys + 68.0, "#ffffff", "#caddf2", 0.2, 0.2);
674         cairo_set_source (cr, pattern);
675         cairo_pattern_destroy (pattern);
676         cairo_move_to (cr, xc, yc);
677         cairo_arc (cr, xc, yc, knob_disc_radius - 1, (-154) * degrees, (-120) * degrees);
678         cairo_move_to (cr, xc, yc);
679         cairo_arc (cr, xc, yc, knob_disc_radius - 1, (G_PI / 2.0) - (60 * degrees), (G_PI / 2.0) - (29 * degrees));
680         cairo_fill (cr);
681
682         pattern = prolooks_create_gradient_str (pxs + 50.0, pys + 40.0, pxs + 62.0, pys + 60.0, "#a1adb6", "#47535c", 0.07, 0.15);
683         cairo_set_source (cr, pattern);
684         cairo_pattern_destroy (pattern);
685         cairo_move_to (cr, xc, yc);
686         cairo_arc (cr, xc, yc, knob_disc_radius - 1, (-67) * degrees, (-27) * degrees);
687         cairo_move_to (cr, xc, yc);
688         cairo_arc (cr, xc, yc, knob_disc_radius - 1, G_PI - (67 * degrees), G_PI - (27 * degrees));
689         cairo_fill (cr);
690
691         knob_ripples = cairo_pattern_create_radial (xc, yc, 0.0, xc, yc, 4.0);
692         prolooks_add_color_stop_str (knob_ripples, 0.0, "#e7ecef", 0.05);
693         prolooks_add_color_stop_str (knob_ripples, 0.5, "#58717d", 0.05);
694         prolooks_add_color_stop_str (knob_ripples, 0.75, "#d1d9de", 0.05);
695         prolooks_add_color_stop_str (knob_ripples, 1.0, "#5d7682", 0.05);
696         cairo_pattern_set_extend (knob_ripples, CAIRO_EXTEND_REPEAT);
697         cairo_set_line_width (cr, 0.0);
698         cairo_set_source (cr, knob_ripples);
699         cairo_arc (cr, xc, yc, knob_disc_radius, 0.0, 2 * G_PI);
700         cairo_fill (cr);
701
702         cairo_save (cr);
703         cairo_translate (cr, xc + (knob_disc_radius * value_x), yc + (knob_disc_radius * value_y));
704         cairo_rotate (cr, value_angle - G_PI);
705         pattern = prolooks_create_gradient_str (pxs + 16.0, pys + -2.0, pxs + 9.0, pys + 13.0, "#e7ecef", "#9cafb8", 0.8, 0.8);
706         cairo_set_source (cr, pattern);
707         cairo_pattern_destroy (pattern);
708         cairo_move_to (cr, 0.0, 4.0);
709         cairo_line_to (cr, 17.0, 4.0);
710         cairo_curve_to (cr, 19.0, 4.0, 21.0, 2.0, 21.0, 0.0);
711         cairo_curve_to (cr, 21.0, -2.0, 19.0, -4.0, 17.0, -4.0);
712         cairo_line_to (cr, 0.0, -4.0);
713         cairo_close_path (cr);
714         cairo_fill (cr);
715
716         pattern = prolooks_create_gradient_str (pxs + 9.0, pys + -2.0, pxs + 9.0, pys + 2.0, "#68625c", "#44494b", 1.0, 1.0);
717         cairo_set_source (cr, pattern);
718         cairo_pattern_destroy (pattern);
719         cairo_move_to (cr, 0.0, 2.0);
720         cairo_line_to (cr, 16.0, 2.0);
721         cairo_curve_to (cr, 17.0, 2.0, 18.0, 1.0, 18.0, 0.0);
722         cairo_curve_to (cr, 18.0, -1.0, 17.0, -2.0, 16.0, -2.0);
723         cairo_line_to (cr, 0.0, -2.0);
724         cairo_close_path (cr);
725         cairo_fill (cr);
726
727         cairo_restore (cr);
728         cairo_set_line_width (cr, 2.0);
729         pattern = prolooks_create_gradient_str (pxs + 38.0, pys + 32.0, pxs + 70.0, pys + 67.0, "#3d3d3d", "#000000", 1.0, 1.0); 
730         cairo_set_source (cr, pattern);
731         cairo_pattern_destroy (pattern);
732         cairo_arc (cr, xc, yc, knob_disc_radius, 0.0, 2 * G_PI);
733         cairo_stroke (cr);
734
735         cairo_pattern_destroy (knob_ripples);
736 }
737
738 void
739 MotionFeedback::set_lamp_color (const std::string& str)
740 {
741         if (base_color) {
742                 *base_color = Gdk::Color (str);
743         } else {
744                 base_color = new Gdk::Color (str);
745         }
746 }