even more consistent labels in the mundle manager
[ardour.git] / gtk2_ardour / bundle_manager.cc
index c5981f055214ef26f3ffcc567480fa208ed6aa1c..2c44a62a342bc4606e1eb11c21137b1c5b4a3280 100644 (file)
 #include <gtkmm/table.h>
 #include <gtkmm/comboboxtext.h>
 #include <gtkmm/alignment.h>
+
 #include "ardour/session.h"
 #include "ardour/user_bundle.h"
-#include "ardour/audioengine.h"
 #include "bundle_manager.h"
-#include "i18n.h"
+#include "gui_thread.h"
+#include "pbd/i18n.h"
 #include "utils.h"
 
 using namespace std;
 using namespace ARDOUR;
+using namespace ARDOUR_UI_UTILS;
 
-BundleEditorMatrix::BundleEditorMatrix (
-       Gtk::Window* parent, Session& session, boost::shared_ptr<Bundle> bundle
-       )
-       : PortMatrix (parent, session, bundle->type()),
-         _bundle (bundle)
+BundleEditorMatrix::BundleEditorMatrix (Gtk::Window* parent, Session* session, boost::shared_ptr<Bundle> bundle)
+       : PortMatrix (parent, session, DataType::NIL)
+       , _bundle (bundle)
 {
        _port_group = boost::shared_ptr<PortGroup> (new PortGroup (""));
        _port_group->add_bundle (_bundle);
+
+       setup_all_ports ();
+       init ();
 }
 
 void
@@ -52,7 +55,12 @@ BundleEditorMatrix::setup_ports (int dim)
                _ports[OURS].add_group (_port_group);
        } else {
                _ports[OTHER].suspend_signals ();
-               _ports[OTHER].gather (_session, _bundle->ports_are_inputs());
+
+               /* when we gather, allow the matrix to contain bundles with duplicate port sets,
+                  otherwise in some cases the basic system IO ports may be hidden, making
+                  the bundle editor useless */
+
+               _ports[OTHER].gather (_session, DataType::NIL, _bundle->ports_are_inputs(), true, show_only_bundles ());
                _ports[OTHER].remove_bundle (_bundle);
                _ports[OTHER].resume_signals ();
        }
