Fix placement of automation child tracks.
[ardour.git] / gtk2_ardour / mixer_strip.cc
index 0c99dc74018d562a569b10d21d157b76f3f31db9..1a8949d1b8d719ff5c86400d76d291fa5679b344 100644 (file)
 #include <ardour/audio_diskstream.h>
 #include <ardour/panner.h>
 #include <ardour/send.h>
-#include <ardour/insert.h>
+#include <ardour/processor.h>
+#include <ardour/profile.h>
 #include <ardour/ladspa_plugin.h>
-#include <ardour/bundle.h>
-#include <ardour/session_bundle.h>
+#include <ardour/auto_bundle.h>
+#include <ardour/user_bundle.h>
 
 #include "ardour_ui.h"
 #include "ardour_dialog.h"
@@ -85,8 +86,9 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
        : AxisView(sess),
          RouteUI (rt, sess, _("Mute"), _("Solo"), _("Record")),
          _mixer(mx),
-         pre_redirect_box (PreFader, sess, rt, mx.plugin_selector(), mx.selection(), in_mixer),
-         post_redirect_box (PostFader, sess, rt, mx.plugin_selector(), mx.selection(), in_mixer),
+         _mixer_owned (in_mixer),
+         pre_processor_box (PreFader, sess, rt, mx.plugin_selector(), mx.selection(), in_mixer),
+         post_processor_box (PostFader, sess, rt, mx.plugin_selector(), mx.selection(), in_mixer),
          gpm (_route, sess),
          panners (_route, sess),
          button_table (3, 2),
@@ -105,7 +107,9 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
        input_selector = 0;
        output_selector = 0;
        group_menu = 0;
-       _marked_for_display = false;
+       if (!_route->is_hidden()) {
+               _marked_for_display = true;
+       }
        route_ops_menu = 0;
        ignore_comment_edit = false;
        ignore_toggle = false;
@@ -114,8 +118,10 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
        comment_area = 0;
        _width_owner = 0;
 
-       width_button.add (*(manage (new Gtk::Image (::get_icon("strip_width")))));
-       hide_button.add (*(manage (new Gtk::Image (::get_icon("hide")))));
+       Gtk::Image *width_icon = manage (new Gtk::Image (::get_icon("strip_width")));
+       Gtk::Image *hide_icon = manage (new Gtk::Image (::get_icon("hide")));
+       width_button.add (*width_icon);
+       hide_button.add (*hide_icon);
 
        input_label.set_text (_("Input"));
        input_button.add (input_label);
@@ -178,12 +184,6 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
        bottom_button_table.attach (group_button, 0, 1, 0, 1);
        
        if (is_audio_track()) {
-               
-               rec_enable_button->signal_button_press_event().connect (mem_fun(*this, &RouteUI::rec_enable_press), false);
-               rec_enable_button->signal_button_release_event().connect (mem_fun(*this, &RouteUI::rec_enable_release));
-
-               rec_enable_button->set_name ("MixerRecordEnableButton");
-
                boost::shared_ptr<AudioTrack> at = audio_track();
 
                at->FreezeChange.connect (mem_fun(*this, &MixerStrip::map_frozen));
@@ -202,9 +202,16 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
                button_table.attach (speed_frame, 0, 2, 5, 6);
 #endif /* VARISPEED_IN_MIXER_STRIP */
 
+       }
+       
+       if(rec_enable_button) {
+               rec_enable_button->signal_button_press_event().connect (mem_fun(*this, &RouteUI::rec_enable_press), false);
+               rec_enable_button->signal_button_release_event().connect (mem_fun(*this, &RouteUI::rec_enable_release));
+       
+               rec_enable_button->set_name ("MixerRecordEnableButton");
                button_table.attach (*rec_enable_button, 0, 2, 2, 3);
        }
-
+       
        name_button.add (name_label);
        name_button.set_name ("MixerNameButton");
        Gtkmm2ext::set_size_request_to_display_given_text (name_button, "longest label", 2, 2);
