add QMDSP library to ardev file so that its discoverable as we try to load VAMP stuff
[ardour.git] / gtk2_ardour / editor_drag.h
index 49d131b52b6518b587a5960a92fa5a5cfc34df69..ce8a1370f4e63ca1ac6ce266f0e75ea1dee97443 100644 (file)
@@ -42,6 +42,7 @@ namespace PBD {
 namespace Gnome {
        namespace Canvas {
                class CanvasNoteEvent;
+               class CanvasPatchChange;
        }
 }
 
@@ -88,7 +89,7 @@ public:
        }
 
        /** @return current pointer frame */
-       nframes64_t current_pointer_frame () const {
+       ARDOUR::framepos_t current_pointer_frame () const {
                return _current_pointer_frame;
        }
 
@@ -98,7 +99,8 @@ private:
        bool _ending; ///< true if end_grab or abort is in progress, otherwise false
        double _current_pointer_x; ///< trackview x of the current pointer
        double _current_pointer_y; ///< trackview y of the current pointer
-       nframes64_t _current_pointer_frame; ///< frame that the pointer is now at
+       ARDOUR::framepos_t _current_pointer_frame; ///< frame that the pointer is now at
+       bool _old_follow_playhead; ///< state of Editor::follow_playhead() before the drags started
 };
 
 /** Abstract base class for dragging of things within the editor */
@@ -121,8 +123,8 @@ public:
        bool motion_handler (GdkEvent*, bool);
        void abort ();
 
-       nframes64_t adjusted_frame (nframes64_t, GdkEvent const *, bool snap = true) const;
-       nframes64_t adjusted_current_frame (GdkEvent const *, bool snap = true) const;
+       ARDOUR::framepos_t adjusted_frame (ARDOUR::framepos_t, GdkEvent const *, bool snap = true) const;
+       ARDOUR::framepos_t adjusted_current_frame (GdkEvent const *, bool snap = true) const;
        
        /** Called to start a grab of an item.
         *  @param e Event that caused the grab to start.
@@ -146,8 +148,9 @@ public:
 
        /** Called to abort a drag and return things to how
         *  they were before it started.
+        *  @param m true if some movement occurred, otherwise false.
         */
-       virtual void aborted () = 0;
+       virtual void aborted (bool m) = 0;
 
        /** @param m Mouse mode.
         *  @return true if this drag should happen in this mouse mode.
@@ -157,7 +160,7 @@ public:
        }
 
        /** @return minimum number of frames (in x) and pixels (in y) that should be considered a movement */
-       virtual std::pair<nframes64_t, int> move_threshold () const {
+       virtual std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
                return std::make_pair (1, 1);
        }
 
@@ -175,6 +178,11 @@ public:
                return true;
        }
 
+       /** Set up the _pointer_frame_offset */
+       virtual void setup_pointer_frame_offset () {
+               _pointer_frame_offset = 0;
+       }
+
 protected:
 
        double grab_x () const {
@@ -185,6 +193,10 @@ protected:
                return _grab_y;
        }
 
+       ARDOUR::framepos_t raw_grab_frame () const {
+               return _raw_grab_frame;
+       }
+
        ARDOUR::framepos_t grab_frame () const {
                return _grab_frame;
        }
@@ -205,7 +217,7 @@ protected:
        DragManager* _drags;
        ArdourCanvas::Item* _item; ///< our item
        /** Offset from the mouse's position for the drag to the start of the thing that is being dragged */
-       nframes64_t _pointer_frame_offset;
+       ARDOUR::framecnt_t _pointer_frame_offset;
        bool _x_constrained; ///< true if x motion is constrained, otherwise false
        bool _y_constrained; ///< true if y motion is constrained, otherwise false
        bool _was_rolling; ///< true if the session was rolling before the drag started, otherwise false
@@ -217,8 +229,9 @@ private:
        double _grab_y; ///< trackview y of the grab start position
        double _last_pointer_x; ///< trackview x of the pointer last time a motion occurred
        double _last_pointer_y; ///< trackview y of the pointer last time a motion occurred
-       nframes64_t _grab_frame; ///< adjusted_frame that the mouse was at when start_grab was called, or 0
-       nframes64_t _last_pointer_frame; ///< adjusted_frame the last time a motion occurred
+       ARDOUR::framepos_t _raw_grab_frame; ///< unsnapped frame that the mouse was at when start_grab was called, or 0
+       ARDOUR::framepos_t _grab_frame; ///< adjusted_frame that the mouse was at when start_grab was called, or 0
+       ARDOUR::framepos_t _last_pointer_frame; ///< adjusted_frame the last time a motion occurred
 };
 
 class RegionDrag;
@@ -234,6 +247,8 @@ struct DraggingView
        /** layer that this region is currently being displayed on */
        ARDOUR::layer_t layer;
        double initial_y; ///< the initial y position of the view before any reparenting
