a397ed232b5f301daeae9f5f570fdfe91a12be54
[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         _strip_type = 0;
65         set_spacing(2);
66         peakbx.set_size_request(-1, 14);
67         namebx.set_size_request(18, 52);
68
69         set_metric_mode(metricmode);
70
71         meter_metric_area.set_size_request(25, 10);
72         meter_metric_area.signal_expose_event().connect (
73                         sigc::mem_fun(*this, &MeterStrip::meter_metrics_expose));
74         RedrawMetrics.connect (sigc::mem_fun(*this, &MeterStrip::redraw_metrics));
75
76         meterbox.pack_start(meter_metric_area, true, false);
77
78         pack_start (peakbx, false, false);
79         pack_start (meterbox, true, true);
80         pack_start (btnbox, false, false);
81         pack_start (namebx, false, false);
82
83         peakbx.show();
84         btnbox.show();
85         meter_metric_area.show();
86         meterbox.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         SessionHandlePtr::set_session (sess);
102
103         _has_midi = false;
104
105         int meter_width = 6;
106         if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
107                 meter_width = 12;
108         }
109
110         // level meter + ticks
111         level_meter = new LevelMeter(sess);
112         level_meter->set_meter (_route->shared_peak_meter().get());
113         level_meter->clear_meters();
114         level_meter->setup_meters (220, meter_width, 6);
115         level_meter->set_type (_route->meter_type());
116         level_meter->ButtonPress.connect_same_thread (level_meter_connection, boost::bind (&MeterStrip::level_meter_button_press, this, _1));
117         level_meter->MeterTypeChanged.connect_same_thread (level_meter_connection, boost::bind (&MeterStrip::meter_type_changed, this, _1));
118
119         meter_align.set(0.5, 0.5, 0.0, 1.0);
120         meter_align.add(*level_meter);
121
122         meterbox.pack_start(meter_ticks1_area, true, false);
123         meterbox.pack_start(meter_align, true, true);
124         meterbox.pack_start(meter_ticks2_area, true, false);
125
126         // peak display
127         peak_display.set_name ("meterbridge peakindicator");
128         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
129         max_peak = minus_infinity();
130         peak_display.unset_flags (Gtk::CAN_FOCUS);
131         peak_display.set_size_request(12, 8);
132         peak_display.set_corner_radius(2);
133
134         peak_align.set(0.5, 1.0, 1.0, 0.8);
135         peak_align.add(peak_display);
136         peakbx.pack_start(peak_align, true, true, 3);
137         peakbx.set_size_request(-1, 14);
138
139         // add track-name label
140         name_label.set_text(_route->name());
141         name_label.set_corner_radius(2);
142         name_label.set_name("meterbridge label");
143         name_label.set_angle(-90.0);
144         name_label.layout()->set_ellipsize (Pango::ELLIPSIZE_END);
145         name_label.layout()->set_width(48 * PANGO_SCALE);
146         name_label.set_size_request(18, 50);
147         name_label.set_alignment(-1.0, .5);
148         ARDOUR_UI::instance()->set_tip (name_label, _route->name());
149
150         namebx.set_size_request(18, 52);
151         namebx.pack_start(name_label, true, false, 3);
152
153         recbox.pack_start(*rec_enable_button, true, false);
154         btnbox.pack_start(recbox, false, false, 1);
155         mutebox.pack_start(*mute_button, true, false);
156         btnbox.pack_start(mutebox, false, false, 1);
157         solobox.pack_start(*solo_button, true, false);
158         btnbox.pack_start(solobox, false, false, 1);
159
160         rec_enable_button->set_corner_radius(2);
161         rec_enable_button->set_size_request(16, 16);
162
163         mute_button->set_corner_radius(2);
164         mute_button->set_size_request(16, 16);
165
166         solo_button->set_corner_radius(2);
167         solo_button->set_size_request(16, 16);
168
169         mutebox.set_size_request(16, 16);
170         solobox.set_size_request(16, 16);
171         recbox.set_size_request(16, 16);
172
173         update_button_box();
174         update_name_box();
175
176         pack_start (peakbx, false, false);
177         pack_start (meterbox, true, true);
178         pack_start (btnbox, false, false);
179         pack_start (namebx, false, false);
180         name_label.show();
181         peak_display.show();
182         peakbx.show();
183         meter_ticks1_area.show();
184         meter_ticks2_area.show();
185         meterbox.show();
186         level_meter->show();
187         meter_align.show();
188         peak_align.show();
189         btnbox.show();
190
191         _route->shared_peak_meter()->ConfigurationChanged.connect (
192                         route_connections, invalidator (*this), boost::bind (&MeterStrip::meter_configuration_changed, this, _1), gui_context()
193                         );
194
195         ResetAllPeakDisplays.connect (sigc::mem_fun(*this, &MeterStrip::reset_peak_display));
196         ResetRoutePeakDisplays.connect (sigc::mem_fun(*this, &MeterStrip::reset_route_peak_display));
197         ResetGroupPeakDisplays.connect (sigc::mem_fun(*this, &MeterStrip::reset_group_peak_display));
198         RedrawMetrics.connect (sigc::mem_fun(*this, &MeterStrip::redraw_metrics));
199         SetMeterTypeMulti.connect (sigc::mem_fun(*this, &MeterStrip::set_meter_type_multi));
200
201         meter_configuration_changed (_route->shared_peak_meter()->input_streams ());
202
203         meter_ticks1_area.set_size_request(3,-1);
204         meter_ticks2_area.set_size_request(3,-1);
205         meter_ticks1_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks1_expose));
206         meter_ticks2_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks2_expose));
207
208         _route->DropReferences.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::self_delete, this), gui_context());
209         _route->PropertyChanged.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::strip_property_changed, this, _1), gui_context());
210
211         peak_display.signal_button_release_event().connect (sigc::mem_fun(*this, &MeterStrip::peak_button_release), false);
212
213         UI::instance()->theme_changed.connect (sigc::mem_fun(*this, &MeterStrip::on_theme_changed));
214         ColorsChanged.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
215         DPIReset.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
216         Config->ParameterChanged.connect (*this, invalidator (*this), ui_bind (&MeterStrip::parameter_changed, this, _1), gui_context());
217         sess->config.ParameterChanged.connect (*this, invalidator (*this), ui_bind (&MeterStrip::parameter_changed, this, _1), gui_context());
218
219         if (_route->is_master()) {
220                 _strip_type = 4;
221         }
222         else if (boost::dynamic_pointer_cast<AudioTrack>(_route) == 0
223                         && boost::dynamic_pointer_cast<MidiTrack>(_route) == 0) {
224                 /* non-master bus */
225                 _strip_type = 3;
226         }
227         else if (boost::dynamic_pointer_cast<MidiTrack>(_route)) {
228                 _strip_type = 2;
229         }
230         else {
231                 _strip_type = 1;
232         }
233 }
234
235 MeterStrip::~MeterStrip ()
236 {
237         delete level_meter;
238         CatchDeletion (this);
239 }
240
241 void
242 MeterStrip::self_delete ()
243 {
244         delete this;
245 }
246
247 void
248 MeterStrip::set_session (Session* s)
249 {
250         SessionHandlePtr::set_session (s);
251         if (!s) return;
252         s->config.ParameterChanged.connect (*this, invalidator (*this), ui_bind (&MeterStrip::parameter_changed, this, _1), gui_context());
253         update_button_box();
254         update_name_box();
255 }
256
257 void
258 MeterStrip::update_rec_display ()
259 {
260         RouteUI::update_rec_display ();
261 }
262
263 std::string
264 MeterStrip::state_id() const
265 {
266         return string_compose ("mtrs %1", _route->id().to_s());
267 }
268
269 void
270 MeterStrip::set_button_names()
271 {
272         mute_button->set_text (_("M"));
273         rec_enable_button->set_text ("");
274         rec_enable_button->set_image (::get_icon (X_("record_normal_red")));
275
276         if (_route && _route->solo_safe()) {
277                 solo_button->remove ();
278                 if (solo_safe_pixbuf == 0) {
279                         solo_safe_pixbuf =::get_icon("solo-safe-icon");
280                 }
281                 solo_button->set_image (solo_safe_pixbuf);
282                 solo_button->set_text (string());
283         } else {
284                 solo_button->set_image (Glib::RefPtr<Gdk::Pixbuf>());
285                 if (!Config->get_solo_control_is_listen_control()) {
286                         solo_button->set_text (_("S"));
287                 } else {
288                         switch (Config->get_listen_position()) {
289                         case AfterFaderListen:
290                                 solo_button->set_text (_("A"));
291                                 break;
292                         case PreFaderListen:
293                                 solo_button->set_text (_("P"));
294                                 break;
295                         }
296                 }
297         }
298
299 }
300
301 void
302 MeterStrip::strip_property_changed (const PropertyChange& what_changed)
303 {
304         if (!what_changed.contains (ARDOUR::Properties::name)) {
305                 return;
306         }
307         ENSURE_GUI_THREAD (*this, &MeterStrip::strip_name_changed, what_changed)
308         name_label.set_text(_route->name());
309         ARDOUR_UI::instance()->set_tip (name_label, _route->name());
310 }
311
312 void
313 MeterStrip::fast_update ()
314 {
315         float mpeak = level_meter->update_meters();
316         if (mpeak > max_peak) {
317                 max_peak = mpeak;
318                 if (mpeak >= Config->get_meter_peak()) {
319                         peak_display.set_name ("meterbridge peakindicator on");
320                         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
321                 }
322         }
323 }
324
325 void
326 MeterStrip::on_theme_changed()
327 {
328         if (level_meter && _route) {
329                 int meter_width = 6;
330                 if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
331                         meter_width = 12;
332                 }
333                 level_meter->setup_meters (220, meter_width, 6);
334         }
335 }
336
337 void
338 MeterStrip::meter_configuration_changed (ChanCount c)
339 {
340         int type = 0;
341         _types.clear ();
342         bool old_has_midi = _has_midi;
343
344         for (DataType::iterator i = DataType::begin(); i != DataType::end(); ++i) {
345                 if (c.get (*i) > 0) {
346                         _types.push_back (*i);
347                         type |= 1 << (*i);
348                 }
349         }
350
351         if (boost::dynamic_pointer_cast<AudioTrack>(_route) == 0
352                         && boost::dynamic_pointer_cast<MidiTrack>(_route) == 0
353                         ) {
354                 meter_ticks1_area.set_name ("AudioBusMetricsLeft");
355                 meter_ticks2_area.set_name ("AudioBusMetricsRight");
356                 _has_midi = false;
357         }
358         else if (type == (1 << DataType::AUDIO)) {
359                 meter_ticks1_area.set_name ("AudioTrackMetricsLeft");
360                 meter_ticks2_area.set_name ("AudioTrackMetricsRight");
361                 _has_midi = false;
362         }
363         else if (type == (1 << DataType::MIDI)) {
364                 meter_ticks1_area.set_name ("MidiTrackMetricsLeft");
365                 meter_ticks2_area.set_name ("MidiTrackMetricsRight");
366                 _has_midi = true;
367         } else {
368                 meter_ticks1_area.set_name ("AudioMidiTrackMetricsLeft");
369                 meter_ticks2_area.set_name ("AudioMidiTrackMetricsRight");
370                 _has_midi = true;
371         }
372
373         if (old_has_midi != _has_midi) MetricChanged();
374         on_theme_changed();
375 }
376
377 void
378 MeterStrip::on_size_request (Gtk::Requisition* r)
379 {
380         meter_clear_pattern_cache();
381         VBox::on_size_request(r);
382 }
383
384 void
385 MeterStrip::on_size_allocate (Gtk::Allocation& a)
386 {
387         meter_clear_pattern_cache();
388         const int wh = a.get_height();
389         int nh = ceilf(wh * .11f);
390         if (nh < 52) nh = 52;
391         if (nh > 148) nh = 148;
392         namebx.set_size_request(18, nh);
393         if (_route) {
394                 name_label.set_size_request(18, nh-2);
395                 name_label.layout()->set_width((nh-4) * PANGO_SCALE);
396         }
397         VBox::on_size_allocate(a);
398 }
399
400 gint
401 MeterStrip::meter_metrics_expose (GdkEventExpose *ev)
402 {
403         return meter_expose_metrics(ev, _types, &meter_metric_area);
404 }
405
406 void
407 MeterStrip::set_metric_mode (int metricmode)
408 {
409         _types.clear ();
410         switch(metricmode) {
411                 case 0:
412                         meter_metric_area.set_name ("MidiTrackMetricsLeft");
413                         _types.push_back (DataType::MIDI);
414                         break;
415                 case 1:
416                         meter_metric_area.set_name ("AudioTrackMetricsLeft");
417                         _types.push_back (DataType::AUDIO);
418                         break;
419                 case 2:
420                         meter_metric_area.set_name ("MidiTrackMetricsRight");
421                         _types.push_back (DataType::MIDI);
422                         break;
423                 case 3:
424                 default:
425                         meter_metric_area.set_name ("AudioTrackMetricsRight");
426                         _types.push_back (DataType::AUDIO);
427                         break;
428         }
429
430         meter_metric_area.queue_draw ();
431 }
432
433 void
434 MeterStrip::set_pos (int pos)
435 {
436 }
437
438 gint
439 MeterStrip::meter_ticks1_expose (GdkEventExpose *ev)
440 {
441         return meter_expose_ticks(ev, _types, &meter_ticks1_area);
442 }
443
444 gint
445 MeterStrip::meter_ticks2_expose (GdkEventExpose *ev)
446 {
447         return meter_expose_ticks(ev, _types, &meter_ticks2_area);
448 }
449
450 void
451 MeterStrip::reset_route_peak_display (Route* route)
452 {
453         if (_route && _route.get() == route) {
454                 reset_peak_display ();
455         }
456 }
457
458 void
459 MeterStrip::reset_group_peak_display (RouteGroup* group)
460 {
461         if (_route && group == _route->route_group()) {
462                 reset_peak_display ();
463         }
464 }
465
466 void
467 MeterStrip::reset_peak_display ()
468 {
469         _route->shared_peak_meter()->reset_max();
470         level_meter->clear_meters();
471         max_peak = -INFINITY;
472         peak_display.set_name ("meterbridge peakindicator");
473         peak_display.set_elements((ArdourButton::Element) (ArdourButton::Edge|ArdourButton::Body));
474 }
475
476 bool
477 MeterStrip::peak_button_release (GdkEventButton* ev)
478 {
479         if (ev->button == 1 && Keyboard::modifier_state_equals (ev->state, Keyboard::PrimaryModifier|Keyboard::TertiaryModifier)) {
480                 ResetAllPeakDisplays ();
481         } else if (ev->button == 1 && Keyboard::modifier_state_equals (ev->state, Keyboard::PrimaryModifier)) {
482                 if (_route) {
483                         ResetGroupPeakDisplays (_route->route_group());
484                 }
485         } else {
486                 ResetRoutePeakDisplays (_route.get());
487         }
488         return true;
489 }
490
491 void
492 MeterStrip::redraw_metrics ()
493 {
494         meter_metric_area.queue_draw();
495         meter_ticks1_area.queue_draw();
496         meter_ticks2_area.queue_draw();
497 }
498
499 void
500 MeterStrip::update_button_box ()
501 {
502         if (!_session) return;
503         int height = 0;
504         if (_session->config.get_show_mute_on_meterbridge()) {
505                 height += 18;
506                 mutebox.show();
507         } else {
508                 mutebox.hide();
509         }
510         if (_session->config.get_show_solo_on_meterbridge()) {
511                 height += 18;
512                 solobox.show();
513         } else {
514                 solobox.hide();
515         }
516         if (_session->config.get_show_rec_on_meterbridge()) {
517                 height += 18;
518                 recbox.show();
519         } else {
520                 recbox.hide();
521         }
522         btnbox.set_size_request(16, height);
523         check_resize();
524 }
525
526 void
527 MeterStrip::update_name_box ()
528 {
529         if (!_session) return;
530         if (_session->config.get_show_name_on_meterbridge()) {
531                 namebx.show();
532         } else {
533                 namebx.hide();
534         }
535 }
536
537 void
538 MeterStrip::parameter_changed (std::string const & p)
539 {
540         if (p == "meter-peak") {
541                 max_peak = -INFINITY;
542         }
543         else if (p == "show-rec-on-meterbridge") {
544                 update_button_box();
545         }
546         else if (p == "show-mute-on-meterbridge") {
547                 update_button_box();
548         }
549         else if (p == "show-solo-on-meterbridge") {
550                 update_button_box();
551         }
552         else if (p == "show-name-on-meterbridge") {
553                 update_name_box();
554         }
555 }
556
557
558 bool
559 MeterStrip::level_meter_button_press (GdkEventButton* ev)
560 {
561         if (ev->button == 3) {
562                 popup_level_meter_menu (ev);
563                 return true;
564         }
565
566         return false;
567 }
568
569 void
570 MeterStrip::popup_level_meter_menu (GdkEventButton* ev)
571 {
572         using namespace Gtk::Menu_Helpers;
573
574         Gtk::Menu* m = manage (new Menu);
575         MenuList& items = m->items ();
576
577         RadioMenuItem::Group group;
578
579         _suspend_menu_callbacks = true;
580         add_level_meter_item (items, group, _("Peak"), MeterPeak);
581         add_level_meter_item (items, group, _("RMS + Peak"), MeterKrms);
582
583         items.push_back (SeparatorElem());
584         items.push_back (MenuElem (_("Change all in Group to Peak"), sigc::bind (SetMeterTypeMulti, -1, _route->route_group(), MeterPeak)));
585         items.push_back (MenuElem (_("Change all in Group to RMS + Peak"), sigc::bind (SetMeterTypeMulti, -1, _route->route_group(), MeterKrms)));
586         items.push_back (MenuElem (_("Change all to Peak"), sigc::bind (SetMeterTypeMulti, 0, _route->route_group(), MeterPeak)));
587         items.push_back (MenuElem (_("Change all to RMS + Peak"), sigc::bind (SetMeterTypeMulti, 0, _route->route_group(), MeterKrms)));
588         items.push_back (MenuElem (_("Change same track-type to Peak"), sigc::bind (SetMeterTypeMulti, _strip_type, _route->route_group(), MeterPeak)));
589         items.push_back (MenuElem (_("Change same track-type to RMS + Peak"), sigc::bind (SetMeterTypeMulti, _strip_type, _route->route_group(), MeterKrms)));
590
591         m->popup (ev->button, ev->time);
592         _suspend_menu_callbacks = false;
593 }
594
595 void
596 MeterStrip::add_level_meter_item (Menu_Helpers::MenuList& items, RadioMenuItem::Group& group, string const & name, MeterType type)
597 {
598         using namespace Menu_Helpers;
599
600         items.push_back (RadioMenuElem (group, name, sigc::bind (sigc::mem_fun (*this, &MeterStrip::set_meter_type), type)));
601         RadioMenuItem* i = dynamic_cast<RadioMenuItem *> (&items.back ());
602         i->set_active (_route->meter_type() == type);
603 }
604
605 void
606 MeterStrip::set_meter_type (MeterType type)
607 {
608         if (_suspend_menu_callbacks) return;
609         level_meter->set_type (type);
610 }
611
612 void
613 MeterStrip::meter_type_changed (MeterType type)
614 {
615         _route->set_meter_type(type);
616 }
617
618 void
619 MeterStrip::set_meter_type_multi (int what, RouteGroup* group, MeterType type)
620 {
621         switch (what) {
622                 case -1:
623                         if (_route && group == _route->route_group()) {
624                                 level_meter->set_type (type);
625                         }
626                         break;
627                 case 0:
628                         level_meter->set_type (type);
629                 default:
630                         if (what == _strip_type) {
631                                 level_meter->set_type (type);
632                         }
633                         break;
634         }
635 }