Fix multiple selection with Ctrl-Click in DnDVBox.
[ardour.git] / gtk2_ardour / processor_box.h
index 6fa11339fffb70c16ea101fba6744a318b3049b6..05fac975791f145a059c052553b5349535304dfc 100644 (file)
 #ifndef __ardour_gtk_processor_box__
 #define __ardour_gtk_processor_box__
 
+#include <cmath>
 #include <vector>
 
-#include <cmath>
+#include <boost/function.hpp>
+
 #include <gtkmm/box.h>
 #include <gtkmm/eventbox.h>
 #include <gtkmm/menu.h>
 #include <gtkmm/scrolledwindow.h>
-#include <gtkmm2ext/dndtreeview.h>
-#include <gtkmm2ext/auto_spin.h>
-#include <gtkmm2ext/click_box.h>
-#include <gtkmm2ext/dndvbox.h>
-#include <gtkmm2ext/pixfader.h>
+#include "gtkmm2ext/dndtreeview.h"
+#include "gtkmm2ext/auto_spin.h"
+#include "gtkmm2ext/click_box.h"
+#include "gtkmm2ext/dndvbox.h"
+#include "gtkmm2ext/pixfader.h"
 
 #include "pbd/stateful.h"
+#include "pbd/signals.h"
 
 #include "ardour/types.h"
 #include "ardour/ardour.h"
@@ -41,6 +44,7 @@
 #include "ardour/port_insert.h"
 #include "ardour/processor.h"
 #include "ardour/route.h"
+#include "ardour/session_handle.h"
 
 #include "pbd/fastlog.h"
 
@@ -48,6 +52,7 @@
 #include "io_selector.h"
 #include "send_ui.h"
 #include "enums.h"
+#include "window_proxy.h"
 
 class MotionController;
 class PluginSelector;
@@ -67,6 +72,32 @@ namespace ARDOUR {
        class Session;
 }
 
+class ProcessorBox;
+
+/** A WindowProxy for Processor UI windows; it knows how to ask a ProcessorBox
+ *  to create a UI window for a particular processor.
+ */
+class ProcessorWindowProxy : public WindowProxy<Gtk::Window>
+{
+public:
+       ProcessorWindowProxy (std::string const &, XMLNode const *, ProcessorBox *, boost::weak_ptr<ARDOUR::Processor>);
+
+       void show ();
+       bool rc_configured () const {
+               return false;
+       }
+
+       boost::weak_ptr<ARDOUR::Processor> processor () const {
+               return _processor;
+       }
+
+       bool marked;
+
+private:
+       ProcessorBox* _processor_box;
+       boost::weak_ptr<ARDOUR::Processor> _processor;
+};
+
 class ProcessorEntry : public Gtkmm2ext::DnDVBoxChild, public sigc::trackable
 {
 public:
@@ -76,7 +107,8 @@ public:
        Gtk::Widget& widget ();
        std::string drag_text () const;
        boost::shared_ptr<ARDOUR::Processor> processor () const;
-       void set_width (Width);
+       void set_enum_width (Width);
+       virtual void set_pixel_width (int) {}
 
 protected:
        
@@ -86,7 +118,7 @@ private:
 
        void active_toggled ();
        void processor_active_changed ();
-       void processor_name_changed ();
+       void processor_property_changed (const PBD::PropertyChange&);
        std::string name () const;
        
        Gtk::EventBox _event_box;
@@ -95,6 +127,8 @@ private:
        Gtk::CheckButton _active;
        boost::shared_ptr<ARDOUR::Processor> _processor;
        Width _width;
+       PBD::ScopedConnection active_connection;
+       PBD::ScopedConnection name_connection;
 };
 
 class SendProcessorEntry : public ProcessorEntry
@@ -104,6 +138,9 @@ public:
 
        static void setup_slider_pix ();
 
+       void set_enum_width (Width, int);
+       void set_pixel_width (int);
+
 private:
        void show_gain ();
        void gain_adjusted ();
@@ -112,15 +149,16 @@ private:
        Gtk::Adjustment _adjustment;
        Gtkmm2ext::HSliderController _fader;
        bool _ignore_gain_change;
-       
+       PBD::ScopedConnection send_gain_connection;
+
        static Glib::RefPtr<Gdk::Pixbuf> _slider;
 };
 
