OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
ossia/dataflow/token_request.hpp
1#pragma once
2#include <ossia/detail/flicks.hpp>
4#include <ossia/editor/scenario/time_signature.hpp>
6
7#include <cassert>
8#include <optional>
9
10namespace ossia
11{
12using quarter_note = double;
13
14struct token_request
15{
16 constexpr token_request() noexcept = default;
17 constexpr token_request(const token_request&) noexcept = default;
18 constexpr token_request(token_request&&) noexcept = default;
19 constexpr token_request& operator=(const token_request&) noexcept = default;
20 constexpr token_request& operator=(token_request&&) noexcept = default;
21
22 constexpr token_request(
23 ossia::time_value prev_d, ossia::time_value d, ossia::time_value parent_duration,
24 ossia::time_value off, double s, time_signature sig, double tempo) noexcept
25 : prev_date{prev_d}
26 , date{d}
27 , parent_duration{parent_duration}
28 , offset{off}
29 , speed{s}
30 , tempo{tempo}
31 , signature{sig}
32 {
33 if(offset.impl < 0)
34 {
35 offset.impl = 0;
36 }
37 }
38
39 [[nodiscard]] constexpr token_request add_offset(ossia::time_value t) const noexcept
40 {
41 token_request other = *this;
42 other.prev_date += t;
43 other.date += t;
44 return other;
45 }
46
47 template <typename Exec, typename Transport>
48 constexpr void loop(
49 ossia::time_value start_offset, ossia::time_value loop_duration, Exec f,
50 Transport transport) const noexcept
51 {
52 ossia::token_request other = *this;
53 ossia::time_value orig_from = other.prev_date;
54 ossia::time_value tick_amount = other.date - other.prev_date;
55
56 while(tick_amount > 0_tv)
57 {
58 const time_value cur_from{orig_from % loop_duration};
59 if(cur_from + tick_amount < loop_duration)
60 {
61 other.prev_date = cur_from + start_offset;
62 other.date = other.prev_date + tick_amount;
63 f(other);
64 break;
65 }
66 else
67 {
68 auto this_tick = loop_duration - cur_from;
69
70 tick_amount -= this_tick;
71 orig_from += this_tick;
72 other.prev_date = cur_from + start_offset;
73 other.date = other.prev_date + this_tick;
74
75 f(other);
76
77 transport(start_offset);
78 other.offset += this_tick;
79 }
80 }
81 }
82
84 [[nodiscard]] constexpr time_value model_read_duration() const noexcept
85 {
86 return date - prev_date;
87 }
88
92 [[nodiscard]] constexpr physical_time
93 start_date_to_physical(double ratio) const noexcept
94 // C++23: [[ expects: speed != 0. ]]
95 {
96 assert(speed != 0.);
97 return this->prev_date.impl * ratio / speed;
98 }
99
101 [[nodiscard]] constexpr physical_time physical_start(double ratio) const noexcept
102 // C++23: [[ expects: speed != 0. ]]
103 {
104 assert(speed != 0.);
105 return this->offset.impl * ratio / speed;
106 }
107
111 [[nodiscard]] constexpr physical_time
112 physical_read_duration(double ratio) const noexcept
113 {
114 return constexpr_ceil(abs(date - prev_date).impl * ratio);
115 }
116
119 [[nodiscard]] constexpr physical_time
120 physical_write_duration(double ratio) const noexcept
121 // C++23: [[ expects: speed != 0. ]]
122 {
123 assert(speed != 0.);
124 return constexpr_ceil(abs(date - prev_date).impl * ratio / speed);
125 }
126
128 [[nodiscard]] constexpr physical_time
129 safe_physical_write_duration(double ratio, int bufferSize) const noexcept
130 // C++23: [[ expects: speed != 0. ]]
131 {
132 assert(speed != 0.);
133 return constexpr_floor(bufferSize - offset.impl * ratio / speed);
134 }
135
137 [[nodiscard]] constexpr bool in_range(ossia::time_value global_time) const noexcept
138 {
139 return global_time.impl >= prev_date.impl && global_time.impl < date.impl;
140 }
141
144 [[nodiscard]] constexpr physical_time
145 to_physical_time_in_tick(ossia::time_value global_time, double ratio) const noexcept
146 {
147 return (global_time - prev_date + offset).impl * ratio / speed;
148 }
149
152 [[nodiscard]] constexpr physical_time
153 to_physical_time_in_tick(int64_t global_time, double ratio) const noexcept
154 {
155 return to_physical_time_in_tick(ossia::time_value{global_time}, ratio);
156 }
157
160 [[nodiscard]] constexpr time_value
161 from_physical_time_in_tick(ossia::physical_time phys_time, double ratio) const noexcept
162 {
163 return time_value{
164 constexpr_floor(phys_time * (speed / ratio) + prev_date.impl - offset.impl)};
165 }
166
169 [[nodiscard]] constexpr double position() const noexcept
170 {
171 return parent_duration.impl > 0 ? date.impl / double(parent_duration.impl) : 0.;
172 }
173
175 [[nodiscard]] constexpr bool forward() const noexcept { return date > prev_date; }
176
178 [[nodiscard]] constexpr bool paused() const noexcept { return date == prev_date; }
179
181 [[nodiscard]] constexpr bool backward() const noexcept { return date < prev_date; }
182
186 [[nodiscard]] constexpr std::optional<time_value>
187 get_quantification_date(double rate) const noexcept
188 {
189 std::optional<time_value> quantification_date;
190
191 if(rate <= 0.)
192 return prev_date;
193
194 const double musical_tick_duration = musical_end_position - musical_start_position;
195 if(musical_tick_duration <= 0.)
196 return prev_date;
197
198 if(rate <= 1.)
199 {
200 // Quantize relative to bars
201 if(musical_end_last_bar != musical_start_last_bar)
202 {
203 // 4 if we're in 4/4 for instance
204 const double musical_bar_duration
205 = musical_end_last_bar - musical_start_last_bar;
206
207 // rate = 0.5 -> 2 bars at 3/4 -> 6 quarter notes
208 const double quantif_duration = musical_bar_duration / rate;
209
210 // we must be on quarter note 6, 12, 18, ... from the previous
211 // signature
212 const double rem = std::fmod(
213 musical_end_last_bar - musical_start_last_signature, quantif_duration);
214 if(rem < 0.0001)
215 {
216 // There is a bar change in this tick and it is when we are going to
217 // trigger
218 const double musical_bar_start = musical_end_last_bar - musical_start_position;
219
220 const double ratio = musical_bar_start / musical_tick_duration;
221 const time_value dt = date - prev_date; // TODO should be tick_offset
222
223 quantification_date = prev_date + dt * ratio;
224 }
225 }
226 }
227 else
228 {
229 // Quantize relative to quarter divisions
230 // TODO ! if there is a bar change,
231 // and no prior quantization date before that, we have to quantize to the
232 // bar change
233 const double start_quarter = (musical_start_position - musical_start_last_bar);
234 const double end_quarter = (musical_end_position - musical_start_last_bar);
235
236 // duration of what we quantify in terms of quarters
237 const double musical_quant_dur = rate / 4.;
238 const double start_quant = std::floor(start_quarter * musical_quant_dur);
239 const double end_quant = std::floor(end_quarter * musical_quant_dur);
240
241 if(start_quant != end_quant)
242 {
243 if(end_quant == end_quarter * musical_quant_dur)
244 {
245 // We want quantization on start, not on end
246 return std::nullopt;
247 }
248 // Date to quantify is the next one :
249 const double musical_tick_duration
250 = musical_end_position - musical_start_position;
251 const double quantified_duration
252 = (musical_start_last_bar + (start_quant + 1) * 4. / rate)
253 - musical_start_position;
254 const double ratio = (date - prev_date).impl / musical_tick_duration;
255
256 quantification_date = prev_date + quantified_duration * ratio;
257 }
258 else if(start_quant == start_quarter * musical_quant_dur)
259 {
260 // We start on a signature change
261 return prev_date;
262 }
263 }
264
265 return quantification_date;
266 }
267
269 [[nodiscard]] constexpr std::optional<physical_time>
270 get_physical_quantification_date(double rate, double modelToSamples) const noexcept
271 {
272 if(auto d = get_quantification_date(rate))
273 return to_physical_time_in_tick(*d, modelToSamples);
274 return {};
275 }
276
277 template <typename Tick, typename Tock>
278 constexpr void
279 metronome(double modelToSamplesRatio, Tick tick, Tock tock) const noexcept
280 {
281 if((musical_end_last_bar != musical_start_last_bar) || musical_start_position == 0.)
282 {
283 // There is a bar change in this tick, start the up tick
284 const double musical_tick_duration = musical_end_position - musical_start_position;
285 if(musical_tick_duration != 0)
286 {
287 const double musical_bar_start = musical_end_last_bar - musical_start_position;
288 const int64_t samples_tick_duration
289 = physical_write_duration(modelToSamplesRatio);
290 if(samples_tick_duration > 0)
291 {
292 const double ratio = musical_bar_start / musical_tick_duration;
293 const int64_t hi_start_sample = samples_tick_duration * ratio;
294 tick(hi_start_sample);
295 }
296 }
297 }
298 else
299 {
300 const int64_t start_quarter
301 = std::floor(musical_start_position - musical_start_last_bar);
302 const int64_t end_quarter
303 = std::floor(musical_end_position - musical_start_last_bar);
304 if(start_quarter != end_quarter)
305 {
306 // There is a quarter change in this tick, start the down tick
307 // start_position is prev_date
308 // end_position is date
309 const double musical_tick_duration
310 = musical_end_position - musical_start_position;
311 if(musical_tick_duration != 0)
312 {
313 const double musical_bar_start
314 = (end_quarter + musical_start_last_bar) - musical_start_position;
315 const int64_t samples_tick_duration
316 = physical_write_duration(modelToSamplesRatio);
317 if(samples_tick_duration > 0)
318 {
319 const double ratio = musical_bar_start / musical_tick_duration;
320 const int64_t lo_start_sample = samples_tick_duration * ratio;
321 tock(lo_start_sample);
322 }
323 }
324 }
325 }
326 }
327
328 [[nodiscard]] constexpr bool unexpected_bar_change() const noexcept
329 {
330 double bar_difference = musical_end_last_bar - musical_start_last_bar;
331 if(bar_difference != 0.)
332 {
333 // If the difference is divisble by the signature,
334 // then the bar change is expected.
335 // e.g. start = 4 -> end = 8 ; signature = 4/4 : good
336 // e.g. start = 4 -> end = 8 ; signature = 6/8 : bad
337 // e.g. start = 4 -> end = 7 ; signature = 6/8 : good
338
339 double quarters_sig = 4. * double(signature.upper) / signature.lower;
340 double div = bar_difference / quarters_sig;
341 bool unexpected = div - int64_t(div) > 0.000001;
342 return unexpected;
343 }
344 return false;
345 }
346
347 constexpr void set_end_time(time_value t) noexcept
348 // C++23: [[ expects: t <= this->date && t > this->prev_date ]]
349 {
350 const auto old_date = date;
351 date = t;
352
353 if(old_date.impl > 0)
354 {
355 double ratio = t.impl / double(old_date.impl);
356 musical_end_position *= ratio;
357 }
358
359 // TODO what if musical_end_position is now before musical_end_last_bar
360 }
361
362 constexpr void set_start_time(time_value t) noexcept
363 // C++23: [[ expects: t <= this->date && t > this->prev_date ]]
364 {
365 const auto old_date = prev_date;
366 prev_date = t;
367
368 if(old_date.impl > 0)
369 {
370 double ratio = t.impl / double(old_date.impl);
371 musical_start_position *= ratio;
372 }
373
374 // TODO what if musical_start_position is now after end_position /
375 // end_last_bar ?
376 }
377
378 ossia::time_value prev_date{}; // Sample we are at
379 ossia::time_value date{}; // Sample we are finishing at
380 ossia::time_value parent_duration{}; // Duration of the parent item of the
381 // one being ticked
382
395 ossia::time_value offset{};
396
397 double speed{1.};
398 double tempo{ossia::root_tempo};
399 time_signature signature{}; // Time signature at start
400
401 ossia::quarter_note musical_start_last_signature{}; // Position of the last bar
402 // signature change in quarter
403 // notes (at prev_date)
404 ossia::quarter_note musical_start_last_bar{}; // Position of the last bar start in
405 // quarter notes (at prev_date)
406 ossia::quarter_note musical_start_position{}; // Current position in quarter notes
407 ossia::quarter_note musical_end_last_bar{}; // Position of the last bar start in
408 // quarter notes (at date)
409 ossia::quarter_note musical_end_position{}; // Current position in quarter notes
410 bool start_discontinuous{};
411 bool end_discontinuous{};
412};
413
414inline bool operator==(const token_request& lhs, const token_request& rhs)
415{
416 return lhs.prev_date == rhs.prev_date && lhs.date == rhs.date
417 && lhs.parent_duration == rhs.parent_duration && lhs.offset == rhs.offset
418 && lhs.speed == rhs.speed && lhs.tempo == rhs.tempo
419 && lhs.signature == rhs.signature
420 && lhs.musical_start_last_bar == rhs.musical_start_last_bar
421 && lhs.musical_start_position == rhs.musical_start_position
422 && lhs.musical_end_last_bar == rhs.musical_end_last_bar
423 && lhs.musical_end_position == rhs.musical_end_position
424 && lhs.start_discontinuous == rhs.start_discontinuous
425 && lhs.end_discontinuous == rhs.end_discontinuous;
426}
427
428inline bool operator!=(const token_request& lhs, const token_request& rhs)
429{
430 return !(lhs == rhs);
431}
432
433// To be used only for simple examples
434struct simple_token_request
435{
436 time_value prev_date{};
437 time_value date{};
438 time_value parent_duration{};
439 time_value offset{};
440
441 operator token_request() const noexcept
442 {
443 return ossia::token_request{prev_date, date, parent_duration, offset, 1.0,
444 {4, 4}, 120.};
445 }
446
447 friend bool operator==(const token_request& lhs, const simple_token_request& self)
448 {
449 return lhs.prev_date == self.prev_date && lhs.date == self.date
450 && lhs.offset == self.offset;
451 }
452 friend bool operator==(const simple_token_request& self, const token_request& rhs)
453 {
454 return rhs == self;
455 }
456 friend bool operator!=(const token_request& lhs, const simple_token_request& self)
457 {
458 return !(lhs == self);
459 }
460 friend bool operator!=(const simple_token_request& self, const token_request& rhs)
461 {
462 return !(rhs == self);
463 }
464};
465}
Definition git_info.h:7
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30