shared peak-display reset in mixer+meterbridge
[ardour.git] / gtk2_ardour / meter_strip.cc
1 /*
2     Copyright (C) 2013 Paul Davis
3     Author: Robin Gareus
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20 #include <list>
21
22 #include <sigc++/bind.h>
23
24 #include "ardour/session.h"
25 #include "ardour/route.h"
26 #include "ardour/route_group.h"
27 #include "ardour/meter.h"
28
29 #include "ardour/audio_track.h"
30 #include "ardour/midi_track.h"
31
32 #include <gtkmm2ext/gtk_ui.h>
33 #include <gtkmm2ext/keyboard.h>
34 #include <gtkmm2ext/utils.h>
35 #include <gtkmm2ext/rgb_macros.h>
36
37 #include "ardour_ui.h"
38 #include "global_signals.h"
39 #include "logmeter.h"
40 #include "gui_thread.h"
41 #include "ardour_window.h"
42 #include "utils.h"
43
44 #include "meterbridge.h"
45 #include "meter_strip.h"
46 #include "meter_patterns.h"
47
48 #include "i18n.h"
49
50 using namespace ARDOUR;
51 using namespace PBD;
52 using namespace Gtk;
53 using namespace Gtkmm2ext;
54 using namespace std;
55
56 PBD::Signal1<void,MeterStrip*> MeterStrip::CatchDeletion;
57 PBD::Signal0<void> MeterStrip::MetricChanged;
58
59 MeterStrip::MeterStrip (int metricmode)
60         : AxisView(0)
61         , RouteUI(0)
62 {
63         level_meter = 0;
64         set_spacing(2);
65         peakbx.set_size_request(-1, 14);
66         btnbox.set_size_request(-1, 16);
67         namebx.set_size_request(18, 52);
68
69         set_metric_mode(metricmode);
70
71         set_size_request_to_display_given_text (meter_metric_area, "-8888", 1, 0);
72         meter_metric_area.signal_expose_event().connect (
73                         sigc::mem_fun(*this, &MeterStrip::meter_metrics_expose));
74
75         meterbox.pack_start(meter_metric_area, true, false);
76
77         pack_start (peakbx, false, false);
78         pack_start (meterbox, true, true);
79         pack_start (btnbox, false, false);
80         pack_start (namebx, false, false);
81
82         peakbx.show();
83         btnbox.show();
84         meter_metric_area.show();
85         meterbox.show();
86         namebx.show();
87
88         UI::instance()->theme_changed.connect (sigc::mem_fun(*this, &MeterStrip::on_theme_changed));
89         ColorsChanged.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
90         DPIReset.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
91 }
92
93 MeterStrip::MeterStrip (Session* sess, boost::shared_ptr<ARDOUR::Route> rt)
94         : AxisView(sess)
95         , RouteUI(sess)
96         , _route(rt)
97         , peak_display()
98 {
99         set_spacing(2);
100         RouteUI::set_route (rt);
101
102         _has_midi = false;
103
104         int meter_width = 6;
105         if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
106                 meter_width = 12;
107         }
108
109         // level meter + ticks
110         level_meter = new LevelMeter(sess);
111         level_meter->set_meter (_route->shared_peak_meter().get());
112         level_meter->clear_meters();
113         level_meter->setup_meters (220, meter_width, 6);
114
115         meter_align.set(0.5, 0.5, 0.0, 1.0);
116         meter_align.add(*level_meter);
117
118         meterbox.pack_start(meter_ticks1_area, true, false);
119         meterbox.pack_start(meter_align, true, true);
120         meterbox.pack_start(meter_ticks2_area, true, false);
121
122         // peak display
123         peak_display.set_name ("meterbridge peakindicator");
124         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
125         max_peak = minus_infinity();
126         peak_display.unset_flags (Gtk::CAN_FOCUS);
127         peak_display.set_size_request(12, 8);
128         peak_display.set_corner_radius(2);
129
130         peak_align.set(0.5, 1.0, 1.0, 0.8);
131         peak_align.add(peak_display);
132         peakbx.pack_start(peak_align, true, true, 3);
133         peakbx.set_size_request(-1, 14);
134
135         // add track-name label
136         name_label.set_text(_route->name().c_str());
137         name_label.set_corner_radius(2);
138         name_label.set_name("meterbridge label");
139         name_label.set_angle(-90.0);
140         name_label.layout()->set_ellipsize (Pango::ELLIPSIZE_END);
141         name_label.layout()->set_width(48 * PANGO_SCALE);
142         name_label.set_size_request(18, 50);
143         name_label.set_alignment(-1.0, .5);
144
145         namebx.set_size_request(18, 52);
146         namebx.pack_start(name_label, true, false, 3);
147
148         // rec-enable button
149         btnbox.pack_start(*rec_enable_button, true, false);
150         rec_enable_button->set_corner_radius(2);
151         btnbox.set_size_request(-1, 16);
152
153         pack_start (peakbx, false, false);
154         pack_start (meterbox, true, true);
155         pack_start (btnbox, false, false);
156         pack_start (namebx, false, false);
157
158         peak_display.show();
159         peakbx.show();
160         meter_ticks1_area.show();
161         meter_ticks2_area.show();
162         meterbox.show();
163         level_meter->show();
164         meter_align.show();
165         peak_align.show();
166         btnbox.show();
167         name_label.show();
168         namebx.show();
169
170         _route->shared_peak_meter()->ConfigurationChanged.connect (
171                         route_connections, invalidator (*this), boost::bind (&MeterStrip::meter_configuration_changed, this, _1), gui_context()
172                         );
173
174         ResetAllPeakDisplays.connect (sigc::mem_fun(*this, &MeterStrip::reset_peak_display));
175         ResetGroupPeakDisplays.connect (sigc::mem_fun(*this, &MeterStrip::reset_group_peak_display));
176
177         meter_configuration_changed (_route->shared_peak_meter()->input_streams ());
178
179         meter_ticks1_area.set_size_request(3,-1);
180         meter_ticks2_area.set_size_request(3,-1);
181         meter_ticks1_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks1_expose));
182         meter_ticks2_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks2_expose));
183
184         _route->DropReferences.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::self_delete, this), gui_context());
185         _route->PropertyChanged.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::strip_property_changed, this, _1), gui_context());
186
187         peak_display.signal_button_release_event().connect (sigc::mem_fun(*this, &MeterStrip::peak_button_release), false);
188
189         UI::instance()->theme_changed.connect (sigc::mem_fun(*this, &MeterStrip::on_theme_changed));
190         ColorsChanged.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
191         DPIReset.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
192 }
193
194 MeterStrip::~MeterStrip ()
195 {
196         delete level_meter;
197         CatchDeletion (this);
198 }
199
200 void
201 MeterStrip::self_delete ()
202 {
203         delete this;
204 }
205
206 void
207 MeterStrip::update_rec_display ()
208 {
209         RouteUI::update_rec_display ();
210 }
211
212 std::string
213 MeterStrip::state_id() const
214 {
215         return string_compose ("mtrs %1", _route->id().to_s());
216 }
217
218 void
219 MeterStrip::set_button_names()
220 {
221         rec_enable_button->set_text ("");
222         rec_enable_button->set_image (::get_icon (X_("record_normal_red")));
223 }
224
225 void
226 MeterStrip::strip_property_changed (const PropertyChange& what_changed)
227 {
228         if (!what_changed.contains (ARDOUR::Properties::name)) {
229                 return;
230         }
231         ENSURE_GUI_THREAD (*this, &MeterStrip::strip_name_changed, what_changed)
232         name_label.set_text(_route->name());
233 }
234
235 void
236 MeterStrip::fast_update ()
237 {
238         float mpeak = level_meter->update_meters();
239         if (mpeak > max_peak) {
240                 max_peak = mpeak;
241                 if (mpeak >= 0.0f) {
242                         peak_display.set_name ("meterbridge peakindicator on");
243                         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
244                 }
245         }
246 }
247
248 void
249 MeterStrip::on_theme_changed()
250 {
251         meter_clear_pattern_cache();
252
253         if (level_meter && _route) {
254                 int meter_width = 6;
255                 if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
256                         meter_width = 12;
257                 }
258                 level_meter->setup_meters (220, meter_width, 6);
259         }
260         meter_metric_area.queue_draw();
261         meter_ticks1_area.queue_draw();
262         meter_ticks2_area.queue_draw();
263 }
264
265 void
266 MeterStrip::meter_configuration_changed (ChanCount c)
267 {
268         int type = 0;
269         _types.clear ();
270         bool old_has_midi = _has_midi;
271
272         for (DataType::iterator i = DataType::begin(); i != DataType::end(); ++i) {
273                 if (c.get (*i) > 0) {
274                         _types.push_back (*i);
275                         type |= 1 << (*i);
276                 }
277         }
278
279         // TODO draw Inactive routes or busses with different styles
280         if (boost::dynamic_pointer_cast<AudioTrack>(_route) == 0
281                         && boost::dynamic_pointer_cast<MidiTrack>(_route) == 0
282                         ) {
283                 meter_ticks1_area.set_name ("AudioBusMetricsLeft");
284                 meter_ticks2_area.set_name ("AudioBusMetricsRight");
285                 _has_midi = false;
286         }
287         else if (type == (1 << DataType::AUDIO)) {
288                 meter_ticks1_area.set_name ("AudioTrackMetricsLeft");
289                 meter_ticks2_area.set_name ("AudioTrackMetricsRight");
290                 _has_midi = false;
291         }
292         else if (type == (1 << DataType::MIDI)) {
293                 meter_ticks1_area.set_name ("MidiTrackMetricsLeft");
294                 meter_ticks2_area.set_name ("MidiTrackMetricsRight");
295                 _has_midi = true;
296         } else {
297                 meter_ticks1_area.set_name ("AudioMidiTrackMetricsLeft");
298                 meter_ticks2_area.set_name ("AudioMidiTrackMetricsRight");
299                 _has_midi = true;
300         }
301
302         if (old_has_midi != _has_midi) MetricChanged();
303         on_theme_changed();
304 }
305
306 void
307 MeterStrip::on_size_request (Gtk::Requisition* r)
308 {
309         meter_clear_pattern_cache();
310         VBox::on_size_request(r);
311 }
312
313 void
314 MeterStrip::on_size_allocate (Gtk::Allocation& a)
315 {
316         meter_clear_pattern_cache();
317         const int wh = a.get_height();
318         int nh = ceilf(wh * .11f);
319         if (nh < 52) nh = 52;
320         if (nh > 148) nh = 148;
321         namebx.set_size_request(18, nh);
322         if (_route) {
323                 name_label.set_size_request(18, nh-2);
324                 name_label.layout()->set_width((nh-4) * PANGO_SCALE);
325         }
326         VBox::on_size_allocate(a);
327 }
328
329 gint
330 MeterStrip::meter_metrics_expose (GdkEventExpose *ev)
331 {
332         return meter_expose_metrics(ev, _types, &meter_metric_area);
333 }
334
335 void
336 MeterStrip::set_metric_mode (int metricmode)
337 {
338         _types.clear ();
339         switch(metricmode) {
340                 case 0:
341                         meter_metric_area.set_name ("MidiTrackMetricsLeft");
342                         _types.push_back (DataType::MIDI);
343                         break;
344                 case 1:
345                         meter_metric_area.set_name ("AudioTrackMetricsLeft");
346                         _types.push_back (DataType::AUDIO);
347                         break;
348                 case 2:
349                         meter_metric_area.set_name ("MidiTrackMetricsRight");
350                         _types.push_back (DataType::MIDI);
351                         break;
352                 case 3:
353                 default:
354                         meter_metric_area.set_name ("AudioTrackMetricsRight");
355                         _types.push_back (DataType::AUDIO);
356                         break;
357         }
358
359         meter_metric_area.queue_draw ();
360 }
361
362 gint
363 MeterStrip::meter_ticks1_expose (GdkEventExpose *ev)
364 {
365         return meter_expose_ticks(ev, _types, &meter_ticks1_area);
366 }
367
368 gint
369 MeterStrip::meter_ticks2_expose (GdkEventExpose *ev)
370 {
371         return meter_expose_ticks(ev, _types, &meter_ticks2_area);
372 }
373
374 void
375 MeterStrip::reset_group_peak_display (RouteGroup* group)
376 {
377         /* UNUSED -- need connection w/mixer || other meters */
378         if (_route && group == _route->route_group()) {
379                 reset_peak_display ();
380         }
381 }
382
383 void
384 MeterStrip::reset_peak_display ()
385 {
386         _route->shared_peak_meter()->reset_max();
387         level_meter->clear_meters();
388         max_peak = -INFINITY;
389         peak_display.set_name ("meterbridge peakindicator");
390         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
391 }
392
393 bool
394 MeterStrip::peak_button_release (GdkEventButton* ev)
395 {
396         if (ev->button == 1 && Keyboard::modifier_state_equals (ev->state, Keyboard::PrimaryModifier|Keyboard::TertiaryModifier)) {
397                 ResetAllPeakDisplays ();
398         } else if (ev->button == 1 && Keyboard::modifier_state_equals (ev->state, Keyboard::PrimaryModifier)) {
399                 if (_route) {
400                         ResetGroupPeakDisplays (_route->route_group());
401                 }
402         } else {
403                 reset_peak_display ();
404         }
405         return true;
406 }