class Editor;
class EditorCursor;
class TimeAxisView;
+class Drag;
-/** Abstract base class for dragging of things within the editor */
-class Drag
+/** Class to manage current drags */
+class DragManager
{
-
public:
- Drag (Editor *, ArdourCanvas::Item *);
- virtual ~Drag () {}
- /** @return the canvas item being dragged */
- ArdourCanvas::Item* item () const {
- return _item;
- }
+ DragManager (Editor* e);
+ ~DragManager ();
- void swap_grab (ArdourCanvas::Item *, Gdk::Cursor *, uint32_t);
- void break_drag ();
+ bool motion_handler (GdkEvent *, bool);
- bool motion_handler (GdkEvent*, bool);
+ void abort ();
+ void break_drag ();
+ void add (Drag *);
+ void set (Drag *, GdkEvent *, Gdk::Cursor* c = 0);
+ void start_grab (GdkEvent *);
+ bool end_grab (GdkEvent *);
+ bool have_item (ArdourCanvas::Item *) const;
- /** @return true if an end drag is in progress */
+ /** @return true if an end drag or break_drag is in progress */
bool ending () const {
return _ending;
}
+ bool active () const {
+ return !_drags.empty ();
+ }
+
/** @return current pointer x position in trackview coordinates */
double current_pointer_x () const {
return _current_pointer_x;
return _current_pointer_y;
}
+ /** @return current pointer frame */
+ nframes64_t current_pointer_frame () const {
+ return _current_pointer_frame;
+ }
+
+private:
+ Editor* _editor;
+ std::list<Drag*> _drags;
+ bool _ending; ///< true if end_grab or break_drag 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
+};
+
+/** Abstract base class for dragging of things within the editor */
+class Drag
+{
+public:
+ Drag (Editor *, ArdourCanvas::Item *);
+ virtual ~Drag () {}
+
+ void set_manager (DragManager* m) {
+ _drags = m;
+ }
+
+ /** @return the canvas item being dragged */
+ ArdourCanvas::Item* item () const {
+ return _item;
+ }
+
+ void swap_grab (ArdourCanvas::Item *, Gdk::Cursor *, uint32_t);
+ bool motion_handler (GdkEvent*, bool);
+ void break_drag ();
+
nframes64_t adjusted_frame (nframes64_t, GdkEvent const *, bool snap = true) const;
nframes64_t adjusted_current_frame (GdkEvent const *, bool snap = true) const;
*/
virtual void finished (GdkEvent* e, bool m) = 0;
+ /** Called to abort a drag and return things to how
+ * they were before it started.
+ */
+ virtual void aborted () = 0;
+
/** @param m Mouse mode.
* @return true if this drag should happen in this mouse mode.
*/
}
Editor* _editor; ///< our editor
+ 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;
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
- bool _have_transaction; ///< true if a transaction has been started, false otherwise. Must be set true by derived class.
private:
- bool _ending; ///< true if end_grab is in progress, otherwise false
bool _move_threshold_passed; ///< true if the move threshold has been passed, otherwise false
- double _original_x; ///< original world x of the thing being dragged
- double _original_y; ///< original world y of the thing being dragged
double _grab_x; ///< trackview x of the grab start position
double _grab_y; ///< trackview y of the grab start position
- double _current_pointer_x; ///< trackview x of the current pointer
- double _current_pointer_y; ///< trackview y of the current pointer
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
- nframes64_t _current_pointer_frame; ///< frame that the pointer is now at
};
virtual void start_grab (GdkEvent *, Gdk::Cursor *);
virtual void motion (GdkEvent *, bool);
virtual void finished (GdkEvent *, bool) = 0;
+ virtual void aborted ();
protected:
struct TimeAxisViewSummary {
bool check_possible (RouteTimeAxisView **, ARDOUR::layer_t *);
bool _brushing;
nframes64_t _last_frame_position; ///< last position of the thing being dragged
+ double _total_x_delta;
};
virtual void start_grab (GdkEvent *, Gdk::Cursor *);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
std::pair<nframes64_t, int> move_threshold () const {
return std::make_pair (4, 4);
RegionInsertDrag (Editor *, boost::shared_ptr<ARDOUR::Region>, RouteTimeAxisView*, nframes64_t);
void finished (GdkEvent *, bool);
+ void aborted ();
};
/** Region drag in splice mode */
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
};
/** Drags to create regions */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
private:
TimeAxisView* _view;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
private:
MidiRegionView* region;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
private:
MidiRegionView* region;
bool active (Editing::MouseMode m) {
return (m == Editing::MouseGain);
}
+
+ void aborted ();
};
/** Drag to trim region(s) */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool y_movement_matters () const {
return false;
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 */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool allow_vertical_autoscroll () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool allow_vertical_autoscroll () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool active (Editing::MouseMode) {
return true;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool y_movement_matters () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool y_movement_matters () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool allow_vertical_autoscroll () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool active (Editing::MouseMode m);
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool active (Editing::MouseMode) {
return true;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
+
+ std::pair<nframes64_t, int> move_threshold () const {
+ return std::make_pair (8, 1);
+ }
};
/** Region drag in time-FX mode */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
};
/** Scrub drag in audition mode */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
};
/** Drag in range select mode */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
private:
Operation _operation;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
+
+ bool allow_vertical_autoscroll () const {
+ return false;
+ }
bool y_movement_matters () const {
return false;
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
};
/** Drag of a range of automation data, changing value but not position */
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
void finished (GdkEvent *, bool);
+ void aborted ();
bool x_movement_matters () const {
return false;