almost done with the new recorder

This commit is contained in:
AlexandreRouma 2022-11-23 10:22:29 +01:00
parent 34b0577f3b
commit f97ca9ac05
2 changed files with 84 additions and 10 deletions

View File

@ -50,6 +50,10 @@ public:
return keys.size(); return keys.size();
} }
bool empty() {
return keys.empty();
}
bool keyExists(K key) { bool keyExists(K key) {
if (std::find(keys.begin(), keys.end(), key) != keys.end()) { return true; } if (std::find(keys.begin(), keys.end(), key) != keys.end()) { return true; }
return false; return false;

View File

@ -42,7 +42,7 @@ public:
// Define option lists // Define option lists
formats.define("WAV", wav::FORMAT_WAV); formats.define("WAV", wav::FORMAT_WAV);
formats.define("RF64", wav::FORMAT_RF64); // formats.define("RF64", wav::FORMAT_RF64); // Disabled for now
sampleDepths.define(wav::SAMP_DEPTH_8BIT, "8-Bit", wav::SAMP_DEPTH_8BIT); sampleDepths.define(wav::SAMP_DEPTH_8BIT, "8-Bit", wav::SAMP_DEPTH_8BIT);
sampleDepths.define(wav::SAMP_DEPTH_16BIT, "16-Bit", wav::SAMP_DEPTH_16BIT); sampleDepths.define(wav::SAMP_DEPTH_16BIT, "16-Bit", wav::SAMP_DEPTH_16BIT);
sampleDepths.define(wav::SAMP_DEPTH_32BIT, "32-Bit", wav::SAMP_DEPTH_32BIT); sampleDepths.define(wav::SAMP_DEPTH_32BIT, "32-Bit", wav::SAMP_DEPTH_32BIT);
@ -98,6 +98,22 @@ public:
} }
void postInit() { void postInit() {
// Enumerate streams
audioStreams.clear();
auto names = sigpath::sinkManager.getStreamNames();
for (const auto& name : names) {
audioStreams.define(name, name, name);
}
// Bind stream register/unregister handlers
onStreamRegisteredHandler.ctx = this;
onStreamRegisteredHandler.handler = streamRegisteredHandler;
sigpath::sinkManager.onStreamRegistered.bindHandler(&onStreamRegisteredHandler);
onStreamUnregisterHandler.ctx = this;
onStreamUnregisterHandler.handler = streamUnregisterHandler;
sigpath::sinkManager.onStreamUnregister.bindHandler(&onStreamUnregisterHandler);
// Select the stream
selectStream(selectedStreamName); selectStream(selectedStreamName);
} }
@ -138,10 +154,8 @@ public:
} }
// Open audio stream or baseband // Open audio stream or baseband
// TODO: DO NOT HARDCODE THE STREAM NAME
if (recMode == RECORDER_MODE_AUDIO) { if (recMode == RECORDER_MODE_AUDIO) {
// TODO: HAS TO BE DONE PROPERLY stereoStream = sigpath::sinkManager.bindStream(selectedStreamName);
stereoStream = sigpath::sinkManager.bindStream("Radio");
stereoSink.setInput(stereoStream); stereoSink.setInput(stereoStream);
stereoSink.start(); stereoSink.start();
} }
@ -164,7 +178,7 @@ public:
if (recMode == RECORDER_MODE_AUDIO) { if (recMode == RECORDER_MODE_AUDIO) {
// TODO: HAS TO BE DONE PROPERLY // TODO: HAS TO BE DONE PROPERLY
stereoSink.stop(); stereoSink.stop();
sigpath::sinkManager.unbindStream("Radio", stereoStream); sigpath::sinkManager.unbindStream(selectedStreamName, stereoStream);
} }
else { else {
// Unbind and destroy IQ stream // Unbind and destroy IQ stream
@ -232,6 +246,15 @@ private:
// Show additional audio options // Show additional audio options
if (_this->recMode == RECORDER_MODE_AUDIO) { if (_this->recMode == RECORDER_MODE_AUDIO) {
ImGui::LeftLabel("Stream");
ImGui::FillWidth();
if (ImGui::Combo(CONCAT("##_recorder_stream_", _this->name), &_this->streamId, _this->audioStreams.txt)) {
_this->selectStream(_this->audioStreams.value(_this->streamId));
config.acquire();
config.conf[_this->name]["audioStream"] = _this->audioStreams.key(_this->streamId);
config.release(true);
}
_this->updateAudioMeter(_this->audioLvl); _this->updateAudioMeter(_this->audioLvl);
ImGui::FillWidth(); ImGui::FillWidth();
ImGui::VolumeMeter(_this->audioLvl.l, _this->audioLvl.l, -60, 10); ImGui::VolumeMeter(_this->audioLvl.l, _this->audioLvl.l, -60, 10);
@ -240,13 +263,11 @@ private:
ImGui::FillWidth(); ImGui::FillWidth();
if (ImGui::SliderFloat(CONCAT("##_recorder_vol_", _this->name), &_this->audioVolume, 0, 1, "")) { if (ImGui::SliderFloat(CONCAT("##_recorder_vol_", _this->name), &_this->audioVolume, 0, 1, "")) {
// TODO: ADD VOLUME CONTROL
_this->volume.setVolume(_this->audioVolume); _this->volume.setVolume(_this->audioVolume);
config.acquire(); config.acquire();
config.conf[_this->name]["audioVolume"] = _this->audioVolume; config.conf[_this->name]["audioVolume"] = _this->audioVolume;
config.release(true); config.release(true);
} }
//ImGui::PopItemWidth();
if (_this->recording) { style::beginDisabled(); } if (_this->recording) { style::beginDisabled(); }
if (ImGui::Checkbox(CONCAT("Stereo##_recorder_stereo_", _this->name), &_this->stereo)) { if (ImGui::Checkbox(CONCAT("Stereo##_recorder_stereo_", _this->name), &_this->stereo)) {
@ -286,20 +307,34 @@ private:
void selectStream(std::string name) { void selectStream(std::string name) {
std::lock_guard<std::recursive_mutex> lck(recMtx); std::lock_guard<std::recursive_mutex> lck(recMtx);
deselectStream(); deselectStream();
if (audioStreams.empty()) {
selectedStreamName.clear();
return;
}
else if (!audioStreams.keyExists(name)) {
selectStream(audioStreams.key(0));
return;
}
audioStream = sigpath::sinkManager.bindStream(name); audioStream = sigpath::sinkManager.bindStream(name);
if (!audioStream) { return; } if (!audioStream) { return; }
selectedStreamName = name; selectedStreamName = name;
streamId = audioStreams.keyId(name);
volume.setInput(audioStream); volume.setInput(audioStream);
startAudioPath(); startAudioPath();
} }
void deselectStream() { void deselectStream() {
std::lock_guard<std::recursive_mutex> lck(recMtx); std::lock_guard<std::recursive_mutex> lck(recMtx);
if (selectedStreamName.empty() || !audioStream) { return; } if (selectedStreamName.empty() || !audioStream) {
selectedStreamName.clear();
return;
}
if (recording && recMode == RECORDER_MODE_AUDIO) { stop(); } if (recording && recMode == RECORDER_MODE_AUDIO) { stop(); }
stopAudioPath(); stopAudioPath();
sigpath::sinkManager.unbindStream(selectedStreamName, audioStream); sigpath::sinkManager.unbindStream(selectedStreamName, audioStream);
selectedStreamName = ""; selectedStreamName.clear();
audioStream = NULL; audioStream = NULL;
} }
@ -315,6 +350,36 @@ private:
meter.stop(); meter.stop();
} }
static void streamRegisteredHandler(std::string name, void* ctx) {
RecorderModule* _this = (RecorderModule*)ctx;
// Add new stream to the list
_this->audioStreams.define(name, name, name);
// If no stream is selected, select new stream. If not, update the menu ID.
if (_this->selectedStreamName.empty()) {
_this->selectStream(name);
}
else {
_this->streamId = _this->audioStreams.keyId(_this->selectedStreamName);
}
}
static void streamUnregisterHandler(std::string name, void* ctx) {
RecorderModule* _this = (RecorderModule*)ctx;
// Remove stream from list
_this->audioStreams.undefineKey(name);
// If the stream is in used, deselect it and reselect default. Otherwise, update ID.
if (_this->selectedStreamName == name) {
_this->selectStream("");
}
else {
_this->streamId = _this->audioStreams.keyId(_this->selectedStreamName);
}
}
void updateAudioMeter(dsp::stereo_t& lvl) { void updateAudioMeter(dsp::stereo_t& lvl) {
// Note: Yes, using the natural log is on purpose, it just gives a more beautiful result. // Note: Yes, using the natural log is on purpose, it just gives a more beautiful result.
double frameTime = 1.0 / ImGui::GetIO().Framerate; double frameTime = 1.0 / ImGui::GetIO().Framerate;
@ -385,6 +450,8 @@ private:
dsp::sink::Handler<dsp::stereo_t> stereoSink; dsp::sink::Handler<dsp::stereo_t> stereoSink;
dsp::sink::Handler<float> monoSink; dsp::sink::Handler<float> monoSink;
OptionList<std::string, std::string> audioStreams;
int streamId = 0;
dsp::stream<dsp::stereo_t>* audioStream = NULL; dsp::stream<dsp::stereo_t>* audioStream = NULL;
dsp::audio::Volume volume; dsp::audio::Volume volume;
dsp::routing::Splitter<dsp::stereo_t> splitter; dsp::routing::Splitter<dsp::stereo_t> splitter;
@ -393,6 +460,9 @@ private:
uint64_t samplerate = 48000; uint64_t samplerate = 48000;
EventHandler<std::string> onStreamRegisteredHandler;
EventHandler<std::string> onStreamUnregisterHandler;
}; };
MOD_EXPORT void _INIT_() { MOD_EXPORT void _INIT_() {