Grey-out patch changes on inactive channels rather than not
[ardour.git] / gtk2_ardour / midi_region_view.h
index e407b2ab4daf340dd448893dbe9a330c0a855dcc..f47c94a1a96fbbbdd0bc6f710362cf345744e956 100644 (file)
@@ -28,7 +28,6 @@
 #include "pbd/signals.h"
 
 #include "ardour/midi_track.h"
-#include "ardour/midi_region.h"
 #include "ardour/midi_model.h"
 #include "ardour/diskstream.h"
 #include "ardour/types.h"
@@ -85,8 +84,7 @@ public:
 
        virtual void init (Gdk::Color const & basic_color, bool wfd);
 
-       inline const boost::shared_ptr<ARDOUR::MidiRegion> midi_region() const
-       { return boost::dynamic_pointer_cast<ARDOUR::MidiRegion>(_region); }
+       const boost::shared_ptr<ARDOUR::MidiRegion> midi_region() const;
 
        inline MidiTimeAxisView* midi_view() const
        { return dynamic_cast<MidiTimeAxisView*>(&trackview); }
@@ -120,11 +118,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
@@ -169,8 +163,6 @@ public:
        void end_write();
        void extend_active_notes();
 
-       void create_note_at(double x, double y, double length, bool);
-
        void display_model(boost::shared_ptr<ARDOUR::MidiModel> model);
 
        void start_note_diff_command (std::string name = "midi edit");
@@ -194,6 +186,8 @@ public:
        void   delete_note (boost::shared_ptr<NoteType>);
        size_t selection_size() { return _selection.size(); }
        void   select_all_notes ();
+       void   select_range(framepos_t start, framepos_t end);
+       void   invert_selection ();
 
        void move_selection(double dx, double dy, double cumulative_dy);
        void note_dropped (ArdourCanvas::CanvasNoteEvent* ev, ARDOUR::frameoffset_t, int8_t d_note);
@@ -220,6 +214,7 @@ public:
 
        void update_resizing (ArdourCanvas::CanvasNoteEvent*, bool, double, bool);
        void commit_resizing (ArdourCanvas::CanvasNoteEvent*, bool, double, bool);
+       void abort_resizing ();
 
        /** Change the channel of the selection.
         * @param channel - the channel number of the new channel, zero-based
@@ -231,6 +226,7 @@ public:
                Pressed,
                SelectTouchDragging,
                SelectRectDragging,
+               SelectVerticalDragging,
                AddDragging
        };
 
@@ -253,21 +249,32 @@ public:
         */
        framepos_t snap_pixel_to_frame(double x);
 
-       ARDOUR::frameoffset_t snap_frame_to_frame (ARDOUR::frameoffset_t);
-
-       /** Convert a timestamp in beats to frames (both relative to region start) */
-       framepos_t beats_to_frames(double beats) const;
-
-       /** Convert a timestamp in frames to beats (both relative to region start) */
-       double frames_to_beats(framepos_t) const;
+       /** 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 position) */
+       double region_frames_to_region_beats(framepos_t) const;
+
+       /** Convert a timestamp in beats measured from source start into absolute frames */
+       framepos_t source_beats_to_absolute_frames(double beats) const;
+       /** Convert a timestamp in beats measured from source start into region-relative frames */
+       framepos_t source_beats_to_region_frames(double beats) const {
+               return source_beats_to_absolute_frames (beats) - _region->position();
+       }
+       /** Convert a timestamp in absolute frames to beats measured from source start*/
+       double absolute_frames_to_source_beats(framepos_t) const;
 
        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 ();
 
@@ -281,6 +288,18 @@ public:
        void trim_front_starting ();
        void trim_front_ending ();
 
+       void create_note_at (framepos_t, double, double, bool);
+
+       void clear_selection (bool signal = true) { clear_selection_except (0, signal); }
+
+       std::string model_name () const {
+               return _model_name;
+       }
+
+       std::string custom_device_mode () const {
+               return _custom_device_mode;
+       }
+       
 protected:
        /** Allows derived types to specify their visibility requirements
         * to the TimeAxisViewItem parent class.
@@ -303,6 +322,17 @@ protected:
 
 private:
 
+       friend class MidiRubberbandSelectDrag;
+       friend class MidiVerticalSelectDrag;
+
+       /** 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 *);
+
        friend class EditNoteDialog;
 
        /** Play the NoteOn event of the given note immediately
@@ -332,9 +362,9 @@ private:
        void trim_note(ArdourCanvas::CanvasNoteEvent* ev, ARDOUR::MidiModel::TimeType start_delta,
                       ARDOUR::MidiModel::TimeType end_delta);
 
-       void clear_selection_except(ArdourCanvas::CanvasNoteEvent* ev);
-       void clear_selection() { clear_selection_except(NULL); }
-       void update_drag_selection(double last_x, double x, double last_y, double y);
+       void clear_selection_except (ArdourCanvas::CanvasNoteEvent* ev, bool signal = true);
+       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*);
@@ -342,7 +372,6 @@ 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;
@@ -367,11 +396,6 @@ private:
        ArdourCanvas::CanvasNote*            _ghost_note;
        double                               _last_ghost_x;
        double                               _last_ghost_y;
-       double                               _drag_start_x;
-       double                               _drag_start_y;
-       double                               _last_x;
-       double                               _last_y;
-       ArdourCanvas::SimpleRect*            _drag_rect;
        ArdourCanvas::SimpleRect*            _step_edit_cursor;
        Evoral::MusicalTime                  _step_edit_cursor_width;
        Evoral::MusicalTime                  _step_edit_cursor_position;
@@ -438,19 +462,22 @@ 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::shared_ptr<ARDOUR::MidiBuffer>, boost::weak_ptr<ARDOUR::MidiSource>);
+       void data_recorded (boost::weak_ptr<ARDOUR::MidiSource>);
 
        void remove_ghost_note ();
        void mouse_mode_changed ();
        double _last_event_x;
        double _last_event_y;
 
+       framepos_t snap_frame_to_grid_underneath (framepos_t p, framecnt_t &) const;
+       
        PBD::ScopedConnection _mouse_mode_connection;
 
-       Gdk::Cursor* _pre_enter_cursor;
+       Gdk::Cursor* pre_enter_cursor;
+       Gdk::Cursor* pre_press_cursor;
 };