Add a double-click handler for items in the editor window.
[ardour.git] / gtk2_ardour / midi_region_view.h
index c3084aee384a24e1f2d3700ee52dbce91ce8686f..5f374da55cc85620e99b808369c15ca57fe6dd4c 100644 (file)
@@ -27,9 +27,7 @@
 
 #include "pbd/signals.h"
 
-#include "ardour/midi_track.h"
 #include "ardour/midi_model.h"
-#include "ardour/diskstream.h"
 #include "ardour/types.h"
 
 #include "editing.h"
@@ -64,6 +62,7 @@ class AutomationRegionView;
 class MidiCutBuffer;
 class MidiListEditor;
 class EditNoteDialog;
+class NotePlayer;
 
 class MidiRegionView : public RegionView
 {
@@ -118,11 +117,7 @@ public:
        void cut_copy_clear (Editing::CutCopyOp);
        void paste (framepos_t pos, float times, const MidiCutBuffer&);
 
-       /** Add a new patch change flag to the canvas.
-        * @param patch the patch change to add
-        * @param the text to display in the flag
-        */
-       void add_canvas_patch_change (ARDOUR::MidiModel::PatchChangePtr patch, const std::string& displaytext);
+       void add_canvas_patch_change (ARDOUR::MidiModel::PatchChangePtr patch, const std::string& displaytext, bool);
 
        /** Look up the given time and channel in the 'automation' and set keys accordingly.
         * @param time the time of the patch change event
@@ -130,7 +125,11 @@ public:
         * @key a reference to an instance of MIDI::Name::PatchPrimaryKey whose fields will
         *        will be set according to the result of the lookup
         */
-       void get_patch_key_at(double time, uint8_t channel, MIDI::Name::PatchPrimaryKey& key);
+       void get_patch_key_at (double time, uint8_t channel, MIDI::Name::PatchPrimaryKey& key) const;
+
+       /** Convert a given PatchChange into a PatchPrimaryKey
+        */
+       MIDI::Name::PatchPrimaryKey patch_change_to_patch_key (ARDOUR::MidiModel::PatchChangePtr);
 
        /** Change old_patch to new_patch.
         * @param old_patch the canvas patch change which is to be altered
@@ -144,6 +143,8 @@ public:
        void delete_patch_change (ArdourCanvas::CanvasPatchChange *);
        void edit_patch_change (ArdourCanvas::CanvasPatchChange *);
 
+       void delete_sysex (ArdourCanvas::CanvasSysEx*);
+
        /** Alter a given patch to be its predecessor in the MIDNAM file.
         */
        void previous_patch (ArdourCanvas::CanvasPatchChange &);
@@ -182,6 +183,8 @@ public:
        void   note_left(ArdourCanvas::CanvasNoteEvent* ev);
        void   patch_entered (ArdourCanvas::CanvasPatchChange *);
        void   patch_left (ArdourCanvas::CanvasPatchChange *);
+       void   sysex_entered (ArdourCanvas::CanvasSysEx* p);
+       void   sysex_left (ArdourCanvas::CanvasSysEx* p);
        void   note_mouse_position (float xfraction, float yfraction, bool can_set_cursor=true);
        void   unique_select(ArdourCanvas::CanvasNoteEvent* ev);
        void   note_selected(ArdourCanvas::CanvasNoteEvent* ev, bool add, bool extend=false);
@@ -236,6 +239,8 @@ public:
 
        MouseState mouse_state() const { return _mouse_state; }
 
+       void note_button_release ();
+
        struct NoteResizeData {
                ArdourCanvas::CanvasNote  *canvas_note;
                ArdourCanvas::SimpleRect  *resize_rect;
@@ -274,13 +279,19 @@ public:
        void goto_previous_note (bool add_to_selection);
        void goto_next_note (bool add_to_selection);
        void change_note_lengths (bool, bool, Evoral::MusicalTime beats, bool start, bool end);
-       void change_velocities (bool up, bool fine, bool allow_smush);
+        void change_velocities (bool up, bool fine, bool allow_smush, bool all_together);
        void transpose (bool up, bool fine, bool allow_smush);
        void nudge_notes (bool forward);
        void channel_edit ();
+       void velocity_edit ();
 
        void show_list_editor ();
 
+       typedef std::set<ArdourCanvas::CanvasNoteEvent*> Selection;
+       Selection selection () const {
+               return _selection;
+       }
+       
        void selection_as_notelist (Notes& selected, bool allow_all_if_none_selected = false);
 
        void enable_display (bool);
@@ -292,6 +303,10 @@ public:
        void trim_front_ending ();
 
        void create_note_at (framepos_t, double, double, bool);
+
+       void clear_selection (bool signal = true) { clear_selection_except (0, signal); }
+
+        ARDOUR::InstrumentInfo& instrument_info() const;
        
 protected:
        /** Allows derived types to specify their visibility requirements
@@ -318,7 +333,10 @@ private:
        friend class MidiRubberbandSelectDrag;
        friend class MidiVerticalSelectDrag;
 
-       /** Emitted when the selection has been cleared in one MidiRegionView */
+       /** Emitted when the selection has been cleared in one MidiRegionView,
+        *  with the expectation that others will clear their selections in
+        *  sympathy.
+        */
        static PBD::Signal1<void, MidiRegionView*> SelectionCleared;
        PBD::ScopedConnection _selection_cleared_connection;
        void selection_cleared (MidiRegionView *);
@@ -328,21 +346,19 @@ private:
        /** Play the NoteOn event of the given note immediately
         * and schedule the playback of the corresponding NoteOff event.
         */
-       void play_midi_note(boost::shared_ptr<NoteType> note);
-       void play_midi_chord (std::vector<boost::shared_ptr<NoteType> > notes);
+       void play_midi_note (boost::shared_ptr<NoteType> note);
+       void start_playing_midi_note (boost::shared_ptr<NoteType> note);
+       void start_playing_midi_chord (std::vector<boost::shared_ptr<NoteType> > notes);
 
-       /** Play the NoteOff-Event of the given note immediately
-        * (scheduled by @ref play_midi_note()).
-        */
-       bool play_midi_note_off(boost::shared_ptr<NoteType> note);
-
-       void clear_events();
+       void clear_events (bool with_selection_signal = true);
 
        bool canvas_event(GdkEvent* ev);
        bool note_canvas_event(GdkEvent* ev);
 
-       void midi_channel_mode_changed(ARDOUR::ChannelMode mode, uint16_t mask);
-       void midi_patch_settings_changed(std::string model, std::string custom_device_mode);
+       void midi_channel_mode_changed ();
+        PBD::ScopedConnection _channel_mode_changed_connection;
+       void instrument_settings_changed ();
+       PBD::ScopedConnection _instrument_changed_connection;
 
        void change_note_channel (ArdourCanvas::CanvasNoteEvent *, int8_t, bool relative=false);
        void change_note_velocity(ArdourCanvas::CanvasNoteEvent* ev, int8_t vel, bool relative=false);
@@ -353,7 +369,6 @@ private:
                       ARDOUR::MidiModel::TimeType end_delta);
 
        void clear_selection_except (ArdourCanvas::CanvasNoteEvent* ev, bool signal = true);
-       void clear_selection (bool signal = true) { clear_selection_except (0, signal); }
        void update_drag_selection (double last_x, double x, double last_y, double y, bool extend);
        void update_vertical_drag_selection (double last_y, double y, bool extend);
 
@@ -363,16 +378,9 @@ private:
        void show_verbose_cursor (std::string const &, double, double) const;
        void show_verbose_cursor (boost::shared_ptr<NoteType>) const;
 
-       uint16_t _last_channel_selection;
        uint8_t  _current_range_min;
        uint8_t  _current_range_max;
 
-       /// MIDNAM information of the current track: Model name of MIDNAM file
-       std::string _model_name;
-
-       /// MIDNAM information of the current track: CustomDeviceMode
-       std::string _custom_device_mode;
-
        typedef std::list<ArdourCanvas::CanvasNoteEvent*> Events;
        typedef std::vector< boost::shared_ptr<ArdourCanvas::CanvasPatchChange> > PatchChanges;
        typedef std::vector< boost::shared_ptr<ArdourCanvas::CanvasSysEx> > SysExes;
@@ -401,8 +409,7 @@ private:
        MouseState _mouse_state;
        int _pressed_button;
 
-       typedef std::set<ArdourCanvas::CanvasNoteEvent*> Selection;
-       /// Currently selected CanvasNoteEvents
+       /** Currently selected CanvasNoteEvents */
        Selection _selection;
 
        bool _sort_needed;
@@ -453,7 +460,7 @@ private:
        void maybe_select_by_position (GdkEventButton* ev, double x, double y);
        void get_events (Events& e, Evoral::Sequence<Evoral::MusicalTime>::NoteOperator op, uint8_t val, int chan_mask = 0);
 
-       void display_patch_changes_on_channel (uint8_t);
+       void display_patch_changes_on_channel (uint8_t, bool);
 
        void connect_to_diskstream ();
        void data_recorded (boost::weak_ptr<ARDOUR::MidiSource>);
@@ -469,6 +476,11 @@ private:
 
        Gdk::Cursor* pre_enter_cursor;
        Gdk::Cursor* pre_press_cursor;
+
+       NotePlayer* _note_player;
+
+        ARDOUR::ChannelMode get_channel_mode() const;
+        uint16_t get_selected_channels () const;
 };