This commit is contained in:
AlexandreRouma 2023-01-02 01:24:30 +01:00
parent 8771e4bf09
commit 385b34a0a1

View File

@ -40,6 +40,7 @@ public:
RecorderModule(std::string name) : folderSelect("%ROOT%/recordings") { RecorderModule(std::string name) : folderSelect("%ROOT%/recordings") {
this->name = name; this->name = name;
root = (std::string)core::args["root"]; root = (std::string)core::args["root"];
strcpy(nameTemplate, "$t_$f_$h-$m-$s_$d-$M-$y");
// Define option lists // Define option lists
containers.define("WAV", wav::FORMAT_WAV); containers.define("WAV", wav::FORMAT_WAV);
@ -61,8 +62,8 @@ public:
if (config.conf[name].contains("recPath")) { if (config.conf[name].contains("recPath")) {
folderSelect.setPath(config.conf[name]["recPath"]); folderSelect.setPath(config.conf[name]["recPath"]);
} }
if (config.conf[name].contains("format") && containers.keyExists(config.conf[name]["format"])) { if (config.conf[name].contains("container") && containers.keyExists(config.conf[name]["container"])) {
containerId = containers.keyId(config.conf[name]["format"]); containerId = containers.keyId(config.conf[name]["container"]);
} }
if (config.conf[name].contains("sampleType") && sampleTypes.keyExists(config.conf[name]["sampleType"])) { if (config.conf[name].contains("sampleType") && sampleTypes.keyExists(config.conf[name]["sampleType"])) {
sampleTypeId = sampleTypes.keyId(config.conf[name]["sampleType"]); sampleTypeId = sampleTypes.keyId(config.conf[name]["sampleType"]);
@ -76,6 +77,13 @@ public:
if (config.conf[name].contains("ignoreSilence")) { if (config.conf[name].contains("ignoreSilence")) {
ignoreSilence = config.conf[name]["ignoreSilence"]; ignoreSilence = config.conf[name]["ignoreSilence"];
} }
if (config.conf[name].contains("nameTemplate")) {
std::string _nameTemplate = config.conf[name]["nameTemplate"];
if (_nameTemplate.length() > sizeof(nameTemplate)-1) {
_nameTemplate = _nameTemplate.substr(0, sizeof(nameTemplate)-1);
}
strcpy(nameTemplate, _nameTemplate.c_str());
}
config.release(); config.release();
// Init audio path // Init audio path
@ -91,13 +99,18 @@ public:
monoSink.init(&s2m.out, monoHandler, this); monoSink.init(&s2m.out, monoHandler, this);
gui::menu.registerEntry(name, menuHandler, this); gui::menu.registerEntry(name, menuHandler, this);
core::modComManager.registerInterface("recorder", name, moduleInterfaceHandler, this);
} }
~RecorderModule() { ~RecorderModule() {
std::lock_guard<std::recursive_mutex> lck(recMtx);
core::modComManager.unregisterInterface(name);
gui::menu.removeEntry(name);
stop(); stop();
deselectStream(); deselectStream();
sigpath::sinkManager.onStreamRegistered.unbindHandler(&onStreamRegisteredHandler);
sigpath::sinkManager.onStreamUnregister.unbindHandler(&onStreamUnregisterHandler);
meter.stop(); meter.stop();
gui::menu.removeEntry(name);
} }
void postInit() { void postInit() {
@ -150,8 +163,10 @@ public:
writer.setSamplerate(samplerate); writer.setSamplerate(samplerate);
// Open file // Open file
std::string prefix = (recMode == RECORDER_MODE_AUDIO) ? "/audio_" : "/baseband_"; std::string type = (recMode == RECORDER_MODE_AUDIO) ? "audio" : "baseband";
std::string expandedPath = expandString(folderSelect.path + genFileName(prefix, false)); std::string vfoName = (recMode == RECORDER_MODE_AUDIO) ? gui::waterfall.selectedVFO : "";
std::string extension = ".wav";
std::string expandedPath = expandString(folderSelect.path + "/" + genFileName(nameTemplate, type, vfoName) + extension);
if (!writer.open(expandedPath)) { if (!writer.open(expandedPath)) {
spdlog::error("Failed to open file for recording: {0}", expandedPath); spdlog::error("Failed to open file for recording: {0}", expandedPath);
return; return;
@ -242,6 +257,14 @@ private:
} }
} }
ImGui::LeftLabel("Name template");
ImGui::FillWidth();
if (ImGui::InputText(CONCAT("##_recorder_name_template_", _this->name), _this->nameTemplate, 1023)) {
config.acquire();
config.conf[_this->name]["nameTemplate"] = _this->nameTemplate;
config.release(true);
}
ImGui::LeftLabel("Container"); ImGui::LeftLabel("Container");
ImGui::FillWidth(); ImGui::FillWidth();
if (ImGui::Combo(CONCAT("##_recorder_container_", _this->name), &_this->containerId, _this->containers.txt)) { if (ImGui::Combo(CONCAT("##_recorder_container_", _this->name), &_this->containerId, _this->containers.txt)) {
@ -399,7 +422,6 @@ private:
double frameTime = 1.0 / ImGui::GetIO().Framerate; double frameTime = 1.0 / ImGui::GetIO().Framerate;
lvl.l = std::clamp<float>(lvl.l - (frameTime * 50.0), -90.0f, 10.0f); lvl.l = std::clamp<float>(lvl.l - (frameTime * 50.0), -90.0f, 10.0f);
lvl.r = std::clamp<float>(lvl.r - (frameTime * 50.0), -90.0f, 10.0f); lvl.r = std::clamp<float>(lvl.r - (frameTime * 50.0), -90.0f, 10.0f);
// TODO: FINISH METER
dsp::stereo_t rawLvl = meter.getLevel(); dsp::stereo_t rawLvl = meter.getLevel();
meter.resetLevel(); meter.resetLevel();
dsp::stereo_t dbLvl = { 10.0f * logf(rawLvl.l), 10.0f * logf(rawLvl.r) }; dsp::stereo_t dbLvl = { 10.0f * logf(rawLvl.l), 10.0f * logf(rawLvl.r) };
@ -407,19 +429,44 @@ private:
if (dbLvl.r > lvl.r) { lvl.r = dbLvl.r; } if (dbLvl.r > lvl.r) { lvl.r = dbLvl.r; }
} }
// TODO: REPLACE WITH SOMETHING CLEAN std::string genFileName(std::string templ, std::string type, std::string name) {
std::string genFileName(std::string prefix, bool isVfo, std::string name = "") { // Get data
time_t now = time(0); time_t now = time(0);
tm* ltm = localtime(&now); tm* ltm = localtime(&now);
char buf[1024]; char buf[1024];
double freq = gui::waterfall.getCenterFrequency(); double freq = gui::waterfall.getCenterFrequency();
; if (gui::waterfall.vfos.find(name) != gui::waterfall.vfos.end()) {
if (isVfo && gui::waterfall.vfos.find(name) != gui::waterfall.vfos.end()) {
freq += gui::waterfall.vfos[name]->generalOffset; freq += gui::waterfall.vfos[name]->generalOffset;
} }
sprintf(buf, "%.0lfHz_%02d-%02d-%02d_%02d-%02d-%02d.wav", freq, ltm->tm_hour, ltm->tm_min, ltm->tm_sec, ltm->tm_mday, ltm->tm_mon + 1, ltm->tm_year + 1900);
return prefix + buf; // Format to string
char freqStr[128];
char hourStr[128];
char minStr[128];
char secStr[128];
char dayStr[128];
char monStr[128];
char yearStr[128];
sprintf(freqStr, "%.0lfHz", freq);
sprintf(hourStr, "%02d", ltm->tm_hour);
sprintf(minStr, "%02d", ltm->tm_min);
sprintf(secStr, "%02d", ltm->tm_sec);
sprintf(dayStr, "%02d", ltm->tm_mday);
sprintf(monStr, "%02d", ltm->tm_mon + 1);
sprintf(yearStr, "%02d", ltm->tm_year + 1900);
// Replace in template
templ = std::regex_replace(templ, std::regex("\\$t"), type);
templ = std::regex_replace(templ, std::regex("\\$f"), freqStr);
templ = std::regex_replace(templ, std::regex("\\$h"), hourStr);
templ = std::regex_replace(templ, std::regex("\\$m"), minStr);
templ = std::regex_replace(templ, std::regex("\\$s"), secStr);
templ = std::regex_replace(templ, std::regex("\\$d"), dayStr);
templ = std::regex_replace(templ, std::regex("\\$M"), monStr);
templ = std::regex_replace(templ, std::regex("\\$y"), yearStr);
return templ;
} }
std::string expandString(std::string input) { std::string expandString(std::string input) {
input = std::regex_replace(input, std::regex("%ROOT%"), root); input = std::regex_replace(input, std::regex("%ROOT%"), root);
return std::regex_replace(input, std::regex("//"), "/"); return std::regex_replace(input, std::regex("//"), "/");
@ -438,9 +485,30 @@ private:
_this->writer.write(data, count); _this->writer.write(data, count);
} }
static void moduleInterfaceHandler(int code, void* in, void* out, void* ctx) {
RecorderModule* _this = (RecorderModule*)ctx;
std::lock_guard lck(_this->recMtx);
if (code == RECORDER_IFACE_CMD_GET_MODE) {
int* _out = (int*)out;
*_out = _this->recMode;
}
else if (code == RECORDER_IFACE_CMD_SET_MODE) {
if (_this->recording) { return; }
int* _in = (int*)in;
_this->recMode = std::clamp<int>(*_in, 0, 1);
}
else if (code == RECORDER_IFACE_CMD_START) {
if (!_this->recording) { _this->start(); }
}
else if (code == RECORDER_IFACE_CMD_STOP) {
if (_this->recording) { _this->stop(); }
}
}
std::string name; std::string name;
bool enabled = true; bool enabled = true;
std::string root; std::string root;
char nameTemplate[1024];
OptionList<std::string, wav::Format> containers; OptionList<std::string, wav::Format> containers;
OptionList<int, wav::SampleType> sampleTypes; OptionList<int, wav::SampleType> sampleTypes;