Merge branch 'master' into windows
[ardour.git] / gtk2_ardour / editor_drag.h
1 /*
2     Copyright (C) 2009 Paul Davis
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #ifndef __gtk2_ardour_editor_drag_h_
21 #define __gtk2_ardour_editor_drag_h_
22
23 #include <list>
24
25 #include <gdk/gdk.h>
26 #include <stdint.h>
27
28 #include "ardour/types.h"
29
30 #include "canvas.h"
31 #include "editor_items.h"
32
33 namespace ARDOUR {
34         class Location;
35 }
36
37 namespace PBD {
38         class StatefulDiffCommand;
39 }
40
41 namespace Gnome {
42         namespace Canvas {
43                 class CanvasNoteEvent;
44                 class CanvasPatchChange;
45         }
46 }
47
48 class Editor;
49 class EditorCursor;
50 class TimeAxisView;
51 class MidiTimeAxisView;
52 class Drag;
53
54 /** Class to manage current drags */
55 class DragManager
56 {
57 public:
58
59         DragManager (Editor* e);
60         ~DragManager ();
61
62         bool motion_handler (GdkEvent *, bool);
63
64         void abort ();
65         void add (Drag *);
66         void set (Drag *, GdkEvent *, Gdk::Cursor* c = 0);
67         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
68         bool end_grab (GdkEvent *);
69         bool have_item (ArdourCanvas::Item *) const;
70
71         /** @return true if an end drag or abort is in progress */
72         bool ending () const {
73                 return _ending;
74         }
75
76         bool active () const {
77                 return !_drags.empty ();
78         }
79
80         /** @return current pointer x position in trackview coordinates */
81         double current_pointer_x () const {
82                 return _current_pointer_x;
83         }
84
85         /** @return current pointer y position in trackview coordinates */
86         double current_pointer_y () const {
87                 return _current_pointer_y;
88         }
89
90         /** @return current pointer frame */
91         ARDOUR::framepos_t current_pointer_frame () const {
92                 return _current_pointer_frame;
93         }
94
95 private:
96         Editor* _editor;
97         std::list<Drag*> _drags;
98         bool _ending; ///< true if end_grab or abort is in progress, otherwise false
99         double _current_pointer_x; ///< trackview x of the current pointer
100         double _current_pointer_y; ///< trackview y of the current pointer
101         ARDOUR::framepos_t _current_pointer_frame; ///< frame that the pointer is now at
102         bool _old_follow_playhead; ///< state of Editor::follow_playhead() before the drags started
103 };
104
105 /** Abstract base class for dragging of things within the editor */
106 class Drag
107 {
108 public:
109         Drag (Editor *, ArdourCanvas::Item *);
110         virtual ~Drag () {}
111
112         void set_manager (DragManager* m) {
113                 _drags = m;
114         }
115
116         /** @return the canvas item being dragged */
117         ArdourCanvas::Item* item () const {
118                 return _item;
119         }
120
121         void swap_grab (ArdourCanvas::Item *, Gdk::Cursor *, uint32_t);
122         bool motion_handler (GdkEvent*, bool);
123         void abort ();
124
125         ARDOUR::framepos_t adjusted_frame (ARDOUR::framepos_t, GdkEvent const *, bool snap = true) const;
126         ARDOUR::framepos_t adjusted_current_frame (GdkEvent const *, bool snap = true) const;
127
128         /** Called to start a grab of an item.
129          *  @param e Event that caused the grab to start.
130          *  @param c Cursor to use, or 0.
131          */
132         virtual void start_grab (GdkEvent* e, Gdk::Cursor* c = 0);
133
134         virtual bool end_grab (GdkEvent *);
135
136         /** Called when a drag motion has occurred.
137          *  @param e Event describing the motion.
138          *  @param f true if this is the first movement, otherwise false.
139          */
140         virtual void motion (GdkEvent* e, bool f) = 0;
141
142         /** Called when a drag has finished.
143          *  @param e Event describing the finish.
144          *  @param m true if some movement occurred, otherwise false.
145          */
146         virtual void finished (GdkEvent* e, bool m) = 0;
147
148         /** Called to abort a drag and return things to how
149          *  they were before it started.
150          *  @param m true if some movement occurred, otherwise false.
151          */
152         virtual void aborted (bool m) = 0;
153
154         /** @param m Mouse mode.
155          *  @return true if this drag should happen in this mouse mode.
156          */
157         virtual bool active (Editing::MouseMode m) {
158                 return (m != Editing::MouseGain);
159         }
160
161         /** @return minimum number of frames (in x) and pixels (in y) that should be considered a movement */
162         virtual std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
163                 return std::make_pair (1, 1);
164         }
165
166         virtual bool allow_vertical_autoscroll () const {
167                 return true;
168         }
169
170         /** @return true if x movement matters to this drag */
171         virtual bool x_movement_matters () const {
172                 return true;
173         }
174
175         /** @return true if y movement matters to this drag */
176         virtual bool y_movement_matters () const {
177                 return true;
178         }
179
180         /** Set up the _pointer_frame_offset */
181         virtual void setup_pointer_frame_offset () {
182                 _pointer_frame_offset = 0;
183         }
184
185 protected:
186
187         double grab_x () const {
188                 return _grab_x;
189         }
190
191         double grab_y () const {
192                 return _grab_y;
193         }
194
195         ARDOUR::framepos_t raw_grab_frame () const {
196                 return _raw_grab_frame;
197         }
198
199         ARDOUR::framepos_t grab_frame () const {
200                 return _grab_frame;
201         }
202
203         double last_pointer_x () const {
204                 return _last_pointer_x;
205         }
206
207         double last_pointer_y () const {
208                 return _last_pointer_y;
209         }
210
211         double last_pointer_frame () const {
212                 return _last_pointer_frame;
213         }
214
215         boost::shared_ptr<ARDOUR::Region> add_midi_region (MidiTimeAxisView*);
216
217         void show_verbose_cursor_time (framepos_t);
218         void show_verbose_cursor_duration (framepos_t, framepos_t, double xoffset = 0);
219         void show_verbose_cursor_text (std::string const &);
220
221         Editor* _editor; ///< our editor
222         DragManager* _drags;
223         ArdourCanvas::Item* _item; ///< our item
224         /** Offset from the mouse's position for the drag to the start of the thing that is being dragged */
225         ARDOUR::framecnt_t _pointer_frame_offset;
226         bool _x_constrained; ///< true if x motion is constrained, otherwise false
227         bool _y_constrained; ///< true if y motion is constrained, otherwise false
228         bool _was_rolling; ///< true if the session was rolling before the drag started, otherwise false
229
230 private:
231
232         bool _move_threshold_passed; ///< true if the move threshold has been passed, otherwise false
233         double _grab_x; ///< trackview x of the grab start position
234         double _grab_y; ///< trackview y of the grab start position
235         double _last_pointer_x; ///< trackview x of the pointer last time a motion occurred
236         double _last_pointer_y; ///< trackview y of the pointer last time a motion occurred
237         ARDOUR::framepos_t _raw_grab_frame; ///< unsnapped frame that the mouse was at when start_grab was called, or 0
238         ARDOUR::framepos_t _grab_frame; ///< adjusted_frame that the mouse was at when start_grab was called, or 0
239         ARDOUR::framepos_t _last_pointer_frame; ///< adjusted_frame the last time a motion occurred
240 };
241
242 class RegionDrag;
243
244 /** Container for details about a region being dragged */
245 class DraggingView
246 {
247 public:
248         DraggingView (RegionView *, RegionDrag *);
249
250         RegionView* view; ///< the view
251         /** index into RegionDrag::_time_axis_views of the view that this region is currently being displayed on,
252          *  or -1 if it is not visible.
253          */
254         int time_axis_view;
255         /** layer that this region is currently being displayed on.  This is a double
256             rather than a layer_t as we use fractional layers during drags to allow the user
257             to indicate a new layer to put a region on.
258         */
259         double layer;
260         double initial_y; ///< the initial y position of the view before any reparenting
261         framepos_t initial_position; ///< initial position of the region
262         framepos_t initial_end; ///< initial end position of the region
263         boost::shared_ptr<ARDOUR::Playlist> initial_playlist;
264 };
265
266 /** Abstract base class for drags that involve region(s) */
267 class RegionDrag : public Drag, public sigc::trackable
268 {
269 public:
270         RegionDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
271         virtual ~RegionDrag () {}
272
273 protected:
274
275         RegionView* _primary; ///< the view that was clicked on (or whatever) to start the drag
276         std::list<DraggingView> _views; ///< information about all views that are being dragged
277
278         /** a list of the non-hidden TimeAxisViews sorted by editor order key */
279         std::vector<TimeAxisView*> _time_axis_views;
280         int find_time_axis_view (TimeAxisView *) const;
281
282         int _visible_y_low;
283         int _visible_y_high;
284
285         friend class DraggingView;
286
287 private:
288
289         void region_going_away (RegionView *);
290         PBD::ScopedConnection death_connection;
291 };
292
293
294 /** Drags involving region motion from somewhere */
295 class RegionMotionDrag : public RegionDrag
296 {
297 public:
298
299         RegionMotionDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &, bool);
300         virtual ~RegionMotionDrag () {}
301
302         virtual void start_grab (GdkEvent *, Gdk::Cursor *);
303         virtual void motion (GdkEvent *, bool);
304         virtual void finished (GdkEvent *, bool);
305         virtual void aborted (bool);
306
307         /** @return true if the regions being `moved' came from somewhere on the canvas;
308          *  false if they came from outside (e.g. from the region list).
309          */
310         virtual bool regions_came_from_canvas () const = 0;
311
312 protected:
313
314         double compute_x_delta (GdkEvent const *, ARDOUR::framepos_t *);
315         bool y_movement_allowed (int, double) const;
316
317         bool _brushing;
318         ARDOUR::framepos_t _last_frame_position; ///< last position of the thing being dragged
319         double _total_x_delta;
320         int _last_pointer_time_axis_view;
321         double _last_pointer_layer;
322 };
323
324
325 /** Drags to move (or copy) regions that are already shown in the GUI to
326  *  somewhere different.
327  */
328 class RegionMoveDrag : public RegionMotionDrag
329 {
330 public:
331         RegionMoveDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &, bool, bool);
332         virtual ~RegionMoveDrag () {}
333
334         void motion (GdkEvent *, bool);
335         void finished (GdkEvent *, bool);
336         void aborted (bool);
337
338         bool regions_came_from_canvas () const {
339                 return true;
340         }
341
342         std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
343                 return std::make_pair (4, 4);
344         }
345
346         void setup_pointer_frame_offset ();
347
348 private:
349         typedef std::set<boost::shared_ptr<ARDOUR::Playlist> > PlaylistSet;
350
351         void finished_no_copy (
352                 bool const,
353                 bool const,
354                 ARDOUR::framecnt_t const
355                 );
356
357         void finished_copy (
358                 bool const,
359                 bool const,
360                 ARDOUR::framecnt_t const
361                 );
362
363         RegionView* insert_region_into_playlist (
364                 boost::shared_ptr<ARDOUR::Region>,
365                 RouteTimeAxisView*,
366                 ARDOUR::layer_t,
367                 ARDOUR::framecnt_t,
368                 PlaylistSet&
369                 );
370
371         void remove_region_from_playlist (
372                 boost::shared_ptr<ARDOUR::Region>,
373                 boost::shared_ptr<ARDOUR::Playlist>,
374                 PlaylistSet& modified_playlists
375                 );
376
377         void add_stateful_diff_commands_for_playlists (PlaylistSet const &);
378
379         void collect_new_region_view (RegionView *);
380
381         bool _copy;
382         RegionView* _new_region_view;
383 };
384
385 /** Drag to insert a region from somewhere */
386 class RegionInsertDrag : public RegionMotionDrag
387 {
388 public:
389         RegionInsertDrag (Editor *, boost::shared_ptr<ARDOUR::Region>, RouteTimeAxisView*, ARDOUR::framepos_t);
390
391         void finished (GdkEvent *, bool);
392         void aborted (bool);
393
394         bool regions_came_from_canvas () const {
395                 return false;
396         }
397 };
398
399 /** Region drag in splice mode */
400 class RegionSpliceDrag : public RegionMoveDrag
401 {
402 public:
403         RegionSpliceDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
404
405         void motion (GdkEvent *, bool);
406         void finished (GdkEvent *, bool);
407         void aborted (bool);
408 };
409
410 /** Drags to create regions */
411 class RegionCreateDrag : public Drag
412 {
413 public:
414         RegionCreateDrag (Editor *, ArdourCanvas::Item *, TimeAxisView *);
415
416         void motion (GdkEvent *, bool);
417         void finished (GdkEvent *, bool);
418         void aborted (bool);
419
420 private:
421         MidiTimeAxisView* _view;
422         boost::shared_ptr<ARDOUR::Region> _region;
423 };
424
425 /** Drags to resize MIDI notes */
426 class NoteResizeDrag : public Drag
427 {
428 public:
429         NoteResizeDrag (Editor *, ArdourCanvas::Item *);
430
431         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
432         void motion (GdkEvent *, bool);
433         void finished (GdkEvent *, bool);
434         void aborted (bool);
435
436 private:
437         MidiRegionView*     region;
438         bool                relative;
439         bool                at_front;
440 };
441
442 /** Drags to move MIDI notes */
443 class NoteDrag : public Drag
444 {
445   public:
446         NoteDrag (Editor*, ArdourCanvas::Item*);
447
448         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
449         void motion (GdkEvent *, bool);
450         void finished (GdkEvent *, bool);
451         void aborted (bool);
452
453   private:
454
455         ARDOUR::frameoffset_t total_dx () const;
456         int8_t total_dy () const;
457
458         MidiRegionView* _region;
459         Gnome::Canvas::CanvasNoteEvent* _primary;
460         double _cumulative_dx;
461         double _cumulative_dy;
462         bool _was_selected;
463         double _note_height;
464 };
465
466 class NoteCreateDrag : public Drag
467 {
468 public:
469         NoteCreateDrag (Editor *, ArdourCanvas::Item *, MidiRegionView *);
470         ~NoteCreateDrag ();
471
472         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
473         void motion (GdkEvent *, bool);
474         void finished (GdkEvent *, bool);
475         void aborted (bool);
476
477 private:
478         double y_to_region (double) const;
479         framecnt_t grid_frames (framepos_t) const;
480         
481         MidiRegionView* _region_view;
482         ArdourCanvas::SimpleRect* _drag_rect;
483         framepos_t _note[2];
484 };
485
486 /** Drag to move MIDI patch changes */
487 class PatchChangeDrag : public Drag
488 {
489 public:
490         PatchChangeDrag (Editor *, ArdourCanvas::CanvasPatchChange *, MidiRegionView *);
491
492         void motion (GdkEvent *, bool);
493         void finished (GdkEvent *, bool);
494         void aborted (bool);
495
496         bool y_movement_matters () const {
497                 return false;
498         }
499
500         void setup_pointer_frame_offset ();
501
502 private:
503         MidiRegionView* _region_view;
504         ArdourCanvas::CanvasPatchChange* _patch_change;
505         double _cumulative_dx;
506 };
507
508 /** Container for details about audio regions being dragged along with video */
509 class AVDraggingView
510 {
511 public:
512         AVDraggingView (RegionView *);
513
514         RegionView* view; ///< the view
515         framepos_t initial_position; ///< initial position of the region
516 };
517
518 /** Drag of video offset */
519 class VideoTimeLineDrag : public Drag
520 {
521 public:
522         VideoTimeLineDrag (Editor *e, ArdourCanvas::Item *i);
523
524         void motion (GdkEvent *, bool);
525         void finished (GdkEvent *, bool);
526         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
527
528         bool y_movement_matters () const {
529                 return false;
530         }
531
532         bool allow_vertical_autoscroll () const {
533                 return false;
534         }
535
536         void aborted (bool);
537
538 protected:
539         std::list<AVDraggingView> _views; ///< information about all audio that are being dragged along
540
541 private:
542         ARDOUR::frameoffset_t _startdrag_video_offset;
543         ARDOUR::frameoffset_t _max_backwards_drag;
544 };
545
546 /** Drag to trim region(s) */
547 class TrimDrag : public RegionDrag
548 {
549 public:
550         enum Operation {
551                 StartTrim,
552                 EndTrim,
553                 ContentsTrim,
554         };
555
556         TrimDrag (Editor *, ArdourCanvas::Item *, RegionView*, std::list<RegionView*> const &, bool preserve_fade_anchor = false);
557
558         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
559         void motion (GdkEvent *, bool);
560         void finished (GdkEvent *, bool);
561         void aborted (bool);
562
563         bool y_movement_matters () const {
564                 return false;
565         }
566
567         void setup_pointer_frame_offset ();
568
569 private:
570
571         Operation _operation;
572         
573         bool _preserve_fade_anchor;
574 };
575
576 /** Meter marker drag */
577 class MeterMarkerDrag : public Drag
578 {
579 public:
580         MeterMarkerDrag (Editor *, ArdourCanvas::Item *, bool);
581
582         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
583         void motion (GdkEvent *, bool);
584         void finished (GdkEvent *, bool);
585         void aborted (bool);
586
587         bool allow_vertical_autoscroll () const {
588                 return false;
589         }
590
591         bool y_movement_matters () const {
592                 return false;
593         }
594
595         void setup_pointer_frame_offset ();
596
597 private:
598         MeterMarker* _marker;
599         bool _copy;
600         XMLNode* before_state;
601 };
602
603 /** Tempo marker drag */
604 class TempoMarkerDrag : public Drag
605 {
606 public:
607         TempoMarkerDrag (Editor *, ArdourCanvas::Item *, bool);
608
609         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
610         void motion (GdkEvent *, bool);
611         void finished (GdkEvent *, bool);
612         void aborted (bool);
613
614         bool allow_vertical_autoscroll () const {
615                 return false;
616         }
617
618         bool y_movement_matters () const {
619                 return false;
620         }
621
622         void setup_pointer_frame_offset ();
623
624 private:
625         TempoMarker* _marker;
626         bool _copy;
627         XMLNode* before_state;
628 };
629
630
631 /** Drag of the playhead cursor */
632 class CursorDrag : public Drag
633 {
634 public:
635         CursorDrag (Editor *, ArdourCanvas::Item *, bool);
636
637         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
638         void motion (GdkEvent *, bool);
639         void finished (GdkEvent *, bool);
640         void aborted (bool);
641
642         bool active (Editing::MouseMode) {
643                 return true;
644         }
645
646         bool allow_vertical_autoscroll () const {
647                 return false;
648         }
649
650         bool y_movement_matters () const {
651                 return true;
652         }
653
654 private:
655         void fake_locate (framepos_t);
656
657         bool _stop; ///< true to stop the transport on starting the drag, otherwise false
658         double _grab_zoom; ///< editor frames per unit when our grab started
659 };
660
661 /** Region fade-in drag */
662 class FadeInDrag : public RegionDrag
663 {
664 public:
665         FadeInDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
666
667         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
668         void motion (GdkEvent *, bool);
669         void finished (GdkEvent *, bool);
670         void aborted (bool);
671
672         bool y_movement_matters () const {
673                 return false;
674         }
675
676         void setup_pointer_frame_offset ();
677 };
678
679 /** Region fade-out drag */
680 class FadeOutDrag : public RegionDrag
681 {
682 public:
683         FadeOutDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
684
685         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
686         void motion (GdkEvent *, bool);
687         void finished (GdkEvent *, bool);
688         void aborted (bool);
689
690         bool y_movement_matters () const {
691                 return false;
692         }
693
694         void setup_pointer_frame_offset ();
695 };
696
697 /** Marker drag */
698 class MarkerDrag : public Drag
699 {
700 public:
701         MarkerDrag (Editor *, ArdourCanvas::Item *);
702         ~MarkerDrag ();
703
704         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
705         void motion (GdkEvent *, bool);
706         void finished (GdkEvent *, bool);
707         void aborted (bool);
708
709         bool allow_vertical_autoscroll () const {
710                 return false;
711         }
712
713         bool y_movement_matters () const {
714                 return false;
715         }
716
717         void setup_pointer_frame_offset ();
718
719 private:
720         void update_item (ARDOUR::Location *);
721
722         Marker* _marker; ///< marker being dragged
723
724         struct CopiedLocationMarkerInfo {
725             ARDOUR::Location* location;
726             std::vector<Marker*> markers;
727             bool    move_both;
728             CopiedLocationMarkerInfo (ARDOUR::Location* l, Marker* m);
729         };
730
731         typedef std::list<CopiedLocationMarkerInfo> CopiedLocationInfo;
732         CopiedLocationInfo _copied_locations;
733         ArdourCanvas::Points _points;
734 };
735
736 /** Control point drag */
737 class ControlPointDrag : public Drag
738 {
739 public:
740         ControlPointDrag (Editor *, ArdourCanvas::Item *);
741
742         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
743         void motion (GdkEvent *, bool);
744         void finished (GdkEvent *, bool);
745         void aborted (bool);
746
747         bool active (Editing::MouseMode m);
748
749 private:
750
751         ControlPoint* _point;
752         double _fixed_grab_x;
753         double _fixed_grab_y;
754         double _cumulative_x_drag;
755         double _cumulative_y_drag;
756         bool     _pushing;
757         uint32_t _final_index;
758         static double _zero_gain_fraction;
759 };
760
761 /** Gain or automation line drag */
762 class LineDrag : public Drag
763 {
764 public:
765         LineDrag (Editor *e, ArdourCanvas::Item *i);
766
767         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
768         void motion (GdkEvent *, bool);
769         void finished (GdkEvent *, bool);
770         void aborted (bool);
771
772         bool active (Editing::MouseMode) {
773                 return true;
774         }
775
776 private:
777
778         AutomationLine* _line;
779         double _fixed_grab_x;
780         double _fixed_grab_y;
781         uint32_t _before;
782         uint32_t _after;
783         double _cumulative_y_drag;
784 };
785
786 /** Transient feature line drags*/
787 class FeatureLineDrag : public Drag
788 {
789 public:
790         FeatureLineDrag (Editor *e, ArdourCanvas::Item *i);
791
792         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
793         void motion (GdkEvent *, bool);
794         void finished (GdkEvent *, bool);
795         void aborted (bool);
796
797         bool active (Editing::MouseMode) {
798                 return true;
799         }
800
801 private:
802
803         ArdourCanvas::Line* _line;
804         AudioRegionView* _arv;
805
806         double _region_view_grab_x;
807         double _cumulative_x_drag;
808
809         float _before;
810         uint32_t _max_x;
811 };
812
813 /** Dragging of a rubberband rectangle for selecting things */
814 class RubberbandSelectDrag : public Drag
815 {
816 public:
817         RubberbandSelectDrag (Editor *, ArdourCanvas::Item *);
818
819         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
820         void motion (GdkEvent *, bool);
821         void finished (GdkEvent *, bool);
822         void aborted (bool);
823
824         std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
825                 return std::make_pair (8, 1);
826         }
827
828         void do_select_things (GdkEvent *, bool);
829
830         /** Select some things within a rectangle.
831          *  @param button_state The button state from the GdkEvent.
832          *  @param x1 The left-hand side of the rectangle in session frames.
833          *  @param x2 The right-hand side of the rectangle in session frames.
834          *  @param y1 The top of the rectangle in trackview coordinates.
835          *  @param y2 The bottom of the rectangle in trackview coordinates.
836          *  @param drag_in_progress true if the drag is currently happening.
837          */
838         virtual void select_things (int button_state, framepos_t x1, framepos_t x2, double y1, double y2, bool drag_in_progress) = 0;
839         
840         virtual void deselect_things () = 0;
841
842   protected:
843         bool _vertical_only;
844 };
845
846 /** A general editor RubberbandSelectDrag (for regions, automation points etc.) */
847 class EditorRubberbandSelectDrag : public RubberbandSelectDrag
848 {
849 public:
850         EditorRubberbandSelectDrag (Editor *, ArdourCanvas::Item *);
851
852         void select_things (int, framepos_t, framepos_t, double, double, bool);
853         void deselect_things ();
854 };
855
856 /** A RubberbandSelectDrag for selecting MIDI notes */
857 class MidiRubberbandSelectDrag : public RubberbandSelectDrag
858 {
859 public:
860         MidiRubberbandSelectDrag (Editor *, MidiRegionView *);
861
862         void select_things (int, framepos_t, framepos_t, double, double, bool);
863         void deselect_things ();
864
865 private:
866         MidiRegionView* _region_view;
867 };
868
869 /** A RubberbandSelectDrag for selecting MIDI notes but with no horizonal component */
870 class MidiVerticalSelectDrag : public RubberbandSelectDrag
871 {
872 public:
873         MidiVerticalSelectDrag (Editor *, MidiRegionView *);
874
875         void select_things (int, framepos_t, framepos_t, double, double, bool);
876         void deselect_things ();
877
878 private:
879         MidiRegionView* _region_view;
880 };
881
882 /** Region drag in time-FX mode */
883 class TimeFXDrag : public RegionDrag
884 {
885 public:
886         TimeFXDrag (Editor *, ArdourCanvas::Item *, RegionView *, std::list<RegionView*> const &);
887
888         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
889         void motion (GdkEvent *, bool);
890         void finished (GdkEvent *, bool);
891         void aborted (bool);
892 };
893
894 /** Scrub drag in audition mode */
895 class ScrubDrag : public Drag
896 {
897 public:
898         ScrubDrag (Editor *, ArdourCanvas::Item *);
899
900         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
901         void motion (GdkEvent *, bool);
902         void finished (GdkEvent *, bool);
903         void aborted (bool);
904 };
905
906 /** Drag in range select mode */
907 class SelectionDrag : public Drag
908 {
909 public:
910         enum Operation {
911                 CreateSelection,
912                 SelectionStartTrim,
913                 SelectionEndTrim,
914                 SelectionMove,
915                 SelectionExtend
916         };
917
918         SelectionDrag (Editor *, ArdourCanvas::Item *, Operation);
919
920         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
921         void motion (GdkEvent *, bool);
922         void finished (GdkEvent *, bool);
923         void aborted (bool);
924
925         void setup_pointer_frame_offset ();
926
927 private:
928         Operation _operation;
929         bool _add;
930         bool _extend;
931         int _original_pointer_time_axis;
932         int _last_pointer_time_axis;
933         std::list<TimeAxisView*> _added_time_axes;
934         bool _time_selection_at_start;
935         framepos_t start_at_start;
936         framepos_t end_at_start;
937 };
938
939 /** Range marker drag */
940 class RangeMarkerBarDrag : public Drag
941 {
942 public:
943         enum Operation {
944                 CreateRangeMarker,
945                 CreateTransportMarker,
946                 CreateCDMarker
947         };
948
949         RangeMarkerBarDrag (Editor *, ArdourCanvas::Item *, Operation);
950
951         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
952         void motion (GdkEvent *, bool);
953         void finished (GdkEvent *, bool);
954         void aborted (bool);
955
956         bool allow_vertical_autoscroll () const {
957                 return false;
958         }
959
960         bool y_movement_matters () const {
961                 return false;
962         }
963
964 private:
965         void update_item (ARDOUR::Location *);
966
967         Operation _operation;
968         ArdourCanvas::SimpleRect* _drag_rect;
969         bool _copy;
970 };
971
972 /** Drag of rectangle to set zoom */
973 class MouseZoomDrag : public Drag
974 {
975 public:
976         MouseZoomDrag (Editor *, ArdourCanvas::Item *);
977
978         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
979         void motion (GdkEvent *, bool);
980         void finished (GdkEvent *, bool);
981         void aborted (bool);
982
983         std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
984                 return std::make_pair (4, 4);
985         }
986
987 private:
988         bool _zoom_out;
989 };
990
991 /** Drag of a range of automation data (either on an automation track or region gain),
992  *  changing value but not position.
993  */
994 class AutomationRangeDrag : public Drag
995 {
996 public:
997         AutomationRangeDrag (Editor *, AutomationTimeAxisView *, std::list<ARDOUR::AudioRange> const &);
998         AutomationRangeDrag (Editor *, AudioRegionView *, std::list<ARDOUR::AudioRange> const &);
999
1000         void start_grab (GdkEvent *, Gdk::Cursor* c = 0);
1001         void motion (GdkEvent *, bool);
1002         void finished (GdkEvent *, bool);
1003         void aborted (bool);
1004
1005         bool x_movement_matters () const {
1006                 return false;
1007         }
1008
1009         bool active (Editing::MouseMode) {
1010                 return true;
1011         }
1012
1013 private:
1014         void setup (std::list<boost::shared_ptr<AutomationLine> > const &);
1015         double y_fraction (boost::shared_ptr<AutomationLine>, double global_y_position) const;
1016
1017         std::list<ARDOUR::AudioRange> _ranges;
1018
1019         /** A line that is part of the drag */
1020         struct Line {
1021                 boost::shared_ptr<AutomationLine> line; ///< the line
1022                 std::list<ControlPoint*> points; ///< points to drag on the line
1023                 std::pair<ARDOUR::framepos_t, ARDOUR::framepos_t> range; ///< the range of all points on the line, in session frames
1024                 XMLNode* state; ///< the XML state node before the drag
1025                 double original_fraction; ///< initial y-fraction before the drag
1026         };
1027
1028         std::list<Line> _lines;
1029         double y_origin;
1030         bool _nothing_to_drag;
1031 };
1032
1033 /** Drag of one edge of an xfade
1034  */
1035 class CrossfadeEdgeDrag : public Drag
1036 {
1037   public:
1038         CrossfadeEdgeDrag (Editor*, AudioRegionView*, ArdourCanvas::Item*, bool start);
1039
1040         void start_grab (GdkEvent*, Gdk::Cursor* c = 0);
1041         void motion (GdkEvent*, bool);
1042         void finished (GdkEvent*, bool);
1043         void aborted (bool);
1044         
1045         bool y_movement_matters () const {
1046                 return false;
1047         }
1048
1049         virtual std::pair<ARDOUR::framecnt_t, int> move_threshold () const {
1050                 return std::make_pair (4, 4);
1051         }
1052
1053   private:
1054         AudioRegionView* arv;
1055         bool start;
1056 };
1057
1058 #endif /* __gtk2_ardour_editor_drag_h_ */
1059