enough with umpteen "i18n.h" files. Consolidate on pbd/i18n.h
[ardour.git] / gtk2_ardour / mixer_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 <boost/foreach.hpp>
21
22 #include "gtkmm2ext/utils.h"
23
24 #include "ardour/route_group.h"
25
26 #include "canvas/colors.h"
27
28 #include "mixer_group_tabs.h"
29 #include "mixer_strip.h"
30 #include "mixer_ui.h"
31 #include "rgb_macros.h"
32 #include "route_group_dialog.h"
33 #include "ui_config.h"
34 #include "utils.h"
35
36 #include "pbd/i18n.h"
37
38 using namespace std;
39 using namespace Gtk;
40 using namespace ARDOUR;
41 using namespace ARDOUR_UI_UTILS;
42 using namespace PBD;
43
44 MixerGroupTabs::MixerGroupTabs (Mixer_UI* m)
45         : _mixer (m)
46 {
47
48 }
49
50
51 list<GroupTabs::Tab>
52 MixerGroupTabs::compute_tabs () const
53 {
54         list<Tab> tabs;
55
56         Tab tab;
57         tab.from = 0;
58         tab.group = 0;
59
60         int32_t x = 0;
61         TreeModel::Children rows = _mixer->track_model->children ();
62         for (TreeModel::Children::iterator i = rows.begin(); i != rows.end(); ++i) {
63
64                 AxisView* av = (*i)[_mixer->stripable_columns.strip];
65                 MixerStrip* s = dynamic_cast<MixerStrip*> (av);
66
67                 if (!s) {
68                         continue;
69                 }
70
71                 if (s->route()->is_master() || s->route()->is_monitor() || !s->marked_for_display()) {
72                         continue;
73                 }
74
75                 RouteGroup* g = s->route_group ();
76
77                 if (g != tab.group) {
78                         if (tab.group) {
79                                 tab.to = x;
80                                 tabs.push_back (tab);
81                         }
82
83                         tab.from = x;
84                         tab.group = g;
85
86                         if (g) {
87                                 tab.color = group_color (g);
88                         }
89                 }
90
91                 int ww = 0, wh = 0;
92                 s->get_size_request (ww, wh); // widget may not be realized, get_width() is invalid.
93                 x += ww;
94         }
95
96         if (tab.group) {
97                 tab.to = x;
98                 tabs.push_back (tab);
99         }
100
101         return tabs;
102 }
103
104 void
105 MixerGroupTabs::draw_tab (cairo_t* cr, Tab const & tab) const
106 {
107         double const arc_radius = get_height();
108         double r, g, b, a;
109
110         if (tab.group && tab.group->is_active()) {
111                 ArdourCanvas::color_to_rgba (tab.color, r, g, b, a);
112         } else {
113                 ArdourCanvas::color_to_rgba (UIConfiguration::instance().color ("inactive group tab"), r, g, b, a);
114         }
115
116         a = 1.0;
117
118         cairo_set_source_rgba (cr, r, g, b, a);
119         cairo_arc (cr, tab.from + arc_radius, get_height(), arc_radius, M_PI, 3 * M_PI / 2);
120         cairo_line_to (cr, tab.to - arc_radius, 0);
121         cairo_arc (cr, tab.to - arc_radius, get_height(), arc_radius, 3 * M_PI / 2, 2 * M_PI);
122         cairo_line_to (cr, tab.from, get_height());
123         cairo_fill (cr);
124
125         if (tab.group) {
126                 pair<string, double> const f = Gtkmm2ext::fit_to_pixels (cr, tab.group->name(), tab.to - tab.from - arc_radius * 2);
127
128                 cairo_text_extents_t ext;
129                 cairo_text_extents (cr, tab.group->name().c_str(), &ext);
130
131                 ArdourCanvas::Color c = ArdourCanvas::contrasting_text_color (ArdourCanvas::rgba_to_color (r, g, b, a));
132                 ArdourCanvas::color_to_rgba (c, r, g, b, a);
133
134                 cairo_set_source_rgb (cr, r, g, b);
135                 cairo_move_to (cr, tab.from + (tab.to - tab.from - f.second) / 2, get_height() - ext.height / 2);
136                 cairo_save (cr);
137                 cairo_show_text (cr, f.first.c_str());
138                 cairo_restore (cr);
139         }
140 }
141
142 double
143 MixerGroupTabs::primary_coordinate (double x, double) const
144 {
145         return x;
146 }
147
148 RouteList
149 MixerGroupTabs::routes_for_tab (Tab const * t) const
150 {
151         RouteList routes;
152         int32_t x = 0;
153
154         TreeModel::Children rows = _mixer->track_model->children ();
155         for (TreeModel::Children::iterator i = rows.begin(); i != rows.end(); ++i) {
156
157                 AxisView* av = (*i)[_mixer->stripable_columns.strip];
158                 MixerStrip* s = dynamic_cast<MixerStrip*> (av);
159
160                 if (!s) {
161                         continue;
162                 }
163
164                 if (s->route()->is_master() || s->route()->is_monitor() || !s->marked_for_display()) {
165                         continue;
166                 }
167
168                 if (x >= t->to) {
169                         /* tab finishes before this track starts */
170                         break;
171                 }
172
173                 double const h = x + s->get_width() / 2;
174
175                 if (t->from < h && t->to > h) {
176                         routes.push_back (s->route ());
177                 }
178
179                 x += s->get_width ();
180         }
181
182         return routes;
183 }
184
185 RouteList
186 MixerGroupTabs::selected_routes () const
187 {
188         RouteList rl;
189         BOOST_FOREACH (AxisView* r, _mixer->selection().axes) {
190                 boost::shared_ptr<Route> rp = boost::dynamic_pointer_cast<Route> (r->stripable());
191                 if (rp) {
192                         rl.push_back (rp);
193                 }
194         }
195         return rl;
196 }
197