#include <gdk/gdk.h>
#include <stdint.h>
+#include "ardour/tempo.h"
#include "ardour/types.h"
+#include "canvas/types.h"
+
+#include "cursor_context.h"
#include "editor_items.h"
+#include "mouse_cursors.h"
+#include "editing.h"
namespace ARDOUR {
class Location;
+ class TempoSection;
+}
+
+namespace ArdourCanvas {
+ class Item;
+ class Line;
+ class Rectangle;
}
namespace PBD {
class MidiTimeAxisView;
class Drag;
class NoteBase;
+class RegionView;
+class TimeAxisView;
+class RouteTimeAxisView;
+class RegionSelection;
+class MidiRegionView;
+class MeterMarker;
+class ArdourMarker;
+class TempoMarker;
+class ControlPoint;
+class AudioRegionView;
+class AutomationLine;
+class AutomationTimeAxisView;
/** Class to manage current drags */
class DragManager
void abort ();
void add (Drag *);
- void set (Drag *, GdkEvent *, Gdk::Cursor* c = 0);
- void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
+ void set (Drag *, GdkEvent *, Gdk::Cursor* c = MouseCursors::invalid_cursor());
+ void start_grab (GdkEvent *, Gdk::Cursor* c = MouseCursors::invalid_cursor());
bool end_grab (GdkEvent *);
bool have_item (ArdourCanvas::Item *) const;
* @return true if this drag should happen in this mouse mode.
*/
virtual bool active (Editing::MouseMode m) {
- return (m != Editing::MouseGain);
+ return true;
}
/** @return minimum number of frames (in x) and pixels (in y) that should be considered a movement */
return true;
}
+ bool initially_vertical() const {
+ return _initially_vertical;
+ }
+
/** Set up the _pointer_frame_offset */
virtual void setup_pointer_frame_offset () {
_pointer_frame_offset = 0;
return _last_pointer_y;
}
- double last_pointer_frame () const {
+ ARDOUR::framepos_t last_pointer_frame () const {
return _last_pointer_frame;
}
+ ARDOUR::frameoffset_t snap_delta (guint const) const;
+
+ double current_pointer_x () const;
double current_pointer_y () const;
- boost::shared_ptr<ARDOUR::Region> add_midi_region (MidiTimeAxisView*);
+ /* sets snap delta from unsnapped pos */
+ void setup_snap_delta (framepos_t pos);
+
+ boost::shared_ptr<ARDOUR::Region> add_midi_region (MidiTimeAxisView*, bool commit);
void show_verbose_cursor_time (framepos_t);
void show_verbose_cursor_duration (framepos_t, framepos_t, double xoffset = 0);
private:
bool _trackview_only; ///< true if pointer y value should always be relative to the top of the trackview group
bool _move_threshold_passed; ///< true if the move threshold has been passed, otherwise false
+ bool _starting_point_passed; ///< true if we called move () with first_move flag, otherwise false
+ bool _initially_vertical; ///< true if after move threshold is passed we appear to be moving vertically; undefined before that
bool _was_double_click; ///< true if drag initiated by a double click event
double _grab_x; ///< trackview x of the grab start position
double _grab_y; ///< y of the grab start position, possibly adjusted if _trackview_only is true
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
+
+ /* difference between some key position's snapped and unsnapped
+ * framepos. used for relative snap.
+ */
+ ARDOUR::frameoffset_t _snap_delta;
+ CursorContext::Handle _cursor_ctx; ///< cursor change context
};
class RegionDrag;
/** a list of the non-hidden TimeAxisViews sorted by editor order key */
std::vector<TimeAxisView*> _time_axis_views;
int find_time_axis_view (TimeAxisView *) const;
+ int apply_track_delta (const int start, const int delta, const int skip, const bool distance_only = false) const;
int _visible_y_low;
int _visible_y_high;
+ uint32_t _ntracks;
friend class DraggingView;
protected:
double compute_x_delta (GdkEvent const *, ARDOUR::framepos_t *);
- bool y_movement_allowed (int, double) const;
+ virtual bool y_movement_allowed (int, double, int skip_invisible = 0) const;
bool _brushing;
+ bool _ignore_video_lock;
ARDOUR::framepos_t _last_frame_position; ///< last position of the thing being dragged
double _total_x_delta;
int _last_pointer_time_axis_view;
double _last_pointer_layer;
+private:
+ uint32_t _ndropzone;
+ uint32_t _pdropzone;
+ uint32_t _ddropzone;
};
void setup_pointer_frame_offset ();
-private:
+protected:
typedef std::set<boost::shared_ptr<ARDOUR::Playlist> > PlaylistSet;
+ void add_stateful_diff_commands_for_playlists (PlaylistSet const &);
+private:
void finished_no_copy (
bool const,
bool const,
PlaylistSet& modified_playlists
);
- void add_stateful_diff_commands_for_playlists (PlaylistSet const &);
void collect_new_region_view (RegionView *);
RouteTimeAxisView* create_destination_time_axis (boost::shared_ptr<ARDOUR::Region>, TimeAxisView* original);
void aborted (bool);
};
+/** Region drag in ripple mode */
+
+class RegionRippleDrag : public RegionMoveDrag
+{
+public:
+ RegionRippleDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
+ ~RegionRippleDrag () { delete exclude; }
+
+ void motion (GdkEvent *, bool);
+ void finished (GdkEvent *, bool);
+ void aborted (bool);
+protected:
+ bool y_movement_allowed (int delta_track, double delta_layer, int skip_invisible = 0) const;
+
+private:
+ TimeAxisView *prev_tav; // where regions were most recently dragged from
+ TimeAxisView *orig_tav; // where drag started
+ ARDOUR::framecnt_t prev_amount;
+ ARDOUR::framepos_t prev_position;
+ ARDOUR::framecnt_t selection_length;
+ bool allow_moves_across_tracks; // only if all selected regions are on one track
+ ARDOUR::RegionList *exclude;
+ void add_all_after_to_views (TimeAxisView *tav, ARDOUR::framepos_t where, const RegionSelection &exclude, bool drag_in_progress);
+ void remove_unselected_from_views (ARDOUR::framecnt_t amount, bool move_regions);
+
+};
+
+/** "Drag" to cut a region (action only on button release) */
+class RegionCutDrag : public Drag
+{
+ public:
+ RegionCutDrag (Editor*, ArdourCanvas::Item*, framepos_t);
+ ~RegionCutDrag ();
+
+ void motion (GdkEvent*, bool);
+ void finished (GdkEvent*, bool);
+ void aborted (bool);
+
+ private:
+ EditorCursor* line;
+};
+
/** Drags to create regions */
class RegionCreateDrag : public Drag
{
MidiRegionView* region;
bool relative;
bool at_front;
+ bool _was_selected;
+ double _snap_delta;
};
/** Drags to move MIDI notes */
private:
- ARDOUR::frameoffset_t total_dx () const;
+ ARDOUR::frameoffset_t total_dx (const guint) const;
int8_t total_dy () const;
MidiRegionView* _region;
NoteBase* _primary;
double _cumulative_dx;
double _cumulative_dy;
- bool _was_selected;
+ bool _was_selected;
double _note_height;
};
void finished (GdkEvent *, bool);
void aborted (bool);
+ bool active (Editing::MouseMode mode) {
+ return mode == Editing::MouseDraw || mode == Editing::MouseContent;
+ }
+
+ bool y_movement_matters () const {
+ return false;
+ }
+
private:
double y_to_region (double) const;
- framecnt_t grid_frames (framepos_t) const;
-
+ ARDOUR::framecnt_t grid_frames (framepos_t) const;
+
MidiRegionView* _region_view;
ArdourCanvas::Rectangle* _drag_rect;
framepos_t _note[2];
private:
ARDOUR::frameoffset_t _startdrag_video_offset;
ARDOUR::frameoffset_t _max_backwards_drag;
+ bool _stuck;
};
/** Drag to trim region(s) */
private:
Operation _operation;
-
+
bool _preserve_fade_anchor;
bool _jump_position_when_done;
};
private:
MeterMarker* _marker;
+ ARDOUR::MeterSection* _real_section;
+
bool _copy;
+ Editing::SnapType _old_snap_type;
+ Editing::SnapMode _old_snap_mode;
XMLNode* before_state;
};
}
bool y_movement_matters () const {
- return false;
+ return true;
}
void setup_pointer_frame_offset ();
private:
TempoMarker* _marker;
+ ARDOUR::TempoSection* _real_section;
+
bool _copy;
+ bool _movable;
XMLNode* before_state;
};
+/** BBT Ruler drag */
+class BBTRulerDrag : public Drag
+{
+public:
+ BBTRulerDrag (Editor *, ArdourCanvas::Item *);
+
+ void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
+ void motion (GdkEvent *, bool);
+ void finished (GdkEvent *, bool);
+ void aborted (bool);
+
+ bool allow_vertical_autoscroll () const {
+ return false;
+ }
+
+ bool y_movement_matters () const {
+ return false;
+ }
+
+ void setup_pointer_frame_offset ();
+
+private:
+ double _pulse;
+ ARDOUR::TempoSection* _tempo;
+ XMLNode* before_state;
+};
/** Drag of the playhead cursor */
class CursorDrag : public Drag
void finished (GdkEvent *, bool);
void aborted (bool);
- bool active (Editing::MouseMode) {
- return true;
- }
-
bool allow_vertical_autoscroll () const {
return false;
}
private:
void update_item (ARDOUR::Location *);
- Marker* _marker; ///< marker being dragged
-
+ ArdourMarker* _marker; ///< marker being dragged
+ bool _selection_changed;
struct CopiedLocationMarkerInfo {
ARDOUR::Location* location;
- std::vector<Marker*> markers;
+ std::vector<ArdourMarker*> markers;
bool move_both;
- CopiedLocationMarkerInfo (ARDOUR::Location* l, Marker* m);
+ CopiedLocationMarkerInfo (ARDOUR::Location* l, ArdourMarker* m);
};
typedef std::list<CopiedLocationMarkerInfo> CopiedLocationInfo;
void finished (GdkEvent *, bool);
void aborted (bool);
- bool active (Editing::MouseMode) {
- return true;
- }
-
private:
AutomationLine* _line;
double _fixed_grab_x;
double _fixed_grab_y;
+ double _cumulative_y_drag;
uint32_t _before;
uint32_t _after;
- double _cumulative_y_drag;
};
/** Transient feature line drags*/
void finished (GdkEvent *, bool);
void aborted (bool);
- bool active (Editing::MouseMode) {
- return true;
- }
-
private:
ArdourCanvas::Line* _line;
* @param drag_in_progress true if the drag is currently happening.
*/
virtual void select_things (int button_state, framepos_t x1, framepos_t x2, double y1, double y2, bool drag_in_progress) = 0;
-
+
virtual void deselect_things () = 0;
protected:
private:
Operation _operation;
bool _add;
- bool _extend;
- int _original_pointer_time_axis;
- int _last_pointer_time_axis;
std::list<TimeAxisView*> _added_time_axes;
bool _time_selection_at_start;
framepos_t start_at_start;
{
public:
enum Operation {
+ CreateSkipMarker,
CreateRangeMarker,
CreateTransportMarker,
CreateCDMarker
};
RangeMarkerBarDrag (Editor *, ArdourCanvas::Item *, Operation);
+ ~RangeMarkerBarDrag ();
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
{
public:
AutomationRangeDrag (Editor *, AutomationTimeAxisView *, std::list<ARDOUR::AudioRange> const &);
- AutomationRangeDrag (Editor *, AudioRegionView *, std::list<ARDOUR::AudioRange> const &);
+ AutomationRangeDrag (Editor *, RegionView *, std::list<ARDOUR::AudioRange> const &);
void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
void motion (GdkEvent *, bool);
return false;
}
- bool active (Editing::MouseMode) {
- return true;
- }
-
private:
void setup (std::list<boost::shared_ptr<AutomationLine> > const &);
- double y_fraction (boost::shared_ptr<AutomationLine>, double global_y_position) const;
+ double y_fraction (boost::shared_ptr<AutomationLine>, double global_y_position) const;
+ double value (boost::shared_ptr<ARDOUR::AutomationList> list, double x) const;
std::list<ARDOUR::AudioRange> _ranges;
};
std::list<Line> _lines;
- double y_origin;
- bool _nothing_to_drag;
+ double _y_origin;
+ bool _nothing_to_drag;
+ bool _integral;
};
/** Drag of one edge of an xfade
void motion (GdkEvent*, bool);
void finished (GdkEvent*, bool);
void aborted (bool);
-
+
bool y_movement_matters () const {
return false;
}