Declick before the end of seamless loops, not after the end, so that loops are render...
[ardour.git] / libs / ardour / ardour / session_event.h
1 #ifndef __ardour_session_event_h__
2 #define __ardour_session_event_h__
3
4 #include <list>
5 #include <boost/function.hpp>
6 #include <boost/shared_ptr.hpp>
7
8 #include "pbd/pool.h"
9 #include "pbd/ringbuffer.h"
10 #include "pbd/event_loop.h"
11
12 #include "ardour/types.h"
13
14 namespace ARDOUR {
15
16 class Slave;
17 class Region;
18
19 class SessionEvent {
20 public:
21         enum Type {
22                 SetTransportSpeed,
23                 SetTrackSpeed,
24                 Locate,
25                 LocateRoll,
26                 LocateRollLocate,
27                 SetLoop,
28                 PunchIn,
29                 PunchOut,
30                 RangeStop,
31                 RangeLocate,
32                 Overwrite,
33                 SetSyncSource,
34                 Audition,
35                 InputConfigurationChange,
36                 SetPlayAudioRange,
37                 RealTimeOperation,
38                 AdjustPlaybackBuffering,
39                 AdjustCaptureBuffering,
40                 SetTimecodeTransmission,
41
42                 /* only one of each of these events can be queued at any one time */
43
44                 StopOnce,
45                 AutoLoop,
46                 AutoLoopDeclick,
47         };
48
49         enum Action {
50                 Add,
51                 Remove,
52                 Replace,
53                 Clear
54         };
55
56         Type       type;
57         Action     action;
58         framepos_t action_frame;
59         framepos_t target_frame;
60         double     speed;
61
62         union {
63                 void*        ptr;
64                 bool         yes_or_no;
65                 framepos_t  target2_frame;
66                 Slave*       slave;
67                 Route*       route;
68         };
69
70         union {
71                 bool second_yes_or_no;
72         };
73
74         /* 4 members to handle a multi-group event handled in RT context */
75
76         typedef boost::function<void (SessionEvent*)> RTeventCallback;
77
78         boost::shared_ptr<RouteList> routes;    /* apply to */
79         boost::function<void (void)> rt_slot;   /* what to call in RT context */
80         RTeventCallback              rt_return; /* called after rt_slot, with this event as an argument */
81         PBD::EventLoop*              event_loop;
82
83         std::list<AudioRange> audio_range;
84         std::list<MusicRange> music_range;
85
86         boost::shared_ptr<Region> region;
87
88         SessionEvent (Type t, Action a, framepos_t when, framepos_t where, double spd, bool yn = false, bool yn2 = false)
89                 : type (t)
90                 , action (a)
91                 , action_frame (when)
92                 , target_frame (where)
93                 , speed (spd)
94                 , yes_or_no (yn)
95                 , second_yes_or_no (yn2)
96                 , event_loop (0) {}
97
98         void set_ptr (void* p) {
99                 ptr = p;
100         }
101
102         bool before (const SessionEvent& other) const {
103                 return action_frame < other.action_frame;
104         }
105
106         bool after (const SessionEvent& other) const {
107                 return action_frame > other.action_frame;
108         }
109
110         static bool compare (const SessionEvent *e1, const SessionEvent *e2) {
111                 return e1->before (*e2);
112         }
113
114         void* operator new (size_t);
115         void  operator delete (void *ptr, size_t /*size*/);
116
117         static const framepos_t Immediate = 0;
118
119         static void create_per_thread_pool (const std::string& n, uint32_t nitems);
120         static void init_event_pool ();
121
122 private:
123         static PerThreadPool* pool;
124         CrossThreadPool* own_pool;
125
126         friend class Butler;
127 };
128
129 class SessionEventManager {
130 public:
131         SessionEventManager () : pending_events (2048),
132                                  auto_loop_event(0), punch_out_event(0), punch_in_event(0) {}
133         virtual ~SessionEventManager() {}
134
135         virtual void queue_event (SessionEvent *ev) = 0;
136         void clear_events (SessionEvent::Type type);
137
138 protected:
139         RingBuffer<SessionEvent*> pending_events;
140         typedef std::list<SessionEvent *> Events;
141         Events           events;
142         Events           immediate_events;
143         Events::iterator next_event;
144
145         /* there can only ever be one of each of these */
146
147         SessionEvent *auto_loop_event;
148         SessionEvent *punch_out_event;
149         SessionEvent *punch_in_event;
150
151         void dump_events () const;
152         void merge_event (SessionEvent*);
153         void replace_event (SessionEvent::Type, framepos_t action_frame, framepos_t target = 0);
154         bool _replace_event (SessionEvent*);
155         bool _remove_event (SessionEvent *);
156         void _clear_event_type (SessionEvent::Type);
157
158         void add_event (framepos_t action_frame, SessionEvent::Type type, framepos_t target_frame = 0);
159         void remove_event (framepos_t frame, SessionEvent::Type type);
160
161         virtual void process_event(SessionEvent*) = 0;
162         virtual void set_next_event () = 0;
163 };
164
165 } /* namespace */
166
167 #endif /* __ardour_session_event_h__ */