+       framepos_t initial_position; ///< initial position of the region
+       framepos_t initial_end; ///< initial end position of the region
        boost::shared_ptr<ARDOUR::Playlist> initial_playlist;
 };
 
@@ -276,7 +291,7 @@ public:
        virtual void start_grab (GdkEvent *, Gdk::Cursor *);
        virtual void motion (GdkEvent *, bool);
        virtual void finished (GdkEvent *, bool) = 0;
-       virtual void aborted ();
+       virtual void aborted (bool);
 
        /** @return true if the regions being `moved' came from somewhere on the canvas;
         *  false if they came from outside (e.g. from the region list).
@@ -285,11 +300,11 @@ public:
 
 protected:
 
-       double compute_x_delta (GdkEvent const *, nframes64_t *);
+       double compute_x_delta (GdkEvent const *, ARDOUR::framecnt_t *);
        bool y_movement_allowed (int, ARDOUR::layer_t) const;
 
        bool _brushing;
-       nframes64_t _last_frame_position; ///< last position of the thing being dragged
+       ARDOUR::framepos_t _last_frame_position; ///< last position of the thing being dragged
        double _total_x_delta;
        int _last_pointer_time_axis_view;
        ARDOUR::layer_t _last_pointer_layer;
@@ -305,19 +320,20 @@ public:
        RegionMoveDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &, bool, bool);
        virtual ~RegionMoveDrag () {}
 
-       virtual void start_grab (GdkEvent *, Gdk::Cursor *);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool regions_came_from_canvas () const {
                return true;
        }
 
-       std::pair<nframes64_t, int> move_threshold () const {
+       std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
                return std::make_pair (4, 4);
        }
 
+       void setup_pointer_frame_offset ();
+
 private:
        typedef std::set<boost::shared_ptr<ARDOUR::Playlist> > PlaylistSet;
 
@@ -359,10 +375,10 @@ private:
 class RegionInsertDrag : public RegionMotionDrag
 {
 public:
-       RegionInsertDrag (Editor *, boost::shared_ptr<ARDOUR::Region>, RouteTimeAxisView*, nframes64_t);
+       RegionInsertDrag (Editor *, boost::shared_ptr<ARDOUR::Region>, RouteTimeAxisView*, ARDOUR::framepos_t);
 
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool regions_came_from_canvas () const {
                return false;
@@ -377,7 +393,7 @@ public:
 
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 };
 
 /** Drags to create regions */
@@ -388,11 +404,12 @@ public:
 
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
 private:
        MidiTimeAxisView* _view;
        boost::shared_ptr<ARDOUR::Region> _region;
+        void add_region ();
 };
 
 /** Drags to resize MIDI notes */
@@ -404,7 +421,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
 private:
        MidiRegionView*     region;
@@ -421,7 +438,7 @@ class NoteDrag : public Drag
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
   private:
 
@@ -436,11 +453,33 @@ class NoteDrag : public Drag
        double _note_height;
 };
 
+/** Drag to move MIDI patch changes */
+class PatchChangeDrag : public Drag
+{
+public:
+       PatchChangeDrag (Editor *, ArdourCanvas::CanvasPatchChange *, MidiRegionView *);
+
+       void motion (GdkEvent *, bool);
+       void finished (GdkEvent *, bool);
+       void aborted (bool);
+
+       bool y_movement_matters () const {
+               return false;
+       }
+
+       void setup_pointer_frame_offset ();
+
+private:
+       MidiRegionView* _region_view;
+       ArdourCanvas::CanvasPatchChange* _patch_change;
+       double _cumulative_dx;
+};
+
 /** Drag of region gain */
 class RegionGainDrag : public Drag
 {
 public:
-       RegionGainDrag (Editor *e, ArdourCanvas::Item *i) : Drag (e, i) {}
+       RegionGainDrag (Editor *, ArdourCanvas::Item *);
 
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
@@ -448,7 +487,7 @@ public:
                return (m == Editing::MouseGain);
        }
 
-       void aborted ();
+       void aborted (bool);
 };
 
 /** Drag to trim region(s) */
@@ -466,16 +505,17 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool y_movement_matters () const {
                return false;
        }
 
+       void setup_pointer_frame_offset ();
+       
 private:
 
        Operation _operation;
-       bool _have_transaction; ///< true if a transaction has been started, false otherwise. Must be set true by derived class.
 };
 
 /** Meter marker drag */
@@ -487,7 +527,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool allow_vertical_autoscroll () const {
                return false;
@@ -496,6 +536,8 @@ public:
        bool y_movement_matters () const {
                return false;
        }
+
+       void setup_pointer_frame_offset ();
        
 private:
        MeterMarker* _marker;
@@ -511,7 +553,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool allow_vertical_autoscroll () const {
                return false;
@@ -521,13 +563,15 @@ public:
                return false;
        }
 
+       void setup_pointer_frame_offset ();
+       
 private:
        TempoMarker* _marker;
        bool _copy;
 };
 
 
