/* * Copyright (C) 2016-2017 Paul Davis * Copyright (C) 2017-2019 Robin Gareus * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include "pbd/string_convert.h" #include "ardour/mute_control.h" #include "ardour/profile.h" #include "ardour/session.h" #include "ardour/solo_control.h" #include "ardour/vca.h" #include "gtkmm2ext/doi.h" #include "gtkmm2ext/utils.h" #include "widgets/tooltips.h" #include "gui_thread.h" #include "public_editor.h" #include "mixer_ui.h" #include "ui_config.h" #include "vca_time_axis.h" #include "pbd/i18n.h" using namespace ARDOUR; using namespace ArdourWidgets; using namespace Gtk; using namespace Gtkmm2ext; using namespace PBD; VCATimeAxisView::VCATimeAxisView (PublicEditor& ed, Session* s, ArdourCanvas::Canvas& canvas) : SessionHandlePtr (s) , StripableTimeAxisView (ed, s, canvas) , gain_meter (s, true, 75, 14) // XXX stupid magic numbers, match sizes in RouteTimeAxisView , automation_action_menu (0) { controls_base_selected_name = X_("ControlMasterBaseSelected"); controls_base_unselected_name = X_("ControlMasterBaseUnselected"); solo_button.set_name ("solo button"); set_tooltip (solo_button, _("Solo assigned channels")); solo_button.signal_button_release_event().connect (sigc::mem_fun (*this, &VCATimeAxisView::solo_release), false); solo_button.set_can_focus (false); mute_button.set_name ("mute button"); mute_button.set_text (S_("Mute|M")); set_tooltip (mute_button, _("Mute assigned channels")); mute_button.signal_button_release_event().connect (sigc::mem_fun (*this, &VCATimeAxisView::mute_release), false); mute_button.set_can_focus (false); drop_button.set_name ("mute button"); drop_button.set_text (S_("VCA|D")); set_tooltip (drop_button, _("Unassign all channels")); drop_button.signal_button_release_event().connect (sigc::mem_fun (*this, &VCATimeAxisView::drop_release), false); drop_button.set_can_focus (false); automation_button.set_name ("route button"); automation_button.set_text (S_("RTAV|A")); set_tooltip (automation_button, _("Automation")); automation_button.signal_button_press_event().connect (sigc::mem_fun (*this, &VCATimeAxisView::automation_click), false); automation_button.set_can_focus (false); mute_button.set_tweaks(ArdourButton::TrackHeader); solo_button.set_tweaks(ArdourButton::TrackHeader); drop_button.set_tweaks(ArdourButton::TrackHeader); automation_button.set_tweaks(ArdourButton::TrackHeader); if (ARDOUR::Profile->get_mixbus()) { controls_button_size_group->add_widget(mute_button); Gtk::Fixed *blank = manage(new Gtk::Fixed()); controls_button_size_group->add_widget(*blank); controls_table.attach (*blank, 0, 1, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0); blank->show(); controls_table.attach (mute_button, 1, 2, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (solo_button, 2, 3, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (automation_button, 1, 2, 2, 3, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (drop_button, 2, 3, 2, 3, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (gain_meter.get_gain_slider(), 3, 5, 2, 3, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::EXPAND, 1, 0); } else { controls_table.attach (mute_button, 2, 3, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (solo_button, 3, 4, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (automation_button, 2, 3, 1, 2, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (drop_button, 3, 4, 1, 2, Gtk::SHRINK, Gtk::SHRINK, 0, 0); controls_table.attach (gain_meter.get_gain_slider(), 0, 2, 1, 2, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::EXPAND, 1, 0); } mute_button.show (); solo_button.show (); drop_button.show (); automation_button.show (); gain_meter.get_gain_slider().show (); controls_ebox.set_name (controls_base_unselected_name); time_axis_frame.set_name (controls_base_unselected_name); s->MonitorBusAddedOrRemoved.connect (*this, invalidator (*this), std::bind (&VCATimeAxisView::set_button_names, this), gui_context()); s->config.ParameterChanged.connect (*this, invalidator (*this), std::bind (&VCATimeAxisView::parameter_changed, this, _1), gui_context()); Config->ParameterChanged.connect (*this, invalidator (*this), std::bind (&VCATimeAxisView::parameter_changed, this, _1), gui_context()); UIConfiguration::instance().ParameterChanged.connect (sigc::mem_fun (*this, &VCATimeAxisView::parameter_changed)); } VCATimeAxisView::~VCATimeAxisView () { delete automation_action_menu; CatchDeletion (this); } void VCATimeAxisView::self_delete () { /* reset reference immediately rather than deferring to idle */ _vca.reset (); delete_when_idle (this); } void VCATimeAxisView::parameter_changed (std::string const & p) { if (p == "track-name-number") { update_track_number_visibility(); } else if (p == "solo-control-is-listen-control" || p == "listen-position") { set_button_names (); } } bool VCATimeAxisView::solo_release (GdkEventButton*) { /* We use NoGroup because VCA controls are never part of a group. This is redundant, but clear. */ _vca->solo_control()->set_value (_vca->solo_control()->self_soloed() ? 0.0 : 1.0, Controllable::NoGroup); return true; } bool VCATimeAxisView::mute_release (GdkEventButton*) { /* We use NoGroup because VCA controls are never part of a group. This is redundant, but clear. */ _vca->mute_control()->set_value (_vca->mute_control()->muted_by_self() ? 0.0 : 1.0, Controllable::NoGroup); return true; } void VCATimeAxisView::set_vca (std::shared_ptr v) { StripableTimeAxisView::set_stripable (v); _vca = v; gain_meter.set_controls (v, std::shared_ptr(), std::shared_ptr(), _vca->gain_control()); // Mixer_UI::instance()->show_vca_change.connect (sigc::mem_fun (*this, &VCAMasterStrip::spill_change)); _vca->PropertyChanged.connect (vca_connections, invalidator (*this), std::bind (&VCATimeAxisView::vca_property_changed, this, _1), gui_context()); _vca->solo_control()->Changed.connect (vca_connections, invalidator (*this), std::bind (&VCATimeAxisView::update_solo_display, this), gui_context()); _vca->mute_control()->Changed.connect (vca_connections, invalidator (*this), std::bind (&VCATimeAxisView::update_mute_display, this), gui_context()); _vca->DropReferences.connect (vca_connections, invalidator (*this), std::bind (&VCATimeAxisView::self_delete, this), gui_context()); solo_button.set_controllable (_vca->solo_control()); mute_button.set_controllable (_vca->mute_control()); /* VCA number never changes */ number_label.set_text (PBD::to_string (_vca->number())); set_height (preset_height (HeightNormal)); if (automation_child (GainAutomation) == 0) { create_automation_child (GainAutomation, false); } if (automation_child (MuteAutomation) == 0) { create_automation_child (MuteAutomation, false); } update_vca_name (); set_button_names (); update_solo_display (); update_mute_display (); update_track_number_visibility (); } void VCATimeAxisView::vca_property_changed (PropertyChange const & what_changed) { if (what_changed.contains (ARDOUR::Properties::name)) { update_vca_name (); } } void VCATimeAxisView::update_vca_name () { name_label.set_text (_vca->full_name()); } bool VCATimeAxisView::name_entry_changed (std::string const& str) { _vca->set_name (str); return true; } void VCATimeAxisView::update_mute_display () { if (_vca->mute_control()->muted_by_self()) { mute_button.set_active_state (ExplicitActive); } else if (_vca->mute_control()->muted_by_masters ()) { mute_button.set_active_state (ImplicitActive); } else { mute_button.set_active_state (Gtkmm2ext::Off); } } void VCATimeAxisView::update_solo_display () { if (_vca->solo_control()->self_soloed()) { solo_button.set_active_state (ExplicitActive); } else if (_vca->solo_control()->soloed_by_masters ()) { solo_button.set_active_state (ImplicitActive); } else { solo_button.set_active_state (Gtkmm2ext::Off); } update_mute_display (); } std::string VCATimeAxisView::name() const { return _vca->name(); } std::string VCATimeAxisView::state_id() const { return string_compose ("vtv %1", _vca->id().to_s()); } void VCATimeAxisView::set_button_names () { if (Config->get_solo_control_is_listen_control()) { switch (Config->get_listen_position()) { case AfterFaderListen: solo_button.set_text (S_("AfterFader|A")); set_tooltip (solo_button, _("After-fade listen (AFL)")); break; case PreFaderListen: solo_button.set_text (S_("PreFader|P")); set_tooltip (solo_button, _("Pre-fade listen (PFL)")); break; } } else { solo_button.set_text (S_("Solo|S")); set_tooltip (solo_button, _("Solo")); } } void VCATimeAxisView::update_track_number_visibility () { DisplaySuspender ds; bool show_label = _session->config.get_track_name_number(); if (number_label.get_parent()) { controls_table.remove (number_label); } if (show_label) { if (ARDOUR::Profile->get_mixbus()) { controls_table.attach (number_label, 3, 4, 0, 1, Gtk::SHRINK, Gtk::EXPAND|Gtk::FILL, 1, 0); } else { controls_table.attach (number_label, 0, 1, 0, 1, Gtk::SHRINK, Gtk::EXPAND|Gtk::FILL, 1, 0); } // see ArdourButton::on_size_request(), we should probably use a global size-group here instead. // except the width of the number label is subtracted from the name-hbox, so we // need to explicitly calculate it anyway until the name-label & entry become ArdourWidgets. int tnw = (2 + std::max(2u, _session->track_number_decimals())) * number_label.char_pixel_width(); if (tnw & 1) --tnw; number_label.set_size_request(tnw, -1); number_label.show (); } else { number_label.hide (); } } bool VCATimeAxisView::automation_click (GdkEventButton* ev) { if (ev->button != 1) { return true; } conditionally_add_to_selection (); build_automation_action_menu (false); Gtkmm2ext::anchored_menu_popup (automation_action_menu, &automation_button, "", 1, ev->time); return true; } bool VCATimeAxisView::drop_release (GdkEventButton*) { _vca->Drop (); /* EMIT SIGNAL */ return true; } PresentationInfo const & VCATimeAxisView::presentation_info () const { return _vca->presentation_info(); } std::shared_ptr VCATimeAxisView::stripable () const { return _vca; } Gdk::Color VCATimeAxisView::color () const { return Gtkmm2ext::gdk_color_from_rgb (_vca->presentation_info().color ()); } void VCATimeAxisView::set_height (uint32_t h, TrackHeightMode m, bool from_idle) { TimeAxisView::set_height (h, m, from_idle); if (height >= preset_height (HeightNormal)) { drop_button.show (); automation_button.show (); gain_meter.get_gain_slider().show (); } else { drop_button.hide (); automation_button.hide (); gain_meter.get_gain_slider().hide (); } set_gui_property ("height", h); _vca->gui_changed ("track_height", (void*) 0); /* EMIT SIGNAL */ } bool VCATimeAxisView::marked_for_display () const { return _vca && !_vca->presentation_info().hidden(); } bool VCATimeAxisView::set_marked_for_display (bool yn) { if (_vca && (yn == _vca->presentation_info().hidden())) { _vca->presentation_info().set_hidden (!yn); return true; // things changed } return false; } void VCATimeAxisView::create_gain_automation_child (const Evoral::Parameter& param, bool show) { std::shared_ptr c = _vca->gain_control(); if (!c) { error << "VCA has no gain automation, unable to add automation track view." << endmsg; return; } gain_track.reset (new AutomationTimeAxisView (_session, _vca, std::shared_ptr (), c, param, _editor, *this, false, parent_canvas, /*_route->amp()->describe_parameter(param)*/"Fader")); add_automation_child (Evoral::Parameter(GainAutomation), gain_track, show); } void VCATimeAxisView::create_mute_automation_child (const Evoral::Parameter& param, bool show) { std::shared_ptr c = _vca->mute_control(); if (!c) { error << "VCA has no mute automation, unable to add automation track view." << endmsg; return; } mute_track.reset (new AutomationTimeAxisView (_session, _vca, std::shared_ptr (), c, param, _editor, *this, false, parent_canvas, /*_route->describe_parameter(param)*/ "Mute")); add_automation_child (Evoral::Parameter(MuteAutomation), mute_track, show); } void VCATimeAxisView::create_automation_child (const Evoral::Parameter& param, bool show) { switch (param.type()) { case GainAutomation: create_gain_automation_child (param, show); break; case MuteAutomation: create_mute_automation_child (param, show); break; default: break; } } void VCATimeAxisView::build_display_menu () { using namespace Menu_Helpers; /* prepare it */ TimeAxisView::build_display_menu (); MenuList& items = display_menu->items(); items.push_back (MenuElem (_("Color..."), sigc::mem_fun (*this, &VCATimeAxisView::choose_color))); build_size_menu (); items.push_back (MenuElem (_("Height"), *_size_menu)); items.push_back (SeparatorElem()); build_automation_action_menu (true); items.push_back (MenuElem (_("Automation"), *automation_action_menu)); items.push_back (SeparatorElem()); items.push_back (MenuElem (_("Drop All Assigned Channels"), sigc::mem_fun (*this, &VCATimeAxisView::drop_all_slaves))); items.push_back (SeparatorElem()); items.push_back (MenuElem (_("Remove"), sigc::mem_fun(_editor, &PublicEditor::remove_tracks))); } void VCATimeAxisView::build_automation_action_menu (bool for_selection) { using namespace Menu_Helpers; _main_automation_menu_map.clear (); delete automation_action_menu; automation_action_menu = new Menu; MenuList& items = automation_action_menu->items(); automation_action_menu->set_name ("ArdourContextMenu"); items.push_back (MenuElem (_("Show All Automation"), sigc::bind (sigc::mem_fun (*this, &VCATimeAxisView::show_all_automation), for_selection))); items.push_back (MenuElem (_("Show Existing Automation"), sigc::bind (sigc::mem_fun (*this, &VCATimeAxisView::show_existing_automation), for_selection))); items.push_back (MenuElem (_("Hide All Automation"), sigc::bind (sigc::mem_fun (*this, &VCATimeAxisView::hide_all_automation), for_selection))); if (gain_track) { items.push_back (CheckMenuElem (_("Fader"), sigc::mem_fun (*this, &VCATimeAxisView::update_gain_track_visibility))); gain_automation_item = dynamic_cast (&items.back ()); gain_automation_item->set_active (string_to(gain_track->gui_property ("visible"))); _main_automation_menu_map[Evoral::Parameter(GainAutomation)] = gain_automation_item; } if (trim_track) { items.push_back (CheckMenuElem (_("Trim"), sigc::mem_fun (*this, &VCATimeAxisView::update_trim_track_visibility))); trim_automation_item = dynamic_cast (&items.back ()); trim_automation_item->set_active (string_to(trim_track->gui_property ("visible"))); _main_automation_menu_map[Evoral::Parameter(TrimAutomation)] = trim_automation_item; } if (mute_track) { items.push_back (CheckMenuElem (_("Mute"), sigc::mem_fun (*this, &VCATimeAxisView::update_mute_track_visibility))); mute_automation_item = dynamic_cast (&items.back ()); mute_automation_item->set_active (string_to(mute_track->gui_property ("visible"))); _main_automation_menu_map[Evoral::Parameter(MuteAutomation)] = mute_automation_item; } } void VCATimeAxisView::show_all_automation (bool apply_to_selection) { assert (!apply_to_selection); // VCAs can't yet be selected no_redraw = true; StripableTimeAxisView::show_all_automation (); no_redraw = false; request_redraw (); } void VCATimeAxisView::show_existing_automation (bool apply_to_selection) { assert (!apply_to_selection); // VCAs can't yet be selected no_redraw = true; StripableTimeAxisView::show_existing_automation (); no_redraw = false; request_redraw (); } void VCATimeAxisView::hide_all_automation (bool apply_to_selection) { assert (!apply_to_selection); // VCAs can't yet be selected no_redraw = true; StripableTimeAxisView::hide_all_automation (); no_redraw = false; request_redraw (); } void VCATimeAxisView::drop_all_slaves () { _vca->Drop (); /* EMIT SIGNAL */ if (Mixer_UI::instance()->showing_spill_for (_vca)) { Mixer_UI::instance()->show_spill (std::shared_ptr()); } } void VCATimeAxisView::choose_color () { _color_picker.popup (_vca); }