substantial changes in color management, involving a reduction in the use of Gdk...
[ardour.git] / gtk2_ardour / editor_group_tabs.cc
1 /*
2     Copyright (C) 2009 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 */
19
20 #include "gtkmm2ext/utils.h"
21
22 #include "ardour/route_group.h"
23
24 #include "canvas/utils.h"
25
26 #include "editor.h"
27 #include "editor_group_tabs.h"
28 #include "editor_route_groups.h"
29 #include "editor_routes.h"
30 #include "rgb_macros.h"
31 #include "route_time_axis.h"
32 #include "utils.h"
33
34 #include "i18n.h"
35
36 using namespace std;
37 using namespace ARDOUR;
38
39 EditorGroupTabs::EditorGroupTabs (Editor* e)
40         : EditorComponent (e)
41 {
42
43 }
44
45 list<GroupTabs::Tab>
46 EditorGroupTabs::compute_tabs () const
47 {
48         list<Tab> tabs;
49
50         Tab tab;
51         tab.from = 0;
52         tab.group = 0;
53
54         int32_t y = 0;
55         for (TrackViewList::iterator i = _editor->track_views.begin(); i != _editor->track_views.end(); ++i) {
56
57                 if ((*i)->marked_for_display() == false) {
58                         continue;
59                 }
60
61                 RouteGroup* g = (*i)->route_group ();
62
63                 if (g != tab.group) {
64                         if (tab.group) {
65                                 tab.to = y;
66                                 tabs.push_back (tab);
67                         }
68
69                         tab.from = y;
70                         tab.group = g;
71                         if (g) {
72                                 tab.color = group_color (g);
73                         }
74                 }
75
76                 y += (*i)->effective_height ();
77         }
78
79         if (tab.group) {
80                 tab.to = y;
81                 tabs.push_back (tab);
82         }
83
84         return tabs;
85 }
86
87 void
88 EditorGroupTabs::draw_tab (cairo_t* cr, Tab const & tab) const
89 {
90         double const arc_radius = get_width();
91         double r, g, b, a;
92         
93         if (tab.group && tab.group->is_active()) {
94                 ArdourCanvas::color_to_rgba (tab.color, r, g, b, a);
95         } else {
96                 r = 0.0;
97                 g = 0.0;
98                 b = 0.0;
99         }
100
101         a = 1.0;
102
103         cairo_set_source_rgba (cr, r, g, b, a);
104         cairo_move_to (cr, 0, tab.from + arc_radius);
105         cairo_arc (cr, get_width(), tab.from + arc_radius, arc_radius, M_PI, 3 * M_PI / 2);
106         cairo_line_to (cr, get_width(), tab.to);
107         cairo_arc (cr, get_width(), tab.to - arc_radius, arc_radius, M_PI / 2, M_PI);
108         cairo_line_to (cr, 0, tab.from + arc_radius);
109         cairo_fill (cr);
110
111         if (tab.group) {
112                 pair<string, double> const f = Gtkmm2ext::fit_to_pixels (cr, tab.group->name(), tab.to - tab.from - arc_radius * 2);
113
114                 cairo_text_extents_t ext;
115                 cairo_text_extents (cr, tab.group->name().c_str(), &ext);
116
117                 ArdourCanvas::Color c = contrasting_text_color (ArdourCanvas::rgba_to_color (r, g, b, a));
118                 ArdourCanvas::color_to_rgba (c, r, g, b, a);
119
120                 cairo_set_source_rgb (cr, r, g, b);
121                 cairo_move_to (cr, get_width() - ext.height / 2, tab.from + (f.second + tab.to - tab.from) / 2);
122                 cairo_save (cr);
123                 cairo_rotate (cr, - M_PI / 2);
124                 cairo_show_text (cr, f.first.c_str());
125                 cairo_restore (cr);
126         }
127 }
128
129 double
130 EditorGroupTabs::primary_coordinate (double, double y) const
131 {
132         return y;
133 }
134
135 RouteList
136 EditorGroupTabs::routes_for_tab (Tab const * t) const
137 {
138         RouteList routes;
139         int32_t y = 0;
140
141         for (TrackViewList::iterator i = _editor->track_views.begin(); i != _editor->track_views.end(); ++i) {
142
143                 if ((*i)->marked_for_display() == false) {
144                         continue;
145                 }
146
147                 RouteTimeAxisView* rtv = dynamic_cast<RouteTimeAxisView*> (*i);
148                 if (rtv) {
149
150                         if (y >= t->to) {
151                                 /* tab finishes before this track starts */
152                                 break;
153                         }
154
155                         double const h = y + (*i)->effective_height() / 2;
156
157                         if (t->from < h && t->to > h) {
158                                 routes.push_back (rtv->route ());
159                         }
160                 }
161
162                 y += (*i)->effective_height ();
163         }
164
165         return routes;
166 }
167
168
169 void
170 EditorGroupTabs::add_menu_items (Gtk::Menu* m, RouteGroup* g)
171 {
172         using namespace Gtk::Menu_Helpers;
173
174         if (g) {
175                 MenuList& items = m->items ();
176                 items.push_back (MenuElem (_("Fit to Window"), sigc::bind (sigc::mem_fun (*_editor, &Editor::fit_route_group), g)));
177         }
178 }
179
180 PBD::PropertyList
181 EditorGroupTabs::default_properties () const
182 {
183         PBD::PropertyList plist;
184
185         plist.add (Properties::active, true);
186         plist.add (Properties::mute, true);
187         plist.add (Properties::solo, true);
188         plist.add (Properties::recenable, true);
189
190         return plist;
191 }
192
193 RouteList
194 EditorGroupTabs::selected_routes () const
195 {
196         RouteList rl;
197
198         for (TrackSelection::iterator i = _editor->get_selection().tracks.begin(); i != _editor->get_selection().tracks.end(); ++i) {
199                 RouteTimeAxisView* rtv = dynamic_cast<RouteTimeAxisView*> (*i);
200                 if (rtv) {
201                         rl.push_back (rtv->route());
202                 }
203         }
204
205         return rl;
206 }
207
208 void
209 EditorGroupTabs::sync_order_keys ()
210 {
211         _editor->_routes->sync_order_keys_from_treeview ();
212 }