-/** Drag of a cursor */
+/** Drag of the playhead cursor */
 class CursorDrag : public Drag
 {
 public:
@@ -536,7 +580,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool active (Editing::MouseMode) {
                return true;
@@ -549,11 +593,11 @@ public:
        bool y_movement_matters () const {
                return false;
        }
-
+       
 private:
-       EditorCursor* _cursor; ///< cursor being dragged
+       void fake_locate (framepos_t);
+       
        bool _stop; ///< true to stop the transport on starting the drag, otherwise false
-
 };
 
 /** Region fade-in drag */
@@ -565,11 +609,13 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool y_movement_matters () const {
                return false;
        }
+
+       void setup_pointer_frame_offset ();
 };
 
 /** Region fade-out drag */
@@ -581,11 +627,13 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool y_movement_matters () const {
                return false;
        }
+
+       void setup_pointer_frame_offset ();
 };
 
 /** Marker drag */
@@ -598,7 +646,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool allow_vertical_autoscroll () const {
                return false;
@@ -607,13 +655,14 @@ public:
        bool y_movement_matters () const {
                return false;
        }
+
+       void setup_pointer_frame_offset ();
        
 private:
        void update_item (ARDOUR::Location *);
 
        Marker* _marker; ///< marker being dragged
        std::list<ARDOUR::Location*> _copied_locations;
-       ArdourCanvas::Line* _line;
        ArdourCanvas::Points _points;
 };
 
@@ -626,7 +675,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool active (Editing::MouseMode m);
 
@@ -649,7 +698,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool active (Editing::MouseMode) {
                return true;
@@ -674,7 +723,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool active (Editing::MouseMode) {
                return true;
@@ -682,13 +731,13 @@ public:
 
 private:
 
-       ArdourCanvas::SimpleLine* _line;
+       ArdourCanvas::Line* _line;
        AudioRegionView* _arv;
        
        double _region_view_grab_x;
        double _cumulative_x_drag;
        
-       uint32_t _before;
+       float _before;
        uint32_t _max_x;
 };
 
@@ -696,14 +745,14 @@ private:
 class RubberbandSelectDrag : public Drag
 {
 public:
-       RubberbandSelectDrag (Editor *e, ArdourCanvas::Item *i) : Drag (e, i) {}
+       RubberbandSelectDrag (Editor *, ArdourCanvas::Item *);
 
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
-       std::pair<nframes64_t, int> move_threshold () const {
+       std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
                return std::make_pair (8, 1);
        }
 };
@@ -712,24 +761,24 @@ public:
 class TimeFXDrag : public RegionDrag
 {
 public:
-       TimeFXDrag (Editor *e, ArdourCanvas::Item *i, RegionView* p, std::list<RegionView*> const & v) : RegionDrag (e, i, p, v) {}
+       TimeFXDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
 
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 };
 
 /** Scrub drag in audition mode */
 class ScrubDrag : public Drag
 {
 public:
-       ScrubDrag (Editor *e, ArdourCanvas::Item *i) : Drag (e, i) {}
+       ScrubDrag (Editor *, ArdourCanvas::Item *);
 
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 };
 
 /** Drag in range select mode */
@@ -748,7 +797,9 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
+
+       void setup_pointer_frame_offset ();
 
 private:
        Operation _operation;
@@ -773,7 +824,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool allow_vertical_autoscroll () const {
                return false;
@@ -795,12 +846,19 @@ private:
 class MouseZoomDrag : public Drag
 {
 public:
-       MouseZoomDrag (Editor* e, ArdourCanvas::Item *i) : Drag (e, i) {}
+       MouseZoomDrag (Editor *, ArdourCanvas::Item *);
 
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
+
+       std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
+               return std::make_pair (4, 4);
+       }
+
+private:
+       bool _zoom_out;
 };
 
 /** Drag of a range of automation data, changing value but not position */
@@ -812,7 +870,7 @@ public:
        void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
        void motion (GdkEvent *, bool);
        void finished (GdkEvent *, bool);
-       void aborted ();
+       void aborted (bool);
 
        bool x_movement_matters () const {
                return false;
@@ -821,7 +879,17 @@ public:
 private:
        std::list<ARDOUR::AudioRange> _ranges;
        AutomationTimeAxisView* _atav;
-       boost::shared_ptr<AutomationLine> _line;
+
+       /** A line that is part of the drag */
+       struct Line {
+               boost::shared_ptr<AutomationLine> line; ///< the line
+               std::list<ControlPoint*> points; ///< points to drag on the line
+               std::pair<ARDOUR::framepos_t, ARDOUR::framepos_t> range; ///< the range of all points on the line, in session frames
+               XMLNode* state; ///< the XML state node before the drag
+       };
+       
+       std::list<Line> _lines;
+       
        bool _nothing_to_drag;
 };