OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
dataflow/nodes/midi.hpp
1#pragma once
2#include <ossia/dataflow/graph_node.hpp>
3#include <ossia/dataflow/node_process.hpp>
4#include <ossia/dataflow/port.hpp>
5#include <ossia/detail/flat_multiset.hpp>
6
7#include <libremidi/ump_events.hpp>
8
9namespace ossia::nodes
10{
11
12using midi_size_t = uint8_t;
13struct note_data
14{
15 time_value start{};
16 time_value duration{};
17
18 midi_size_t pitch{};
19 midi_size_t velocity{};
20};
21
22struct note_comparator
23{
24 using is_transparent = std::true_type;
25 bool operator()(const note_data& lhs, const note_data& rhs) const
26 {
27 return lhs.start < rhs.start;
28 }
29 bool operator()(const note_data& lhs, int64_t rhs) const
30 {
31 return lhs.start.impl < rhs;
32 }
33};
34
35class midi final : public ossia::nonowning_graph_node
36{
37 ossia::midi_outlet midi_out;
38
39public:
40 using note_set = ossia::flat_multiset<note_data, note_comparator>;
41 explicit midi(int64_t notes)
42 {
43 m_outlets.push_back(&midi_out);
44 int64_t to_reserve = std::max(notes * 1.1, 128.);
45 m_notes.reserve(to_reserve);
46 m_orig_notes.reserve(to_reserve);
47 m_playing_notes.reserve(to_reserve);
48 m_to_stop.reserve(64);
49 }
50
51 ~midi() override = default;
52
53 std::string label() const noexcept override { return "midi"; }
54
55 void set_channel(int c) { m_channel = c - 1; }
56
57 void add_note(note_data nd)
58 {
59 m_orig_notes.insert(nd);
60 if(nd.start > m_prev_date)
61 {
62 m_notes.insert(nd);
63 }
64 }
65
66 void remove_note(note_data nd)
67 {
68 m_orig_notes.erase(nd);
69 m_notes.erase(nd);
70 auto it = m_playing_notes.find(nd);
71 if(it != m_playing_notes.end())
72 {
73 m_to_stop.insert(nd);
74 m_playing_notes.erase(it);
75 }
76 }
77
78 void replace_notes(note_set&& notes)
79 {
80 for(auto& note : m_playing_notes)
81 m_to_stop.insert(note);
82 m_playing_notes.clear();
83
84 using namespace std;
85 swap(m_orig_notes, notes);
86 m_notes.clear();
87
88 auto start_it = m_orig_notes.lower_bound(m_prev_date.impl);
89 if(start_it != m_orig_notes.end())
90 {
91 m_notes.tree().get_sequence_ref().assign(start_it, m_orig_notes.end());
92 }
93 }
94
95 void transport(ossia::time_value date)
96 {
97 requestTransport = true;
98 m_transport_date = date;
99 }
100
101 void transport_impl(ossia::time_value date)
102 {
103 // 1. Send note-offs
104 m_to_stop.insert(m_playing_notes.begin(), m_playing_notes.end());
105 m_playing_notes.clear();
106
107 // 2. Re-add following notes
108 if(date < m_prev_date)
109 {
110 if(date == 0_tv)
111 {
112 m_notes = m_orig_notes;
113 }
114 else
115 {
116 auto min_it = m_orig_notes.lower_bound({date});
117 auto max_it = m_orig_notes.lower_bound({m_prev_date});
118
119 if(min_it != m_orig_notes.end())
120 m_notes.insert(min_it, max_it);
121
122 // all of these will have it->start < date
123 for(auto it = m_orig_notes.begin(); it != min_it; ++it)
124 {
125 if((it->start + it->duration) > date)
126 {
127 m_notes.insert(*it);
128 }
129 }
130 }
131 }
132 else if(date > m_prev_date)
133 {
134 // remove previous notes
135 auto min_it = m_notes.lower_bound({date});
136 if(min_it != m_notes.begin() && min_it != m_notes.end())
137 {
138 std::advance(min_it, -1);
139 m_notes.erase(m_notes.begin(), min_it);
140 }
141 // todo resume current notes
142 }
143
144 m_prev_date = date;
145 }
146
147 void update_note(note_data oldNote, note_data newNote)
148 {
149 // OPTIMIZEME
150 remove_note(oldNote);
151 add_note(newNote);
152 }
153
154 void set_notes(note_set&& notes)
155 {
156 m_notes = std::move(notes);
157 m_orig_notes = m_notes;
158
159 auto max_it = m_notes.lower_bound({m_prev_date});
160 if(max_it != m_notes.begin()) // TODO handle the begin case correctly
161 m_notes.erase(m_notes.begin(), max_it);
162 }
163
164 bool mustStop{};
165 bool requestTransport{};
166 bool doTransport{};
167
168private:
169 void run(const ossia::token_request& t, ossia::exec_state_facade e) noexcept override
170 {
171 struct scope_guard
172 {
173 midi& self;
174 const ossia::token_request& t;
175 ~scope_guard()
176 {
177 self.m_prev_date = t.date;
178
179 if(self.requestTransport)
180 {
181 self.transport_impl(self.m_transport_date);
182 self.requestTransport = false;
183 }
184 }
185 } guard{*this, t};
186
187 ossia::midi_port& mp = *midi_out;
188 const auto samplesratio = e.modelToSamples();
189 const auto tick_start = t.physical_start(samplesratio);
190
191 if(t.end_discontinuous)
192 {
193 auto& mess = mp.messages;
194 for(auto note : m_playing_notes)
195 {
196 mess.push_back(libremidi::from_midi1::note_off(m_channel, note.pitch, 0));
197 mess.back().timestamp = 0;
198 }
199 for(auto note : m_to_stop)
200 {
201 mess.push_back(libremidi::from_midi1::note_off(m_channel, note.pitch, 0));
202 mess.back().timestamp = 0;
203 }
204 m_playing_notes.clear();
205 m_to_stop.clear();
206 return;
207 }
208
209 for(const note_data& note : m_to_stop)
210 {
211 mp.messages.push_back(libremidi::from_midi1::note_off(m_channel, note.pitch, 0));
212 mp.messages.back().timestamp = tick_start;
213 }
214 m_to_stop.clear();
215
216 if(mustStop)
217 {
218 for(auto& note : m_playing_notes)
219 {
220 mp.messages.push_back(libremidi::from_midi1::note_off(m_channel, note.pitch, 0));
221 mp.messages.back().timestamp = tick_start;
222 }
223
224 m_notes = m_orig_notes;
225 m_playing_notes.clear();
226
227 mustStop = false;
228 }
229 else
230 {
231 if(m_notes.empty() && m_playing_notes.empty())
232 return;
233 if(doTransport)
234 {
235 auto it = m_notes.begin();
236
237 while(it != m_notes.end() && it->start < t.date)
238 {
239 auto& note = *it;
240 mp.messages.push_back(
241 libremidi::from_midi1::note_on(m_channel, note.pitch, note.velocity));
242 mp.messages.back().timestamp = tick_start;
243 m_playing_notes.insert(note);
244 it = m_notes.erase(it);
245 }
246
247 doTransport = false;
248 }
249
250 if(t.forward())
251 {
252 // First send note offs
253 for(auto it = m_playing_notes.begin(); it != m_playing_notes.end();)
254 {
255 note_data& note = const_cast<note_data&>(*it);
256 auto end_time = note.start + note.duration;
257
258 if(t.in_range({end_time}))
259 {
260 mp.messages.push_back(
261 libremidi::from_midi1::note_off(m_channel, note.pitch, 0));
262 mp.messages.back().timestamp
263 = t.to_physical_time_in_tick(end_time, samplesratio);
264
265 it = m_playing_notes.erase(it);
266 }
267 else
268 {
269 ++it;
270 }
271 }
272
273 // Look for all the messages
274 auto max_it = m_notes.lower_bound({t.date});
275 for(auto it = m_notes.begin(); it < max_it;)
276 {
277 note_data& note = const_cast<note_data&>(*it);
278 auto start_time = note.start;
279 if(start_time >= t.prev_date && start_time < t.date)
280 {
281 // Send note_on
282 mp.messages.push_back(
283 libremidi::from_midi1::note_on(m_channel, note.pitch, note.velocity));
284 mp.messages.back().timestamp
285 = t.to_physical_time_in_tick(start_time, samplesratio);
286
287 m_playing_notes.insert(note);
288 it = m_notes.erase(it);
289 max_it = std::lower_bound(
290 it, m_notes.end(), t.date.impl + 1, note_comparator{});
291 }
292 else
293 {
294 ++it;
295 }
296 }
297 }
298 }
299 }
300
301 note_set m_notes;
302 note_set m_orig_notes;
303 note_set m_playing_notes;
304 note_set m_to_stop;
305 time_value m_prev_date{};
306 time_value m_transport_date{};
307
308 int m_channel{};
309};
310
311class midi_node_process final : public ossia::node_process
312{
313public:
314 using ossia::node_process::node_process;
315
316 void transport_impl(ossia::time_value date) override
317 {
318 midi& n = *static_cast<midi*>(node.get());
319 n.transport(date);
320 }
321
322 void stop() override
323 {
324 midi& n = *static_cast<midi*>(node.get());
325 n.request(ossia::token_request{});
326 n.mustStop = true;
327 }
328};
329}
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30