add icons for cut mouse mode button and cut cursor (both need improving)
[ardour.git] / gtk2_ardour / ardour_button.cc
index 7df6b50490dd3e03bef19198ed6e998860da2836..2525d57c07cf9d8613afd468eb183f5d23f2c437 100644 (file)
@@ -25,6 +25,7 @@
 
 #include "pbd/compose.h"
 #include "pbd/error.h"
+#include "pbd/stacktrace.h"
 
 #include "gtkmm2ext/utils.h"
 #include "gtkmm2ext/rgb_macros.h"
@@ -64,16 +65,16 @@ ArdourButton::ArdourButton (Element e)
        , _angle(0)
        , _xalign(.5)
        , _yalign(.5)
+       , bg_color (0)
        , border_color (0)
-       , fill_color_active (0)
-       , fill_color_active_start(0)
-       , fill_color_active_end(0)
-       , fill_color_inactive_start(0)
-       , fill_color_inactive_end(0)
-       , text_color_inactive(0)
-       , text_color_active(0)
-       , led_color_inactive(0)
-       , led_color_active(0)
+       , fill_start_inactive_color (0)
+       , fill_end_inactive_color (0)
+       , fill_start_active_color (0)
+       , fill_end_active_color (0)
+       , text_active_color(0)
+       , text_inactive_color(0)
+       , led_active_color(0)
+       , led_inactive_color(0)
        , fill_pattern (0)
        , fill_pattern_active (0)
        , shine_pattern (0)
@@ -85,8 +86,10 @@ ArdourButton::ArdourButton (Element e)
        , _fixed_diameter (true)
        , _distinct_led_click (false)
        , _hovering (false)
+       , _focused (false)
+       , _fixed_colors_set (false)
 {
-       ColorsChanged.connect (sigc::mem_fun (*this, &ArdourButton::color_handler));
+       ARDOUR_UI_UTILS::ColorsChanged.connect (sigc::mem_fun (*this, &ArdourButton::color_handler));
 }
 
 ArdourButton::ArdourButton (const std::string& str, Element e)
@@ -100,16 +103,16 @@ ArdourButton::ArdourButton (const std::string& str, Element e)
        , _angle(0)
        , _xalign(.5)
        , _yalign(.5)
+       , bg_color (0)
        , border_color (0)
-       , fill_color_active (0)
-       , fill_color_active_start(0)
-       , fill_color_active_end(0)
-       , fill_color_inactive_start(0)
-       , fill_color_inactive_end(0)
-       , text_color_inactive(0)
-       , text_color_active(0)
-       , led_color_inactive(0)
-       , led_color_active(0)
+       , fill_start_inactive_color (0)
+       , fill_end_inactive_color (0)
+       , fill_start_active_color (0)
+       , fill_end_active_color (0)
+       , text_active_color(0)
+       , text_inactive_color(0)
+       , led_active_color(0)
+       , led_inactive_color(0)
        , fill_pattern (0)
        , fill_pattern_active (0)
        , shine_pattern (0)
@@ -121,6 +124,8 @@ ArdourButton::ArdourButton (const std::string& str, Element e)
        , _fixed_diameter (true)
        , _distinct_led_click (false)
        , _hovering (false)
+       , _focused (false)
+       , _fixed_colors_set (false)
 {
        set_text (str);
 }
@@ -194,7 +199,7 @@ ArdourButton::set_alignment (const float xa, const float ya)
 }
 
 void
-ArdourButton::render (cairo_t* cr)
+ArdourButton::render (cairo_t* cr, cairo_rectangle_t *)
 {
        void (*rounded_function)(cairo_t*, double, double, double, double, double);
 
@@ -244,7 +249,7 @@ ArdourButton::render (cairo_t* cr)
                        
                        if (!(_tweaks & ImplicitUsesSolidColor)) {
                                //border
-                               UINT_TO_RGBA (fill_color_active, &r, &g, &b, &a);
+                               UINT_TO_RGBA (fill_end_active_color, &r, &g, &b, &a);
                                cairo_set_line_width (cr, 1.0);
                                rounded_function (cr, 2, 2, get_width()-4, get_height()-4, _corner_radius - 1.5);
                                cairo_set_source_rgba (cr, r/255.0, g/255.0, b/255.0, a/255.0);
@@ -266,7 +271,7 @@ ArdourButton::render (cairo_t* cr)
                }
        }
 