@@ -251,17 +258,18 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
 
        global_vpacker.pack_start (*whvbox, Gtk::PACK_SHRINK);
        global_vpacker.pack_start (button_table,Gtk::PACK_SHRINK);
-       global_vpacker.pack_start (pre_redirect_box, true, true);
+       global_vpacker.pack_start (pre_processor_box, true, true);
        global_vpacker.pack_start (middle_button_table,Gtk::PACK_SHRINK);
        global_vpacker.pack_start (*gain_meter_alignment,Gtk::PACK_SHRINK);
        global_vpacker.pack_start (bottom_button_table,Gtk::PACK_SHRINK);
-       global_vpacker.pack_start (post_redirect_box, true, true);
-       if (!is_midi_track())
+       global_vpacker.pack_start (post_processor_box, true, true);
+       if (!is_midi_track()) {
                global_vpacker.pack_start (panners, Gtk::PACK_SHRINK);
+       }
        global_vpacker.pack_start (output_button, Gtk::PACK_SHRINK);
        global_vpacker.pack_start (comment_button, Gtk::PACK_SHRINK);
 
-       if (route()->master() || route()->control()) {
+       if (route()->is_master() || route()->is_control()) {
                
                if (scrollbar_height == 0) {
                        HScrollbar scrollbar;
@@ -332,8 +340,8 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
 
        /* now force an update of all the various elements */
 
-       pre_redirect_box.update();
-       post_redirect_box.update();
+       pre_processor_box.update();
+       post_processor_box.update();
        mute_changed (0);
        solo_changed (0);
        name_changed ();
@@ -353,6 +361,45 @@ MixerStrip::MixerStrip (Mixer_UI& mx, Session& sess, boost::shared_ptr<Route> rt
        update_output_display ();
 
        add_events (Gdk::BUTTON_RELEASE_MASK);
+
+       whvbox->show();
+       hide_icon->show();
+       width_icon->show();
+       gain_meter_alignment->show_all();
+
+       pre_processor_box.show();
+
+       if (!route()->is_master() && !route()->is_control()) {
+               /* we don't allow master or control routes to be hidden */
+               hide_button.show();
+       }
+       width_button.show();
+       width_hide_box.show();
+       global_frame.show();
+       global_vpacker.show();
+       button_table.show();
+       middle_button_table.show();
+       bottom_button_table.show();
+       gain_unit_button.show();
+       gain_unit_label.show();
+       meter_point_button.show();
+       meter_point_label.show();
+       diskstream_button.show();
+       diskstream_label.show();
+       input_button.show();
+       input_label.show();
+       output_button.show();
+       output_label.show();
+       name_label.show();
+       name_button.show();
+       comment_button.show();
+       group_button.show();
+       group_label.show();
+       speed_spinner.show();
+       speed_label.show();
+       speed_frame.show();
+
+       show();
 }
 
 MixerStrip::~MixerStrip ()
@@ -400,14 +447,12 @@ MixerStrip::set_width (Width w, void* owner)
 
        gpm.set_width (w);
        panners.set_width (w);
-       pre_redirect_box.set_width (w);
-       post_redirect_box.set_width (w);
+       pre_processor_box.set_width (w);
+       post_processor_box.set_width (w);
 
-       _width_owner = owner;
+       boost::shared_ptr<AutomationList> gain_automation = _route->gain_control()->list();
 
-       if (_width == w) {
-               return;
-       }
+       _width_owner = owner;
 
        ensure_xml_node ();
        
@@ -435,8 +480,8 @@ MixerStrip::set_width (Width w, void* owner)
                       ((Gtk::Label*)comment_button.get_child())->set_text (_("*comments*"));
                }
 
-               ((Gtk::Label*)gpm.gain_automation_style_button.get_child())->set_text (gpm.astyle_string(_route->gain_automation_curve().automation_style()));
-               ((Gtk::Label*)gpm.gain_automation_state_button.get_child())->set_text (gpm.astate_string(_route->gain_automation_curve().automation_state()));
+               ((Gtk::Label*)gpm.gain_automation_style_button.get_child())->set_text (gpm.astyle_string(gain_automation->automation_style()));
+               ((Gtk::Label*)gpm.gain_automation_state_button.get_child())->set_text (gpm.astate_string(gain_automation->automation_state()));
                ((Gtk::Label*)panners.pan_automation_style_button.get_child())->set_text (panners.astyle_string(_route->panner().automation_style()));
                ((Gtk::Label*)panners.pan_automation_state_button.get_child())->set_text (panners.astate_string(_route->panner().automation_state()));
                Gtkmm2ext::set_size_request_to_display_given_text (name_button, "long", 2, 2);
@@ -457,8 +502,8 @@ MixerStrip::set_width (Width w, void* owner)
                       ((Gtk::Label*)comment_button.get_child())->set_text (_("*Cmt*"));
                }
 
-               ((Gtk::Label*)gpm.gain_automation_style_button.get_child())->set_text (gpm.short_astyle_string(_route->gain_automation_curve().automation_style()));
-               ((Gtk::Label*)gpm.gain_automation_state_button.get_child())->set_text (gpm.short_astate_string(_route->gain_automation_curve().automation_state()));
+               ((Gtk::Label*)gpm.gain_automation_style_button.get_child())->set_text (gpm.short_astyle_string(gain_automation->automation_style()));
+               ((Gtk::Label*)gpm.gain_automation_state_button.get_child())->set_text (gpm.short_astate_string(gain_automation->automation_state()));
                ((Gtk::Label*)panners.pan_automation_style_button.get_child())->set_text (panners.short_astyle_string(_route->panner().automation_style()));
                ((Gtk::Label*)panners.pan_automation_state_button.get_child())->set_text (panners.short_astate_string(_route->panner().automation_state()));
                Gtkmm2ext::set_size_request_to_display_given_text (name_button, "longest label", 2, 2);
@@ -502,6 +547,7 @@ MixerStrip::output_press (GdkEventButton *ev)
        switch (ev->button) {
 
        case 1:
+       {
                output_menu.set_name ("ArdourContextMenu");
                citems.clear();
                
@@ -509,11 +555,16 @@ MixerStrip::output_press (GdkEventButton *ev)
                citems.push_back (SeparatorElem());
                citems.push_back (MenuElem (_("Disconnect"), mem_fun (*(static_cast<RouteUI*>(this)), &RouteUI::disconnect_output)));
                citems.push_back (SeparatorElem());
-               
-               _session.foreach_bundle (this, &MixerStrip::add_bundle_to_output_menu);
+
+               std::vector<boost::shared_ptr<Bundle> > current = _route->bundles_connected_to_outputs ();
+
+               _session.foreach_bundle (
+                       bind (mem_fun (*this, &MixerStrip::add_bundle_to_output_menu), current)
+                       );
 
                output_menu.popup (1, ev->time);
                break;
+       }
                
        default:
                break;
@@ -531,7 +582,7 @@ MixerStrip::edit_output_configuration ()
        if (output_selector->is_visible()) {
                output_selector->get_toplevel()->get_window()->raise();
        } else {
-               output_selector->show_all ();
+               output_selector->present ();
        }
 }
 
@@ -545,7 +596,7 @@ MixerStrip::edit_input_configuration ()
        if (input_selector->is_visible()) {
                input_selector->get_toplevel()->get_window()->raise();
        } else {
-               input_selector->show_all ();
+               input_selector->present ();
        }
 }
 