@@ -74,11 +82,15 @@ BundleEditorMatrix::set_state (BundleChannel c[2], bool s)
 PortMatrixNode::State
 BundleEditorMatrix::get_state (BundleChannel c[2]) const
 {
+       if (c[0].bundle->nchannels() == ChanCount::ZERO || c[1].bundle->nchannels() == ChanCount::ZERO) {
+               return PortMatrixNode::NOT_ASSOCIATED;
+       }
+
        Bundle::PortList const& pl = c[OTHER].bundle->channel_ports (c[OTHER].channel);
        if (pl.empty ()) {
                return PortMatrixNode::NOT_ASSOCIATED;
        }
-       
+
        for (Bundle::PortList::const_iterator i = pl.begin(); i != pl.end(); ++i) {
                if (!c[OURS].bundle->port_attached_to_channel (c[OURS].channel, *i)) {
                        return PortMatrixNode::NOT_ASSOCIATED;
@@ -89,33 +101,32 @@ BundleEditorMatrix::get_state (BundleChannel c[2]) const
 }
 
 bool
-BundleEditorMatrix::can_add_channel (boost::shared_ptr<Bundle> b) const
+BundleEditorMatrix::can_add_channels (boost::shared_ptr<Bundle> b) const
 {
        if (b == _bundle) {
                return true;
        }
 
-       return PortMatrix::can_add_channel (b);
+       return PortMatrix::can_add_channels (b);
 }
 
 void
-BundleEditorMatrix::add_channel (boost::shared_ptr<Bundle> b)
+BundleEditorMatrix::add_channel (boost::shared_ptr<Bundle> b, DataType t)
 {
        if (b == _bundle) {
 
                NameChannelDialog d;
-               d.set_position (Gtk::WIN_POS_MOUSE);
 
                if (d.run () != Gtk::RESPONSE_ACCEPT) {
                        return;
                }
 
-               _bundle->add_channel (d.get_name());
+               _bundle->add_channel (d.get_name(), t);
                setup_ports (OURS);
 
        } else {
 
-               PortMatrix::add_channel (b);
+               PortMatrix::add_channel (b, t);
 
        }
 }
@@ -151,7 +162,6 @@ void
 BundleEditorMatrix::rename_channel (BundleChannel bc)
 {
        NameChannelDialog d (bc.bundle, bc.channel);
-       d.set_position (Gtk::WIN_POS_MOUSE);
 
        if (d.run () != Gtk::RESPONSE_ACCEPT) {
                return;
@@ -166,7 +176,13 @@ BundleEditorMatrix::list_is_global (int dim) const
        return (dim == OTHER);
 }
 
-BundleEditor::BundleEditor (Session& session, boost::shared_ptr<UserBundle> bundle)
+string
+BundleEditorMatrix::disassociation_verb () const
+{
+       return _("Disassociate");
+}
+
+BundleEditor::BundleEditor (Session* session, boost::shared_ptr<UserBundle> bundle)
        : ArdourDialog (_("Edit Bundle")), _matrix (this, session, bundle), _bundle (bundle)
 {
        Gtk::Table* t = new Gtk::Table (3, 2);
@@ -187,45 +203,25 @@ BundleEditor::BundleEditor (Session& session, boost::shared_ptr<UserBundle> bund
        a = new Gtk::Alignment (0, 0.5, 0, 1);
        a->add (_input_or_output);
        t->attach (*Gtk::manage (a), 1, 2, 1, 2);
-       _input_or_output.append_text (_("Input"));
-       _input_or_output.append_text (_("Output"));
+       _input_or_output.append_text (_("Destination"));
+       _input_or_output.append_text (_("Source"));
 
        if (bundle->ports_are_inputs()) {
-               _input_or_output.set_active_text (_("Input"));
+               _input_or_output.set_active_text (_("Source"));
        } else {
-               _input_or_output.set_active_text (_("Output"));
+               _input_or_output.set_active_text (_("Destination"));
        }
 
        _input_or_output.signal_changed().connect (sigc::mem_fun (*this, &BundleEditor::input_or_output_changed));
 
-       /* Type (audio or MIDI) */
-       a = new Gtk::Alignment (1, 0.5, 0, 1);
-       a->add (*Gtk::manage (new Gtk::Label (_("Type:"))));
-       t->attach (*Gtk::manage (a), 0, 1, 2, 3, Gtk::FILL, Gtk::FILL);
-       a = new Gtk::Alignment (0, 0.5, 0, 1);
-       a->add (_type);
-       t->attach (*Gtk::manage (a), 1, 2, 2, 3);
-
-       _type.append_text (_("Audio"));
-       _type.append_text (_("MIDI"));
-
-       switch (bundle->type ()) {
-       case DataType::AUDIO:
-               _type.set_active_text (_("Audio"));
-               break;
-       case DataType::MIDI:
-               _type.set_active_text (_("MIDI"));
-               break;
-       }
-
-       _type.signal_changed().connect (sigc::mem_fun (*this, &BundleEditor::type_changed));
-
        get_vbox()->pack_start (*Gtk::manage (t), false, false);
        get_vbox()->pack_start (_matrix);
        get_vbox()->set_spacing (4);
 
        add_button (Gtk::Stock::CLOSE, Gtk::RESPONSE_ACCEPT);
        show_all ();
+
+       signal_key_press_event().connect (sigc::mem_fun (_matrix, &BundleEditorMatrix::key_press));
 }
 
 void
@@ -247,7 +243,7 @@ BundleEditor::input_or_output_changed ()
 {
        _bundle->remove_ports_from_channels ();
 
-       if (_input_or_output.get_active_text() == _("Output")) {
+       if (_input_or_output.get_active_text() == _("Source")) {
                _bundle->set_ports_are_outputs ();
        } else {
                _bundle->set_ports_are_inputs ();
@@ -256,18 +252,6 @@ BundleEditor::input_or_output_changed ()
        _matrix.setup_all_ports ();
 }
 
-void
-BundleEditor::type_changed ()
-{
-       _bundle->remove_ports_from_channels ();
-
-       DataType const t = _type.get_active_text() == _("Audio") ?
-               DataType::AUDIO : DataType::MIDI;
-
-       _bundle->set_type (t);
-       _matrix.set_type (t);
-}
-
 void
 BundleEditor::on_map ()
 {
@@ -276,15 +260,19 @@ BundleEditor::on_map ()
 }
 
 
-BundleManager::BundleManager (Session& session)
-       : ArdourDialog (_("Bundle Manager")), _session (session), edit_button (_("Edit")), delete_button (_("Delete"))
+BundleManager::BundleManager (Session* session)
+       : ArdourDialog (_("Bundle Manager"))
+       , edit_button (_("Edit"))
+       , delete_button (_("Delete"))
 {
+       set_session (session);
+
        _list_model = Gtk::ListStore::create (_list_model_columns);
        _tree_view.set_model (_list_model);
        _tree_view.append_column (_("Name"), _list_model_columns.name);
        _tree_view.set_headers_visible (false);
 
-       boost::shared_ptr<BundleList> bundles = _session.bundles ();
+       boost::shared_ptr<BundleList> bundles = _session->bundles ();
        for (BundleList::iterator i = bundles->begin(); i != bundles->end(); ++i) {
                add_bundle (*i);
        }
@@ -299,7 +287,7 @@ BundleManager::BundleManager (Session& session)
        edit_button.set_image (*Gtk::manage (new Gtk::Image (Gtk::Stock::EDIT, Gtk::ICON_SIZE_BUTTON)));
        edit_button.signal_clicked().connect (sigc::mem_fun (*this, &BundleManager::edit_clicked));
        buttons->pack_start (edit_button, false, false);
-       delete_button.set_image (*Gtk::manage (new Gtk::Image (Gtk::Stock::DELETE, Gtk::ICON_SIZE_BUTTON)));
+       delete_button.set_image (*Gtk::manage (new Gtk::Image (Gtk::StockID(GTK_STOCK_DELETE), Gtk::ICON_SIZE_BUTTON)));
        delete_button.signal_clicked().connect (sigc::mem_fun (*this, &BundleManager::delete_clicked));
        buttons->pack_start (delete_button, false, false);
 
@@ -322,6 +310,9 @@ BundleManager::BundleManager (Session& session)
                sigc::mem_fun (*this, &BundleManager::row_activated)
                );
 
+       Gtk::Button* close_but = add_button (Gtk::Stock::CLOSE, Gtk::RESPONSE_ACCEPT);
+       close_but->signal_clicked ().connect (sigc::mem_fun (*this, &Gtk::Window::hide));
+
        set_button_sensitivity ();
 
        show_all ();
@@ -342,9 +333,10 @@ BundleManager::new_clicked ()
        boost::shared_ptr<UserBundle> b (new UserBundle (_("Bundle")));
 
        /* Start off with a single channel */
-       b->add_channel ("1");
+       /* XXX: allow user to specify type */
+       b->add_channel ("1", DataType::AUDIO);
 
-       _session.add_bundle (b);
+       _session->add_bundle (b);
        add_bundle (b);
 
        BundleEditor e (_session, b);
@@ -368,7 +360,7 @@ BundleManager::delete_clicked ()
        Gtk::TreeModel::iterator i = _tree_view.get_selection()->get_selected();
        if (i) {
                boost::shared_ptr<UserBundle> b = (*i)[_list_model_columns.bundle];
-               _session.remove_bundle (b);
+               _session->remove_bundle (b);
                _list_model->erase (i);
        }
 }
@@ -385,7 +377,7 @@ BundleManager::add_bundle (boost::shared_ptr<Bundle> b)
        (*i)[_list_model_columns.name] = u->name ();
        (*i)[_list_model_columns.bundle] = u;
 
-       u->Changed.connect (sigc::bind (sigc::mem_fun (*this, &BundleManager::bundle_changed), u));
+       u->Changed.connect (bundle_connections, invalidator (*this), boost::bind (&BundleManager::bundle_changed, this, _1, u), gui_context());
 }
 
 void
@@ -410,29 +402,28 @@ BundleManager::bundle_changed (Bundle::Change c, boost::shared_ptr<UserBundle> b
 }
 
 void
-BundleManager::row_activated (Gtk::TreeModel::Path const & p, Gtk::TreeViewColumn* c)
+BundleManager::row_activated (Gtk::TreeModel::Path const & p, Gtk::TreeViewColumn*)
 {
        Gtk::TreeModel::iterator i = _list_model->get_iter (p);
        if (!i) {
                return;
        }
-       
+
        boost::shared_ptr<UserBundle> b = (*i)[_list_model_columns.bundle];
        BundleEditor e (_session, b);
        e.run ();
 }
 
 NameChannelDialog::NameChannelDialog ()
-       : ArdourDialog (_("Add channel")),
+       : ArdourDialog (_("Add Channel")),
          _adding (true)
 {
        setup ();
 }
 
 NameChannelDialog::NameChannelDialog (boost::shared_ptr<Bundle> b, uint32_t c)
-       : ArdourDialog (_("Rename channel")),
+       : ArdourDialog (_("Rename Channel")),
          _bundle (b),
-         _channel (c),
          _adding (false)
 {
        _name.set_text (b->channel_name (c));