-       if ( ((_elements & FlatFace)==FlatFace) && (active_state() != Gtkmm2ext::ExplicitActive) ) {
+       if ( ((_elements & Inset)==Inset) && (active_state() != Gtkmm2ext::ExplicitActive) ) {
 
                if ( !_flat_buttons ) {
                        float rheight = get_height()*0.5-REFLECTION_HEIGHT;
@@ -277,7 +282,7 @@ ArdourButton::render (cairo_t* cr)
 
                if (active_state() == Gtkmm2ext::ExplicitActive) {
 
-                       UINT_TO_RGBA (fill_color_active, &r, &g, &b, &a);
+                       UINT_TO_RGBA (fill_start_active_color, &r, &g, &b, &a);
                        cairo_set_line_width (cr, 2.0);
                        rounded_function (cr, 2, 2, get_width()-4, get_height()-4, _corner_radius - 2.0);
                        cairo_set_source_rgba (cr, r/255.0, g/255.0, b/255.0, a/255.0);
@@ -285,7 +290,7 @@ ArdourButton::render (cairo_t* cr)
 
                } else {
 
-                       UINT_TO_RGBA (fill_color_inactive_end, &r, &g, &b, &a);
+                       UINT_TO_RGBA (fill_start_inactive_color, &r, &g, &b, &a);
                        cairo_set_line_width (cr, 2.0);
                        rounded_function (cr, 2, 2, get_width()-4, get_height()-4, _corner_radius - 2.0);
                        cairo_set_source_rgba (cr, r/255.0, g/255.0, b/255.0, a/255.0);
@@ -323,7 +328,10 @@ ArdourButton::render (cairo_t* cr)
                cairo_new_path (cr);    
                cairo_set_source_rgba (cr, text_r, text_g, text_b, text_a);
 
-               if (_elements & Indicator) {
+               if ( (_elements & Menu) == Menu) {
+                       cairo_move_to (cr, text_margin, get_height()/2.0 - _text_height/2.0);
+                       pango_cairo_show_layout (cr, _layout->gobj());
+               } else if ( (_elements & Indicator)  == Indicator) {
                        if (_led_left) {
                                cairo_move_to (cr, text_margin + _diameter + 4, get_height()/2.0 - _text_height/2.0);
                        } else {
@@ -362,6 +370,23 @@ ArdourButton::render (cairo_t* cr)
                cairo_restore (cr);
        } 
 
+       if (((_elements & Menu)==Menu)) {
+       
+               cairo_save (cr);
+
+               cairo_translate (cr, 0,0 );
+               
+               //white arrow
+               cairo_set_source_rgba (cr, 1, 1, 1, 0.4);
+               cairo_move_to(cr, get_width() - ((_diameter/2.0) + 6.0), get_height()/2.0 +_diameter/4);
+               cairo_rel_line_to(cr, -_diameter/2, -_diameter/2);
+               cairo_rel_line_to(cr, _diameter, 0);
+               cairo_close_path(cr);
+               cairo_fill(cr);
+                       
+               cairo_restore (cr);
+       }
+       
        if (((_elements & Indicator)==Indicator)) {
 
                /* move to the center of the indicator/led */
@@ -426,6 +451,16 @@ ArdourButton::render (cairo_t* cr)
                        cairo_fill (cr);
                }
        }
+       if (_focused) {
+               rounded_function (cr, 1.5, 1.5, get_width() - 3, get_height() - 3, _corner_radius);
+               cairo_set_source_rgba (cr, 0.905, 0.917, 0.925, 0.8);
+               double dashes = 1;
+               cairo_set_dash (cr, &dashes, 1, 0);
+               cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
+               cairo_set_line_width (cr, 1.0);
+               cairo_stroke (cr);
+               cairo_set_dash (cr, 0, 0, 0);
+       }
 }
 
 void
@@ -437,7 +472,7 @@ ArdourButton::set_diameter (float d)
                _fixed_diameter = true;
        }
 
-       set_colors ();
+       build_patterns ();
        queue_resize ();
 }
 
@@ -492,35 +527,42 @@ ArdourButton::on_size_request (Gtk::Requisition* req)
        req->width += _corner_radius;
 }
 
+/**
+ * This sets the colors used for rendering based on the name of the button, and
+ * thus uses information from the GUI config data. 
+ */
 void
 ArdourButton::set_colors ()
 {
-       Glib::ustring button_name = get_name();
-
-       if (button_name == "") {
+       if (_fixed_colors_set) {
                return;
        }
-       
+       std::string name = get_name();
+
        border_color = ARDOUR_UI::config()->color_by_name ("button border");
+
+       fill_start_active_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill start active", name));
+       fill_end_active_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill end active", name));
        
-       fill_color_active_start = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill start active", button_name));
-       fill_color_active_end = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill end active", button_name));
-       
-       fill_color_inactive_start = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill start", button_name));
-       fill_color_inactive_end = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill end", button_name));
-       
-       text_color_active = ARDOUR_UI::config()->color_by_name (string_compose ("%1: text active", button_name));
-       text_color_inactive = ARDOUR_UI::config()->color_by_name (string_compose ("%1: text", button_name));
-    
-       led_color_active = ARDOUR_UI::config()->color_by_name (string_compose ("%1: led active", button_name));
-       led_color_inactive = ARDOUR_UI::config()->color_by_name (string_compose ("%1: led", button_name));
-}
+       fill_start_inactive_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill start", name));
+        fill_end_inactive_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill end", name));
 
-void ArdourButton::set_bg_colors (const uint32_t color_active, const uint32_t color_inactive)
+       text_active_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: text active", name));
+       text_inactive_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: text", name));
+
+       led_active_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: led active", name));
+       led_inactive_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: led", name));
+}
+/**
+ * This sets the colors used for rendering based on two fixed values, rather
+ * than basing them on the button name, and thus information in the GUI config
+ * data.
+ */
+void ArdourButton::set_fixed_colors (const uint32_t color_active, const uint32_t color_inactive)
 {
-       set_colors ();
-       set_name ("");
-       fill_color_active_start = fill_color_active_end = color_active;
+       _fixed_colors_set = true;
+
+       fill_start_active_color = fill_end_active_color = color_active;
 
        unsigned char r, g, b, a;
        UINT_TO_RGBA(color_active, &r, &g, &b, &a);
@@ -533,30 +575,27 @@ void ArdourButton::set_bg_colors (const uint32_t color_active, const uint32_t co
                (max (double(g), 0.) - min (double(g), 0.)) +
                (max (double(b), 0.) - min (double(b), 0.));
        
-       text_color_active =
-               text_color_inactive = (white_contrast > black_contrast) ?
+       text_active_color =
+               text_inactive_color = (white_contrast > black_contrast) ?
                RGBA_TO_UINT(255, 255, 255, 255) : /* use white */
                RGBA_TO_UINT(  0,   0,   0,   255);  /* use black */
        
-       fill_color_inactive_start = fill_color_inactive_end = color_inactive;
+       fill_start_inactive_color = fill_end_inactive_color = color_inactive;
+
+       /* XXX what about led colors ? */
 
        build_patterns ();
+       set_name (""); /* this will trigger a "style-changed" message and then set_colors() */
 }
 
 void
-ArdourButton::build_patterns()
+ArdourButton::build_patterns ()
 {
        uint32_t start_color;
        uint32_t end_color;
-       uint32_t r, g, b, a;
        uint32_t text_color;
        uint32_t led_color;
-
-       if (active_state() == Gtkmm2ext::ImplicitActive && (_tweaks & ImplicitUsesSolidColor)) {
-               fill_color_active = led_color_active;
-       } else {
-               fill_color_active = fill_color_active_end;
-       }
+       uint32_t r, g, b, a;
 
        if (shine_pattern) {
                cairo_pattern_destroy (shine_pattern);
@@ -575,13 +614,11 @@ ArdourButton::build_patterns()
 
        if (_elements & Body) {
 
-               start_color = ARDOUR_UI::config()->color_by_name (string_compose ("%1: fill start active", get_name()));
-               start_color = fill_color_active_start;
-               
                if (_flat_buttons) {
-                       end_color = start_color;
+                       end_color = start_color = fill_start_active_color;
                } else {
-                       end_color = fill_color_active;
+                       start_color = fill_start_active_color;
+                       end_color = fill_end_active_color;
                }
                UINT_TO_RGBA (start_color, &r, &g, &b, &a);
 
@@ -598,10 +635,10 @@ ArdourButton::build_patterns()
 
                fill_pattern = cairo_pattern_create_linear (0.0, 0.0, 0.0, get_height()-3);
                if (_flat_buttons) {
-                       end_color = start_color = fill_color_inactive_end;
+                       end_color = start_color = fill_start_inactive_color;
                } else {
-                       start_color = fill_color_inactive_start;
-                       end_color = fill_color_inactive_end;
+                       start_color = fill_start_inactive_color;
+                       end_color = fill_end_inactive_color;
                }
                UINT_TO_RGBA (start_color, &r, &g, &b, &a);
                cairo_pattern_add_color_stop_rgba (fill_pattern, 0, r/255.0,g/255.0,b/255.0, a/255.0);
@@ -611,17 +648,16 @@ ArdourButton::build_patterns()
                fill_pattern_active = cairo_pattern_create_linear (0.0, 0.0, 0.0, get_height()-3);
                if (_flat_buttons) {
                        if (active_state() == Gtkmm2ext::ImplicitActive && (_tweaks & ImplicitUsesSolidColor)) {
-                               end_color = start_color = led_color_active;
+                               end_color = start_color = led_active_color;
                        } else {
-                               end_color = start_color = fill_color_active_end;
+                               end_color = start_color = fill_end_active_color;
                        }
                } else {
                        if (active_state() == Gtkmm2ext::ImplicitActive && (_tweaks & ImplicitUsesSolidColor)) {
-                               start_color = led_color_inactive;
-                               end_color = led_color_active;
+                               end_color = start_color = led_active_color;
                        } else {
-                               start_color = fill_color_active_start;
-                               end_color = fill_color_active_end;
+                               start_color = fill_start_active_color;
+                               end_color = fill_end_active_color;
                        }
                }
                UINT_TO_RGBA (start_color, &r, &g, &b, &a);
@@ -647,17 +683,15 @@ ArdourButton::build_patterns()
                cairo_pattern_add_color_stop_rgba (reflection_pattern, 0, 1,1,1, active_state() ? 0.4 : 0.2);
                cairo_pattern_add_color_stop_rgba (reflection_pattern, 1, 1,1,1, 0.0);
        }
-       
-       /* text and LED colors */
 
        if (active_state() == Gtkmm2ext::ExplicitActive || ((_tweaks & ImplicitUsesSolidColor) && active_state() == Gtkmm2ext::ImplicitActive)) {
-               text_color = text_color_active;
-               led_color = led_color_active;
+               text_color = text_active_color;
+               led_color = led_active_color;
        } else {
-               text_color = text_color_inactive;
-               led_color = led_color_inactive;
+               text_color = text_inactive_color;
+               led_color = led_inactive_color;
        }
-
+       
        UINT_TO_RGBA (text_color, &r, &g, &b, &a);
        text_r = r/255.0;
        text_g = g/255.0;
@@ -832,6 +866,13 @@ ArdourButton::on_style_changed (const RefPtr<Gtk::Style>&)
        build_patterns ();
 }
 
+void
+ArdourButton::on_name_changed ()
+{
+       set_colors ();
+       build_patterns ();
+}
+
 void
 ArdourButton::setup_led_rect ()
 {
@@ -880,6 +921,7 @@ ArdourButton::set_active_state (Gtkmm2ext::ActiveState s)
        bool changed = (_active_state != s);
        CairoWidget::set_active_state (s);
        if (changed) {
+               set_colors ();
                build_patterns ();
        }
 }
@@ -890,10 +932,42 @@ ArdourButton::set_visual_state (Gtkmm2ext::VisualState s)
        bool changed = (_visual_state != s);
        CairoWidget::set_visual_state (s);
        if (changed) {
+               set_colors ();
                build_patterns ();
        }
 }
        
+
+bool
+ArdourButton::on_focus_in_event (GdkEventFocus* ev)
+{
+       _focused = true;
+       queue_draw ();
+       return CairoWidget::on_focus_in_event (ev);
+}
+
+bool
+ArdourButton::on_focus_out_event (GdkEventFocus* ev)
+{
+       _focused = false;
+       queue_draw ();
+       return CairoWidget::on_focus_out_event (ev);
+}
+
+bool
+ArdourButton::on_key_release_event (GdkEventKey *ev) {
+       if (_focused &&
+                       (ev->keyval == GDK_KEY_space || ev->keyval == GDK_Return))
+       {
+               signal_clicked();
+               if (_action) {
+                       _action->activate ();
+               }
+               return true;
+       }
+       return CairoWidget::on_key_release_event (ev);
+}
+
 bool
 ArdourButton::on_enter_notify_event (GdkEventCrossing* ev)
 {
@@ -938,6 +1012,7 @@ ArdourButton::action_sensitivity_changed ()
        
 }
 
+
 void
 ArdourButton::action_visibility_changed ()
 {
@@ -966,14 +1041,12 @@ void
 ArdourButton::set_elements (Element e)
 {
        _elements = e;
-       build_patterns ();
 }
 
 void
 ArdourButton::add_elements (Element e)
 {
        _elements = (ArdourButton::Element) (_elements | e);
-       build_patterns ();
 }
 
 void