add basics of a method to flush MidiStateTracker state to a MidiBuffer
When the buffer is delivered to a receiver, the receiver should (theoretically) end up the same state as the MidiStateTracker
This commit is contained in:
parent
04004d2178
commit
885d2feb6a
@ -54,6 +54,8 @@ public:
|
||||
void resolve_notes (Evoral::EventSink<samplepos_t>& buffer, samplepos_t time);
|
||||
void resolve_notes (MidiSource& src, const Glib::Threads::Mutex::Lock& lock, Temporal::Beats time);
|
||||
|
||||
void flush_notes (MidiBuffer&, samplepos_t);
|
||||
|
||||
bool empty() const { return _on == 0; }
|
||||
uint16_t on() const { return _on; }
|
||||
bool active (uint8_t note, uint8_t channel) {
|
||||
@ -80,7 +82,7 @@ class LIBARDOUR_API MidiStateTracker : public MidiNoteTracker
|
||||
void dump (std::ostream&);
|
||||
void reset ();
|
||||
|
||||
void resolve (MidiBuffer&, samplepos_t);
|
||||
void flush (MidiBuffer&, samplepos_t);
|
||||
|
||||
private:
|
||||
uint8_t have_program[16];
|
||||
|
@ -228,10 +228,6 @@ MidiStateTracker::reset ()
|
||||
|
||||
MidiNoteTracker::reset ();
|
||||
|
||||
for (size_t n = 0; n < n_channels; ++n) {
|
||||
have_program[n] = 0;
|
||||
}
|
||||
|
||||
for (size_t n = 0; n < n_channels; ++n) {
|
||||
program[n] = 0;
|
||||
}
|
||||
@ -259,6 +255,7 @@ MidiStateTracker::track (const uint8_t* evbuf)
|
||||
case MIDI_CTL_ALL_NOTES_OFF:
|
||||
MidiNoteTracker::reset();
|
||||
break;
|
||||
|
||||
case MIDI_CMD_NOTE_ON:
|
||||
add (evbuf[1], chan);
|
||||
break;
|
||||
@ -267,16 +264,15 @@ MidiStateTracker::track (const uint8_t* evbuf)
|
||||
break;
|
||||
|
||||
case MIDI_CMD_CONTROL:
|
||||
control[chan][evbuf[1]] = evbuf[2];
|
||||
control[chan][evbuf[1]] = 0xf|evbuf[2];
|
||||
break;
|
||||
|
||||
case MIDI_CMD_PGM_CHANGE:
|
||||
program[chan] = evbuf[1];
|
||||
have_program[chan] = 1;
|
||||
program[chan] = 0xf|evbuf[1];
|
||||
break;
|
||||
|
||||
case MIDI_CMD_CHANNEL_PRESSURE:
|
||||
pressure[chan] = evbuf[1];
|
||||
pressure[chan] = 0xf|evbuf[1];
|
||||
break;
|
||||
|
||||
case MIDI_CMD_NOTE_PRESSURE:
|
||||
@ -290,12 +286,39 @@ MidiStateTracker::track (const uint8_t* evbuf)
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
MidiStateTracker::resolve (MidiBuffer& buffer, samplepos_t time)
|
||||
MidiStateTracker::flush (MidiBuffer& dst, samplepos_t time)
|
||||
{
|
||||
/* XXX implement me */
|
||||
|
||||
uint8_t buf[3];
|
||||
const size_t n_channels = 16;
|
||||
const size_t n_controls = 127;
|
||||
|
||||
/* XXX need MidiNoteTracker::flush() method that will emit NoteOn for
|
||||
* all currently-on notes.
|
||||
*/
|
||||
|
||||
for (int chn = 0; chn < n_channels; ++chn) {
|
||||
if (program[chn] & 0xf) {
|
||||
buf[0] = MIDI_CMD_PGM_CHANGE|chn;
|
||||
buf[1] = program[chn] & 0x7f;
|
||||
dst.write (time, Evoral::MIDI_EVENT, 2, buf);
|
||||
}
|
||||
}
|
||||
|
||||
for (int chn = 0; chn < 16; ++chn) {
|
||||
for (int ctl = 0; ctl < n_controls; ++ctl) {
|
||||
if (control[chn][ctl] & 0xf) {
|
||||
buf[0] = MIDI_CMD_CONTROL|chn;
|
||||
buf[1] = ctl;
|
||||
buf[2] = control[chn][ctl] & 0x7f;
|
||||
dst.write (time, Evoral::MIDI_EVENT, 3, buf);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user