2 Copyright (C) 2013 Paul Davis
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.
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.
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.
22 #include <sigc++/bind.h>
24 #include "ardour/session.h"
25 #include "ardour/route.h"
26 #include "ardour/route_group.h"
27 #include "ardour/meter.h"
29 #include "ardour/audio_track.h"
30 #include "ardour/midi_track.h"
32 #include <gtkmm2ext/gtk_ui.h>
33 #include <gtkmm2ext/utils.h>
35 #include "ardour_ui.h"
36 #include "global_signals.h"
38 #include "gui_thread.h"
39 #include "ardour_window.h"
41 #include "meterbridge.h"
42 #include "meter_strip.h"
46 using namespace ARDOUR;
49 using namespace Gtkmm2ext;
52 PBD::Signal1<void,MeterStrip*> MeterStrip::CatchDeletion;
54 MeterStrip::MetricPatterns MeterStrip::metric_patterns;
55 MeterStrip::TickPatterns MeterStrip::ticks_patterns;
57 MeterStrip::MeterStrip (int metricmode)
63 peakbx.set_size_request(-1, 14);
64 btnbox.set_size_request(-1, 16);
69 meter_metric_area.set_name ("AudioBusMetrics");
70 _types.push_back (DataType::AUDIO);
73 meter_metric_area.set_name ("AudioTrackMetrics");
74 _types.push_back (DataType::AUDIO);
77 meter_metric_area.set_name ("MidiTrackMetrics");
78 _types.push_back (DataType::MIDI);
81 meter_metric_area.set_name ("AudioMidiTrackMetrics");
82 _types.push_back (DataType::AUDIO);
83 _types.push_back (DataType::MIDI);
86 //meter_metric_area.queue_draw ();
88 set_size_request_to_display_given_text (meter_metric_area, "8888", 1, 0);
89 meter_metric_area.signal_expose_event().connect (
90 sigc::mem_fun(*this, &MeterStrip::meter_metrics_expose));
92 meterbox.pack_start(meter_metric_area, true, false);
95 label.set_name("MeterbridgeLabel");
96 label.set_angle(90.0);
97 label.set_alignment(0.5, 1.0);
98 label.set_size_request(12, 52);
100 pack_start (peakbx, false, false);
101 pack_start (meterbox, true, true);
102 pack_start (btnbox, false, false);
103 pack_start (label, false, false, 2);
108 meter_metric_area.show();
111 UI::instance()->theme_changed.connect (sigc::mem_fun(*this, &MeterStrip::on_theme_changed));
112 ColorsChanged.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
113 DPIReset.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
116 MeterStrip::MeterStrip (Session* sess, boost::shared_ptr<ARDOUR::Route> rt)
122 RouteUI::set_route (rt);
125 if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
129 // level meter + ticks
130 level_meter = new LevelMeter(sess);
131 level_meter->set_meter (_route->shared_peak_meter().get());
132 level_meter->clear_meters();
133 level_meter->setup_meters (400, meter_width, 6);
135 Gtk::Alignment *meter_align = Gtk::manage (new Gtk::Alignment());
136 meter_align->set(0.5, 0.5, 0.0, 1.0);
137 meter_align->add(*level_meter);
139 meterbox.pack_start(meter_ticks1_area, true, false);
140 meterbox.pack_start(*meter_align, true, true);
141 meterbox.pack_start(meter_ticks2_area, true, false);
144 peak_display.set_name ("MixerStripPeakDisplay");
145 max_peak = minus_infinity();
146 peak_display.unset_flags (Gtk::CAN_FOCUS);
147 peak_display.set_size_request(-1, 8);
149 Gtk::Alignment *peak_align = Gtk::manage (new Gtk::Alignment());
150 peak_align->set(0.5, 1.0, 0.9, 0.8);
151 peak_align->add(peak_display);
152 peakbx.pack_start(*peak_align, true, true);
153 peakbx.set_size_request(-1, 14);
155 // add track-name label -- TODO ellipsize
156 label.set_text(_route->name().c_str());
157 label.set_name("MeterbridgeLabel");
158 label.set_angle(90.0);
159 label.set_alignment(0.5, 1.0);
160 label.set_size_request(12, 52);
163 btnbox.pack_start(*rec_enable_button, true, false);
164 btnbox.set_size_request(-1, 16);
166 pack_start (peakbx, false, false);
167 pack_start (meterbox, true, true);
168 pack_start (btnbox, false, false);
169 pack_start (label, false, false, 2);
173 meter_ticks1_area.show();
174 meter_ticks2_area.show();
182 _route->shared_peak_meter()->ConfigurationChanged.connect (
183 route_connections, invalidator (*this), boost::bind (&MeterStrip::meter_configuration_changed, this, _1), gui_context()
185 meter_configuration_changed (_route->shared_peak_meter()->input_streams ());
187 meter_ticks1_area.set_size_request(3,-1);
188 meter_ticks2_area.set_size_request(3,-1);
189 meter_ticks1_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks1_expose));
190 meter_ticks2_area.signal_expose_event().connect (sigc::mem_fun(*this, &MeterStrip::meter_ticks2_expose));
192 _route->DropReferences.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::self_delete, this), gui_context());
193 _route->PropertyChanged.connect (route_connections, invalidator (*this), boost::bind (&MeterStrip::strip_property_changed, this, _1), gui_context());
195 peak_display.signal_button_release_event().connect (sigc::mem_fun(*this, &MeterStrip::peak_button_release), false);
197 UI::instance()->theme_changed.connect (sigc::mem_fun(*this, &MeterStrip::on_theme_changed));
198 ColorsChanged.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
199 DPIReset.connect (sigc::mem_fun (*this, &MeterStrip::on_theme_changed));
202 MeterStrip::~MeterStrip ()
205 CatchDeletion (this);
209 MeterStrip::self_delete ()
215 MeterStrip::update_rec_display ()
217 RouteUI::update_rec_display ();
221 MeterStrip::state_id() const
223 return string_compose ("mtrs %1", _route->id().to_s());
227 MeterStrip::set_button_names()
229 rec_enable_button->set_text (_("R"));
233 MeterStrip::strip_property_changed (const PropertyChange& what_changed)
235 if (!what_changed.contains (ARDOUR::Properties::name)) {
238 ENSURE_GUI_THREAD (*this, &MeterStrip::strip_name_changed, what_changed)
239 label.set_text(_route->name());
243 MeterStrip::fast_update ()
245 float mpeak = level_meter->update_meters();
246 if (mpeak > max_peak) {
249 peak_display.set_name ("MixerStripPeakDisplayPeak");
255 MeterStrip::display_metrics (bool show)
259 meter_metric_area.show();
261 meter_metric_area.hide();
267 MeterStrip::on_theme_changed()
269 metric_patterns.clear();
270 ticks_patterns.clear();
272 if (level_meter && _route) {
274 if (_route->shared_peak_meter()->input_streams().n_total() == 1) {
277 level_meter->setup_meters (400, meter_width, 6);
282 MeterStrip::meter_configuration_changed (ChanCount c)
287 for (DataType::iterator i = DataType::begin(); i != DataType::end(); ++i) {
288 if (c.get (*i) > 0) {
289 _types.push_back (*i);
294 // TODO draw Inactive routes or busses with different styles
295 if (boost::dynamic_pointer_cast<AudioTrack>(_route) == 0
296 && boost::dynamic_pointer_cast<MidiTrack>(_route) == 0
298 meter_ticks1_area.set_name ("AudioBusMetrics");
299 meter_ticks2_area.set_name ("AudioBusMetrics");
301 else if (type == (1 << DataType::AUDIO)) {
302 meter_ticks1_area.set_name ("AudioTrackMetrics");
303 meter_ticks2_area.set_name ("AudioTrackMetrics");
305 else if (type == (1 << DataType::MIDI)) {
306 meter_ticks1_area.set_name ("MidiTrackMetrics");
307 meter_ticks2_area.set_name ("MidiTrackMetrics");
309 meter_ticks1_area.set_name ("AudioMidiTrackMetrics");
310 meter_ticks2_area.set_name ("AudioMidiTrackMetrics");
312 meter_ticks1_area.queue_draw();
313 meter_ticks2_area.queue_draw();
314 metric_patterns.clear();
315 ticks_patterns.clear();
319 MeterStrip::on_size_request (Gtk::Requisition* r)
321 metric_patterns.clear();
322 ticks_patterns.clear();
323 VBox::on_size_request(r);
327 MeterStrip::on_size_allocate (Gtk::Allocation& a)
329 metric_patterns.clear();
330 ticks_patterns.clear();
331 VBox::on_size_allocate(a);
334 /* XXX code-copy from gain_meter.cc -- TODO consolidate
336 * slightly different:
337 * - ticks & label positions are swapped
338 * - more ticks for audio (longer meter by default)
339 * - right-aligned lables, unit-legend
340 * - height limitation of FastMeter::max_pattern_metric_size is included here
343 MeterStrip::render_metrics (Gtk::Widget& w, vector<DataType> types)
345 Glib::RefPtr<Gdk::Window> win (w.get_window());
348 win->get_size (width, height);
350 cairo_surface_t* surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, width, height);
351 cairo_t* cr = cairo_create (surface);
352 Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create(w.get_pango_context());
355 Pango::AttrList audio_font_attributes;
356 Pango::AttrList midi_font_attributes;
357 Pango::AttrList unit_font_attributes;
359 Pango::AttrFontDesc* font_attr;
360 Pango::FontDescription font;
362 font = Pango::FontDescription (""); // use defaults
363 //font = get_font_for_style("gain-fader");
364 //font = w.get_style()->get_font();
366 font.set_weight (Pango::WEIGHT_NORMAL);
367 font.set_size (9.0 * PANGO_SCALE);
368 font_attr = new Pango::AttrFontDesc (Pango::Attribute::create_attr_font_desc (font));
369 audio_font_attributes.change (*font_attr);
372 font.set_weight (Pango::WEIGHT_ULTRALIGHT);
373 font.set_stretch (Pango::STRETCH_ULTRA_CONDENSED);
374 font.set_size (7.5 * PANGO_SCALE);
375 font_attr = new Pango::AttrFontDesc (Pango::Attribute::create_attr_font_desc (font));
376 midi_font_attributes.change (*font_attr);
379 font.set_size (7.0 * PANGO_SCALE);
380 font_attr = new Pango::AttrFontDesc (Pango::Attribute::create_attr_font_desc (font));
381 unit_font_attributes.change (*font_attr);
384 cairo_move_to (cr, 0, 0);
385 cairo_rectangle (cr, 0, 0, width, height);
387 Gdk::Color c = w.get_style()->get_bg (Gtk::STATE_NORMAL);
388 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
392 height = min(1024, height); // XXX see FastMeter::max_pattern_metric_size
394 for (vector<DataType>::const_iterator i = types.begin(); i != types.end(); ++i) {
398 if (types.size() > 1) {
399 /* we're overlaying more than 1 set of marks, so use different colours */
402 case DataType::AUDIO:
403 c = w.get_style()->get_fg (Gtk::STATE_NORMAL);
404 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
407 c = w.get_style()->get_fg (Gtk::STATE_ACTIVE);
408 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
412 c = w.get_style()->get_fg (Gtk::STATE_NORMAL);
413 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
419 case DataType::AUDIO:
420 layout->set_attributes (audio_font_attributes);
421 points.push_back (-50);
422 points.push_back (-40);
423 points.push_back (-30);
424 points.push_back (-25);
425 points.push_back (-20);
426 points.push_back (-18);
427 points.push_back (-15);
428 points.push_back (-10);
429 points.push_back (-5);
430 points.push_back (-3);
431 points.push_back (0);
432 points.push_back (3);
436 layout->set_attributes (midi_font_attributes);
437 points.push_back (0);
438 if (types.size() == 1) {
439 points.push_back (32);
441 /* tweak so as not to overlay the -30dB mark */
442 points.push_back (48);
444 if (types.size() == 1) {
445 points.push_back (64); // very close to -18dB
446 points.push_back (96); // overlays with -6dB mark
448 points.push_back (72);
449 points.push_back (88);
451 points.push_back (127);
458 for (vector<int>::const_iterator j = points.begin(); j != points.end(); ++j) {
462 case DataType::AUDIO:
463 fraction = log_meter (*j);
464 snprintf (buf, sizeof (buf), "%+2d", *j);
467 fraction = *j / 127.0;
468 snprintf (buf, sizeof (buf), "%3d", *j);
469 pos = height - (gint) rintf (height * fraction);
471 cairo_arc(cr, 2, pos, 1.0, 0, 2 * M_PI);
477 pos = height - (gint) rintf (height * fraction);
478 layout->set_text(buf);
480 /* we want logical extents, not ink extents here */
483 layout->get_pixel_size(tw, th);
485 int p = pos - (th / 2);
486 p = min (p, height - th);
489 cairo_move_to (cr, width-1-tw, p);
490 pango_cairo_show_layout (cr, layout->gobj());
494 if (types.size() == 1) {
496 layout->set_attributes (unit_font_attributes);
497 switch (types.at(0)) {
498 case DataType::AUDIO:
499 layout->set_text("dBFS");
500 layout->get_pixel_size(tw, th);
503 layout->set_text("vel");
504 layout->get_pixel_size(tw, th);
507 Gdk::Color c = w.get_style()->get_fg (Gtk::STATE_ACTIVE);
508 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
509 cairo_move_to (cr, 1, height - th);
510 pango_cairo_show_layout (cr, layout->gobj());
513 cairo_pattern_t* pattern = cairo_pattern_create_for_surface (surface);
514 MetricPatterns::iterator p;
516 if ((p = metric_patterns.find (w.get_name())) != metric_patterns.end()) {
517 cairo_pattern_destroy (p->second);
520 metric_patterns[w.get_name()] = pattern;
523 cairo_surface_destroy (surface);
528 /* XXX code-copy from gain_meter.cc -- TODO consolidate */
530 MeterStrip::meter_metrics_expose (GdkEventExpose *ev)
532 Glib::RefPtr<Gdk::Window> win (meter_metric_area.get_window());
535 cr = gdk_cairo_create (win->gobj());
537 /* clip to expose area */
539 gdk_cairo_rectangle (cr, &ev->area);
542 cairo_pattern_t* pattern;
543 MetricPatterns::iterator i = metric_patterns.find (meter_metric_area.get_name());
545 if (i == metric_patterns.end()) {
546 pattern = render_metrics (meter_metric_area, _types);
551 cairo_move_to (cr, 0, 0);
552 cairo_set_source (cr, pattern);
555 win->get_size (width, height);
557 cairo_rectangle (cr, 0, 0, width, height);
566 MeterStrip::render_ticks (Gtk::Widget& w, vector<DataType> types)
568 Glib::RefPtr<Gdk::Window> win (w.get_window());
571 win->get_size (width, height);
573 cairo_surface_t* surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, width, height);
574 cairo_t* cr = cairo_create (surface);
576 cairo_move_to (cr, 0, 0);
577 cairo_rectangle (cr, 0, 0, width, height);
579 Gdk::Color c = w.get_style()->get_bg (Gtk::STATE_NORMAL);
580 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
584 height = min(1024, height); // XXX see FastMeter::max_pattern_metric_size
586 for (vector<DataType>::const_iterator i = types.begin(); i != types.end(); ++i) {
590 if (types.size() > 1) {
591 /* we're overlaying more than 1 set of marks, so use different colours */
594 case DataType::AUDIO:
595 c = w.get_style()->get_fg (Gtk::STATE_NORMAL);
596 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
599 c = w.get_style()->get_fg (Gtk::STATE_ACTIVE);
600 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
604 c = w.get_style()->get_fg (Gtk::STATE_NORMAL);
605 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
608 std::map<int,float> points;
611 case DataType::AUDIO:
612 points.insert (std::pair<int,float>(-60, 0.5));
613 points.insert (std::pair<int,float>(-50, 0.5));
614 points.insert (std::pair<int,float>(-40, 0.5));
615 points.insert (std::pair<int,float>(-30, 0.5));
616 points.insert (std::pair<int,float>(-25, 0.5));
617 points.insert (std::pair<int,float>(-20, 1.0));
619 points.insert (std::pair<int,float>(-19, 0.5));
620 points.insert (std::pair<int,float>(-18, 1.0));
621 points.insert (std::pair<int,float>(-17, 0.5));
622 points.insert (std::pair<int,float>(-16, 0.5));
623 points.insert (std::pair<int,float>(-15, 1.0));
625 points.insert (std::pair<int,float>(-14, 0.5));
626 points.insert (std::pair<int,float>(-13, 0.5));
627 points.insert (std::pair<int,float>(-12, 0.5));
628 points.insert (std::pair<int,float>(-11, 0.5));
629 points.insert (std::pair<int,float>(-10, 1.0));
631 points.insert (std::pair<int,float>( -9, 0.5));
632 points.insert (std::pair<int,float>( -8, 0.5));
633 points.insert (std::pair<int,float>( -7, 0.5));
634 points.insert (std::pair<int,float>( -6, 0.5));
635 points.insert (std::pair<int,float>( -5, 1.0));
636 points.insert (std::pair<int,float>( -4, 0.5));
637 points.insert (std::pair<int,float>( -3, 0.5));
638 points.insert (std::pair<int,float>( -2, 0.5));
639 points.insert (std::pair<int,float>( -1, 0.5));
641 points.insert (std::pair<int,float>( 0, 1.0));
642 points.insert (std::pair<int,float>( 1, 0.5));
643 points.insert (std::pair<int,float>( 2, 0.5));
644 points.insert (std::pair<int,float>( 3, 0.5));
645 points.insert (std::pair<int,float>( 4, 0.5));
646 points.insert (std::pair<int,float>( 5, 0.5));
647 points.insert (std::pair<int,float>( 6, 0.5));
651 points.insert (std::pair<int,float>( 0, 1.0));
652 points.insert (std::pair<int,float>( 16, 0.5));
653 points.insert (std::pair<int,float>( 32, 0.5));
654 points.insert (std::pair<int,float>( 48, 0.5));
655 points.insert (std::pair<int,float>( 64, 1.0));
656 points.insert (std::pair<int,float>( 72, 0.5));
657 points.insert (std::pair<int,float>( 96, 0.5));
658 points.insert (std::pair<int,float>(100, 1.0));
659 points.insert (std::pair<int,float>(112, 0.5));
660 points.insert (std::pair<int,float>(127, 1.0));
664 for (std::map<int,float>::const_iterator j = points.begin(); j != points.end(); ++j) {
665 cairo_set_line_width (cr, (j->second));
671 case DataType::AUDIO:
673 cairo_set_source_rgb (cr, 1.0, 0.0, 0.0);
675 fraction = log_meter (j->first);
676 pos = height - (gint) floor (height * fraction);
677 cairo_move_to(cr, 0, pos + .5);
678 cairo_line_to(cr, 3, pos + .5);
682 fraction = (j->first) / 127.0;
683 pos = height - (gint) floor (height * fraction);
684 cairo_arc(cr, 1.5, pos, (j->second), 0, 2 * M_PI);
685 cairo_fill_preserve(cr);
692 cairo_pattern_t* pattern = cairo_pattern_create_for_surface (surface);
693 TickPatterns::iterator p;
695 if ((p = ticks_patterns.find (w.get_name())) != metric_patterns.end()) {
696 cairo_pattern_destroy (p->second);
699 ticks_patterns[w.get_name()] = pattern;
702 cairo_surface_destroy (surface);
708 MeterStrip::meter_ticks1_expose (GdkEventExpose *ev)
710 return meter_ticks_expose(ev, &meter_ticks1_area);
714 MeterStrip::meter_ticks2_expose (GdkEventExpose *ev)
716 return meter_ticks_expose(ev, &meter_ticks2_area);
720 MeterStrip::meter_ticks_expose (GdkEventExpose *ev, Gtk::DrawingArea *mta)
722 Glib::RefPtr<Gdk::Window> win (mta->get_window());
725 cr = gdk_cairo_create (win->gobj());
727 /* clip to expose area */
729 gdk_cairo_rectangle (cr, &ev->area);
732 cairo_pattern_t* pattern;
733 TickPatterns::iterator i = ticks_patterns.find (mta->get_name());
735 if (i == ticks_patterns.end()) {
736 pattern = render_ticks (*mta, _types);
741 cairo_move_to (cr, 0, 0);
742 cairo_set_source (cr, pattern);
745 win->get_size (width, height);
747 cairo_rectangle (cr, 0, 0, width, height);
756 MeterStrip::reset_group_peak_display (RouteGroup* group)
758 /* UNUSED -- need connection w/mixer || other meters */
759 if (_route && group == _route->route_group()) {
760 reset_peak_display ();
765 MeterStrip::reset_peak_display ()
767 _route->shared_peak_meter()->reset_max();
768 level_meter->clear_meters();
769 max_peak = -INFINITY;
770 peak_display.set_name ("MixerStripPeakDisplay");
774 MeterStrip::peak_button_release (GdkEventButton* ev)
776 reset_peak_display ();