-class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
+class ProcessorBox : public Gtk::HBox, public PluginInterestedObject, public ARDOUR::SessionHandlePtr
 {
   public:
-       ProcessorBox (ARDOUR::Session&, sigc::slot<PluginSelector*> get_plugin_selector,
-                       RouteRedirectSelection&, MixerStrip* parent, bool owner_is_mixer = false);
+       ProcessorBox (ARDOUR::Session*, boost::function<PluginSelector*()> get_plugin_selector,
+                     RouteRedirectSelection&, MixerStrip* parent, bool owner_is_mixer = false);
        ~ProcessorBox ();
 
        void set_route (boost::shared_ptr<ARDOUR::Route>);
@@ -134,20 +172,26 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        void select_all_inserts ();
        void select_all_sends ();
 
+       Gtk::Window* get_processor_ui (boost::shared_ptr<ARDOUR::Processor>) const;
+       void edit_processor (boost::shared_ptr<ARDOUR::Processor>);
+       
        sigc::signal<void,boost::shared_ptr<ARDOUR::Processor> > ProcessorSelected;
        sigc::signal<void,boost::shared_ptr<ARDOUR::Processor> > ProcessorUnselected;
 
        static void register_actions();
 
   private:
+
+       /* prevent copy construction */
+       ProcessorBox (ProcessorBox const &);
+       
        boost::shared_ptr<ARDOUR::Route>  _route;
-       ARDOUR::Session &   _session;
        MixerStrip*         _parent_strip; // null if in RouteParamsUI
        bool                _owner_is_mixer;
        bool                 ab_direction;
-       std::vector<sigc::connection> connections;
+       PBD::ScopedConnectionList connections;
 
-       sigc::slot<PluginSelector*> _get_plugin_selector;
+       boost::function<PluginSelector*()> _get_plugin_selector;
 
        boost::shared_ptr<ARDOUR::Processor> _processor_being_created;
 
@@ -184,11 +228,10 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        void choose_aux (boost::weak_ptr<ARDOUR::Route>);
        void choose_send ();
        void send_io_finished (IOSelector::Result, boost::weak_ptr<ARDOUR::Processor>, IOSelectorWindow*);
-       void choose_return ();
        void return_io_finished (IOSelector::Result, boost::weak_ptr<ARDOUR::Processor>, IOSelectorWindow*);
        void choose_insert ();
        void choose_plugin ();
-       void use_plugins (const SelectedPlugins&);
+       bool use_plugins (const SelectedPlugins&);
 
        bool no_processor_redisplay;
 
@@ -201,8 +244,7 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        void redisplay_processors ();
        void add_processor_to_display (boost::weak_ptr<ARDOUR::Processor>);
        void reordered ();
-
-       void remove_processor_gui (boost::shared_ptr<ARDOUR::Processor>);
+       void route_processors_changed (ARDOUR::RouteProcessorChange);
 
        void processors_reordered (const Gtk::TreeModel::Path&, const Gtk::TreeModel::iterator&, int*);
        void compute_processor_sort_keys ();
@@ -228,20 +270,25 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        void rename_processors ();
 
        void for_selected_processors (void (ProcessorBox::*pmf)(boost::shared_ptr<ARDOUR::Processor>));
-       void get_selected_processors (ProcSelection&);
+       void get_selected_processors (ProcSelection&) const;
+
+        bool can_cut() const;
 
+       static Glib::RefPtr<Gtk::Action> cut_action;
        static Glib::RefPtr<Gtk::Action> paste_action;
+       static Glib::RefPtr<Gtk::Action> rename_action;
+       static Glib::RefPtr<Gtk::Action> edit_action;
        void paste_processor_state (const XMLNodeList&, boost::shared_ptr<ARDOUR::Processor>);
 
        void activate_processor (boost::shared_ptr<ARDOUR::Processor>);
        void deactivate_processor (boost::shared_ptr<ARDOUR::Processor>);
-       void edit_processor (boost::shared_ptr<ARDOUR::Processor>);
        void hide_processor_editor (boost::shared_ptr<ARDOUR::Processor>);
        void rename_processor (boost::shared_ptr<ARDOUR::Processor>);
 
        gint idle_delete_processor (boost::weak_ptr<ARDOUR::Processor>);
 
        void weird_plugin_dialog (ARDOUR::Plugin& p, ARDOUR::Route::ProcessorStreams streams);
+       void on_size_allocate (Gtk::Allocation &);
 
        static ProcessorBox* _current_processor_box;
 
@@ -249,7 +296,6 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        static void rb_choose_plugin ();
        static void rb_choose_insert ();
        static void rb_choose_send ();
-       static void rb_choose_return ();
        static void rb_clear ();
        static void rb_clear_pre ();
        static void rb_clear_post ();
@@ -265,8 +311,15 @@ class ProcessorBox : public Gtk::HBox, public PluginInterestedObject
        static void rb_ab_plugins ();
        static void rb_edit ();
 
-       void route_name_changed ();
+       void route_property_changed (const PBD::PropertyChange&);
        std::string generate_processor_title (boost::shared_ptr<ARDOUR::PluginInsert> pi);
+
+       std::list<ProcessorWindowProxy*> _processor_window_proxies;
+       void set_processor_ui (boost::shared_ptr<ARDOUR::Processor>, Gtk::Window *);
+       void maybe_add_processor_to_ui_list (boost::weak_ptr<ARDOUR::Processor>);
+
+       bool one_processor_can_be_edited ();
+       bool processor_can_be_edited (boost::shared_ptr<ARDOUR::Processor>);
 };
 
 #endif /* __ardour_gtk_processor_box__ */