Merge branch 'master' into windows
[ardour.git] / gtk2_ardour / midi_region_view.h
index b42b7e5a364df47f9e8af9dbdd94abc16c77b264..219d07f37615cf482900cbfb4294b2ba1fa734c7 100644 (file)
 #include <string>
 #include <vector>
 
+#ifdef interface
+#undef interface
+#endif
+
 #include <libgnomecanvasmm.h>
 #include <libgnomecanvasmm/polygon.h>
 
 #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 +66,7 @@ class AutomationRegionView;
 class MidiCutBuffer;
 class MidiListEditor;
 class EditNoteDialog;
+class NotePlayer;
 
 class MidiRegionView : public RegionView
 {
@@ -118,11 +121,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 +129,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 +147,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 +187,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);
@@ -230,11 +237,14 @@ public:
                Pressed,
                SelectTouchDragging,
                SelectRectDragging,
+               SelectVerticalDragging,
                AddDragging
        };
 
        MouseState mouse_state() const { return _mouse_state; }
 
+       void note_button_release ();
+
        struct NoteResizeData {
                ArdourCanvas::CanvasNote  *canvas_note;
                ArdourCanvas::SimpleRect  *resize_rect;
@@ -252,13 +262,13 @@ public:
         */
        framepos_t snap_pixel_to_frame(double x);
 
-       /** Convert a timestamp in beats into frames (both relative to region start) */
+       /** Convert a timestamp in beats into frames (both relative to region position) */
        framepos_t region_beats_to_region_frames(double beats) const;
        /** Convert a timestamp in beats into absolute frames */
        framepos_t region_beats_to_absolute_frames(double beats) const {
                return _region->position() + region_beats_to_region_frames (beats);
        }
-       /** Convert a timestamp in frames to beats (both relative to region start) */
+       /** Convert a timestamp in frames to beats (both relative to region position) */
        double region_frames_to_region_beats(framepos_t) const;
 
        /** Convert a timestamp in beats measured from source start into absolute frames */
@@ -273,13 +283,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);
@@ -290,7 +306,11 @@ public:
        void trim_front_starting ();
        void trim_front_ending ();
 
-       void create_note_at (framepos_t, double, double, bool, bool);
+       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
@@ -315,8 +335,12 @@ protected:
 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 *);
@@ -326,21 +350,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);
@@ -351,8 +373,8 @@ 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);
 
        void add_to_selection (ArdourCanvas::CanvasNoteEvent*);
        void remove_from_selection (ArdourCanvas::CanvasNoteEvent*);
@@ -360,17 +382,9 @@ private:
        void show_verbose_cursor (std::string const &, double, double) const;
        void show_verbose_cursor (boost::shared_ptr<NoteType>) const;
 
-       int8_t   _force_channel;
-       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;
@@ -399,8 +413,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;
@@ -451,7 +464,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>);
@@ -465,7 +478,13 @@ private:
        
        PBD::ScopedConnection _mouse_mode_connection;
 
-       Gdk::Cursor* _pre_enter_cursor;
+       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;
 };