@@ -567,16 +618,21 @@ MixerStrip::input_press (GdkEventButton *ev)
        switch (ev->button) {
 
        case 1:
+       {
                citems.push_back (MenuElem (_("Edit"), mem_fun(*this, &MixerStrip::edit_input_configuration)));
                citems.push_back (SeparatorElem());
                citems.push_back (MenuElem (_("Disconnect"), mem_fun (*(static_cast<RouteUI*>(this)), &RouteUI::disconnect_input)));
                citems.push_back (SeparatorElem());
-               
-               _session.foreach_bundle (this, &MixerStrip::add_bundle_to_input_menu);
+
+               std::vector<boost::shared_ptr<Bundle> > current = _route->bundles_connected_to_inputs ();
+
+               _session.foreach_bundle (
+                       bind (mem_fun (*this, &MixerStrip::add_bundle_to_input_menu), current)
+                       );
 
                input_menu.popup (1, ev->time);
                break;
-               
+       }
        default:
                break;
        }
@@ -584,12 +640,12 @@ MixerStrip::input_press (GdkEventButton *ev)
 }
 
 void
-MixerStrip::bundle_input_chosen (ARDOUR::Bundle *c)
+MixerStrip::bundle_input_chosen (boost::shared_ptr<ARDOUR::Bundle> c)
 {
        if (!ignore_toggle) {
 
                try { 
-                       _route->use_input_bundle (*c, this);
+                       _route->connect_input_ports_to_bundle (c, this);
                }
 
                catch (AudioEngine::PortRegistrationFailure& err) {
@@ -600,12 +656,12 @@ MixerStrip::bundle_input_chosen (ARDOUR::Bundle *c)
 }
 
 void
-MixerStrip::bundle_output_chosen (ARDOUR::Bundle *c)
+MixerStrip::bundle_output_chosen (boost::shared_ptr<ARDOUR::Bundle> c)
 {
        if (!ignore_toggle) {
 
                try { 
-                       _route->use_output_bundle (*c, this);
+                       _route->connect_output_ports_to_bundle (c, this);
                }
 
                catch (AudioEngine::PortRegistrationFailure& err) {
@@ -616,23 +672,23 @@ MixerStrip::bundle_output_chosen (ARDOUR::Bundle *c)
 }
 
 void
-MixerStrip::add_bundle_to_input_menu (ARDOUR::Bundle* c)
+MixerStrip::add_bundle_to_input_menu (boost::shared_ptr<Bundle> b, std::vector<boost::shared_ptr<Bundle> > const & current)
 {
        using namespace Menu_Helpers;
 
-       if (dynamic_cast<InputBundle *> (c) == 0) {
-               return;
-       }
+       /* the input menu needs to contain only output bundles (that we
+          can connect inputs to */
+       if (b->ports_are_outputs() == false) {
+               return;
+       }
 
        MenuList& citems = input_menu.items();
        
-       if (c->nchannels() == _route->n_inputs().n_total()) {
+       if (b->nchannels() == _route->n_inputs()) {
 
-               citems.push_back (CheckMenuElem (c->name(), bind (mem_fun(*this, &MixerStrip::bundle_input_chosen), c)));
-               
-               ARDOUR::Bundle *current = _route->input_bundle();
-               
-               if (current == c) {
+               citems.push_back (CheckMenuElem (b->name(), bind (mem_fun(*this, &MixerStrip::bundle_input_chosen), b)));
+
+               if (std::find (current.begin(), current.end(), b) != current.end()) {
                        ignore_toggle = true;
                        dynamic_cast<CheckMenuItem *> (&citems.back())->set_active (true);
                        ignore_toggle = false;
@@ -641,22 +697,22 @@ MixerStrip::add_bundle_to_input_menu (ARDOUR::Bundle* c)
 }
 
 void
-MixerStrip::add_bundle_to_output_menu (ARDOUR::Bundle* c)
+MixerStrip::add_bundle_to_output_menu (boost::shared_ptr<Bundle> b, std::vector<boost::shared_ptr<Bundle> > const & current)
 {
        using namespace Menu_Helpers;
 
-       if (dynamic_cast<OutputBundle *> (c) == 0) {
-               return;
-       }
+       /* the output menu needs to contain only input bundles (that we
+          can connect outputs to */
+       if (b->ports_are_inputs() == false) {
+               return;
+       }
 
-       if (c->nchannels() == _route->n_outputs().n_total()) {
+       if (b->nchannels() == _route->n_outputs()) {
 
                MenuList& citems = output_menu.items();
-               citems.push_back (CheckMenuElem (c->name(), bind (mem_fun(*this, &MixerStrip::bundle_output_chosen), c)));
-               
-               ARDOUR::Bundle *current = _route->output_bundle();
+               citems.push_back (CheckMenuElem (b->name(), bind (mem_fun(*this, &MixerStrip::bundle_output_chosen), b)));
                
-               if (current == c) {
+               if (std::find (current.begin(), current.end(), b) != current.end()) {
                        ignore_toggle = true;
                        dynamic_cast<CheckMenuItem *> (&citems.back())->set_active (true);
                        ignore_toggle = false;
@@ -667,11 +723,7 @@ MixerStrip::add_bundle_to_output_menu (ARDOUR::Bundle* c)
 void
 MixerStrip::update_diskstream_display ()
 {
-       if (is_audio_track()) {
-
-               map_frozen ();
-
-               update_input_display ();
+       if (is_track()) {
 
                if (input_selector) {
                        input_selector->hide_all ();
@@ -681,9 +733,6 @@ MixerStrip::update_diskstream_display ()
 
        } else {
 
-               map_frozen ();
-
-               update_input_display ();
                show_passthru_color ();
        }
 }
@@ -699,8 +748,8 @@ MixerStrip::connect_to_pan ()
        if (!_route->panner().empty()) {
                StreamPanner* sp = _route->panner().front();
 
-               panstate_connection = sp->automation().automation_state_changed.connect (mem_fun(panners, &PannerUI::pan_automation_state_changed));
-               panstyle_connection = sp->automation().automation_style_changed.connect (mem_fun(panners, &PannerUI::pan_automation_style_changed));
+               panstate_connection = sp->pan_control()->list()->automation_state_changed.connect (mem_fun(panners, &PannerUI::pan_automation_state_changed));
+               panstyle_connection = sp->pan_control()->list()->automation_style_changed.connect (mem_fun(panners, &PannerUI::pan_automation_style_changed));
        }
 
        panners.pan_changed (this);
@@ -709,10 +758,11 @@ MixerStrip::connect_to_pan ()
 void
 MixerStrip::update_input_display ()
 {
-       ARDOUR::Bundle *c;
+       std::vector<boost::shared_ptr<ARDOUR::Bundle> > c = _route->bundles_connected_to_inputs ();
 
-       if ((c = _route->input_bundle()) != 0) {
-               input_label.set_text (c->name());
+       /* XXX: how do we represent >1 connected bundle? */
+       if (c.empty() == false) {
+               input_label.set_text (c[0]->name());
        } else {
                switch (_width) {
                case Wide:
@@ -729,10 +779,11 @@ MixerStrip::update_input_display ()
 void
 MixerStrip::update_output_display ()
 {
-       ARDOUR::Bundle *c;
+       std::vector<boost::shared_ptr<ARDOUR::Bundle> > c = _route->bundles_connected_to_outputs ();
 
-       if ((c = _route->output_bundle()) != 0) {
-               output_label.set_text (c->name());
+       /* XXX: how do we represent >1 connected bundle? */
+       if (c.empty() == false) {
+               output_label.set_text (c[0]->name());
        } else {
                switch (_width) {
                case Wide:
@@ -763,12 +814,14 @@ void
 MixerStrip::input_changed (IOChange change, void *src)
 {
        Gtkmm2ext::UI::instance()->call_slot (mem_fun(*this, &MixerStrip::update_input_display));
+       set_width(_width, this);
 }
 
 void
 MixerStrip::output_changed (IOChange change, void *src)
 {
        Gtkmm2ext::UI::instance()->call_slot (mem_fun(*this, &MixerStrip::update_output_display));
+       set_width(_width, this);
 }
 
 
@@ -984,6 +1037,11 @@ MixerStrip::build_route_ops_menu ()
        items.push_back (CheckMenuElem (_("Active"), mem_fun (*this, &RouteUI::toggle_route_active)));
        route_active_menu_item = dynamic_cast<CheckMenuItem *> (&items.back());
        route_active_menu_item->set_active (_route->active());
+
+       items.push_back (SeparatorElem());
+
+       items.push_back (MenuElem (_("Adjust latency"), mem_fun (*this, &RouteUI::adjust_latency)));
+
        items.push_back (SeparatorElem());
        items.push_back (CheckMenuElem (_("Invert Polarity"), mem_fun (*this, &RouteUI::toggle_polarity)));
        polarity_menu_item = dynamic_cast<CheckMenuItem *> (&items.back());
@@ -995,7 +1053,9 @@ MixerStrip::build_route_ops_menu ()
        build_remote_control_menu ();
        
        items.push_back (SeparatorElem());
-       items.push_back (MenuElem (_("Remote Control ID"), *remote_control_menu));
+       if (!Profile->get_sae()) {
+              items.push_back (MenuElem (_("Remote Control ID"), *remote_control_menu));
+        }
 
        items.push_back (SeparatorElem());
        items.push_back (MenuElem (_("Remove"), mem_fun(*this, &RouteUI::remove_this_route)));
@@ -1138,24 +1198,32 @@ MixerStrip::map_frozen ()
        if (at) {
                switch (at->freeze_state()) {
                case AudioTrack::Frozen:
-                       pre_redirect_box.set_sensitive (false);
-                       post_redirect_box.set_sensitive (false);
+                       pre_processor_box.set_sensitive (false);
+                       post_processor_box.set_sensitive (false);
                        speed_spinner.set_sensitive (false);
                        break;
                default:
-                       pre_redirect_box.set_sensitive (true);
-                       post_redirect_box.set_sensitive (true);
+                       pre_processor_box.set_sensitive (true);
+                       post_processor_box.set_sensitive (true);
                        speed_spinner.set_sensitive (true);
+                       // XXX need some way, maybe, to retoggle redirect editors
                        break;
                }
        }
-       _route->foreach_insert (this, &MixerStrip::hide_insert_editor);
+       
+       hide_redirect_editors ();
+}
+
+void
+MixerStrip::hide_redirect_editors ()
+{
+       _route->foreach_processor (this, &MixerStrip::hide_processor_editor);
 }
 
 void
-MixerStrip::hide_insert_editor (boost::shared_ptr<Insert> insert)
+MixerStrip::hide_processor_editor (boost::shared_ptr<Processor> processor)
 {
-       void* gui = insert->get_gui ();
+       void* gui = processor->get_gui ();
        
        if (gui) {
                static_cast<Gtk::Widget*>(gui)->hide ();
@@ -1179,19 +1247,19 @@ MixerStrip::route_active_changed ()
        } else if (is_audio_track()) {
                if (_route->active()) {
                        set_name ("AudioTrackStripBase");
-                       gpm.set_meter_strip_name ("AudioTrackStripBase");
+                       gpm.set_meter_strip_name ("AudioTrackMetrics");
                } else {
                        set_name ("AudioTrackStripBaseInactive");
-                       gpm.set_meter_strip_name ("AudioTrackStripBaseInactive");
+                       gpm.set_meter_strip_name ("AudioTrackMetricsInactive");
                }
                gpm.set_fader_name ("AudioTrackFader");
        } else {
                if (_route->active()) {
                        set_name ("AudioBusStripBase");
-                       gpm.set_meter_strip_name ("AudioBusStripBase");
+                       gpm.set_meter_strip_name ("AudioBusMetrics");
                } else {
                        set_name ("AudioBusStripBaseInactive");
-                       gpm.set_meter_strip_name ("AudioBusStripBaseInactive");
+                       gpm.set_meter_strip_name ("AudioBusMetricsInactive");
                }
                gpm.set_fader_name ("AudioBusFader");
                
@@ -1236,5 +1304,8 @@ MixerStrip::meter_changed (void *src)
        }
 
        gpm.setup_meters ();
+               // reset peak when meter point changes
+               gpm.reset_peak_display();
+               set_width(_width, this);
 }