diff --git a/oscar/Resources.qrc b/oscar/Resources.qrc index e2807622..07a1980a 100644 --- a/oscar/Resources.qrc +++ b/oscar/Resources.qrc @@ -68,5 +68,6 @@ icons/question_mark.png icons/checkmark.png icons/empty_box.png + icons/resvent.png diff --git a/oscar/SleepLib/loader_plugins/resvent_loader.cpp b/oscar/SleepLib/loader_plugins/resvent_loader.cpp new file mode 100644 index 00000000..364e814a --- /dev/null +++ b/oscar/SleepLib/loader_plugins/resvent_loader.cpp @@ -0,0 +1,636 @@ +/* SleepLib Resvent Loader Implementation + * + * Copyright (c) 2019-2023 The OSCAR Team + * Copyright (c) 2011-2018 Mark Watkins + * + * This file is subject to the terms and conditions of the GNU General Public + * License. See the file COPYING in the main directory of the source code + * for more details. */ + +//******************************************************************************************** +// Please only INCREMENT the resvent_data_version in resvent_loader.h when making changes +// that change loader behaviour or modify channels in a manner that fixes old data imports. +// Note that changing the data version will require a reimport of existing data for which OSCAR +// does not keep a backup - so it should be avoided if possible. +// i.e. there is no need to change the version when adding support for new devices +//******************************************************************************************** + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "resvent_loader.h" + +#ifdef DEBUG_EFFICIENCY +#include // only available in 4.8 +#endif + +// Files WXX_XX contain flow rate and pressure and PXX_XX contain Pressure, IPAP, EPAP, Leak, Vt, MV, RR, Ti, IE, Spo2, PR +// Both files contain a little header of size 0x24 bytes. In offset 0x12 contain the total number of different records in +// the different files of the same type. And later contain the previous describe quantity of description header of size 0x20 +// containing the details for every type of record (e.g. sample chunk size). + +ResventLoader::ResventLoader() +{ + const QString RESVENT_ICON = ":/icons/resvent.png"; + + QString s = newInfo().series; + m_pixmap_paths[s] = RESVENT_ICON; + m_pixmaps[s] = QPixmap(RESVENT_ICON); + + m_type = MT_CPAP; +} +ResventLoader::~ResventLoader() +{ +} + +const QString kResventTherapyFolder = "THERAPY"; +const QString kResventConfigFolder = "CONFIG"; +const QString kResventRecordFolder = "RECORD"; +const QString kResventSysConfigFilename = "SYSCFG"; +constexpr qint64 kDateTimeOffset = 7 * 60 * 60 * 1000; +constexpr int kMainHeaderSize = 0x24; +constexpr int kDescriptionHeaderSize = 0x20; +constexpr int kChunkDurationInSecOffset = 0x10; +constexpr int kDescriptionCountOffset = 0x12; +constexpr int kDescriptionSamplesByChunk = 0x1e; + +bool ResventLoader::Detect(const QString & givenpath) +{ + QDir dir(givenpath); + + if (!dir.exists()) { + return false; + } + + if (!dir.exists(kResventTherapyFolder)) { + return false; + } + + dir.cd(kResventTherapyFolder); + if (!dir.exists(kResventConfigFolder)) { + return false; + } + + if (!dir.exists(kResventRecordFolder)) { + return false; + } + + return true; +} + + +MachineInfo ResventLoader::PeekInfo(const QString & path) +{ + if (!Detect(path)) { + return MachineInfo(); + } + + MachineInfo info = newInfo(); + + const auto sys_config_path = path + QDir::separator() + kResventTherapyFolder + QDir::separator() + kResventConfigFolder + QDir::separator() + kResventSysConfigFilename; + if (!QFile::exists(sys_config_path)) { + qDebug() << "Resvent Data card has no" << kResventSysConfigFilename << "file in " << sys_config_path; + return MachineInfo(); + } + QFile f(sys_config_path); + f.open(QIODevice::ReadOnly | QIODevice::Text); + f.seek(4); + while (!f.atEnd()) { + QString line = f.readLine().trimmed(); + + const auto elems = line.split("="); + assert(elems.size() == 2); + + if (elems[0] == "models") { + info.model = elems[1]; + } + else if (elems[0] == "sn") { + info.serial = elems[1]; + } + else if (elems[0] == "num") { + info.version = elems[1].toInt(); + } + else if (elems[0] == "num") { + info.type = MachineType::MT_CPAP; + } + } + + if (info.model.contains("Point", Qt::CaseInsensitive)) { + info.brand = "Hoffrichter"; + } else { + info.brand = "Resvent"; + } + + return info; +} + +std::vector GetSessionsDate(const QString& dirpath) { + std::vector sessions_date; + + const auto records_path = dirpath + QDir::separator() + kResventTherapyFolder + QDir::separator() + kResventRecordFolder; + QDir records_folder(records_path); + const auto year_month_folders = records_folder.entryList(QStringList(), QDir::Dirs|QDir::NoDotAndDotDot, QDir::Name); + std::for_each(year_month_folders.cbegin(), year_month_folders.cend(), [&](const QString& year_month_folder_name){ + if (year_month_folder_name.length() != 6) { + return; + } + + const int year = std::stoi(year_month_folder_name.left(4).toStdString()); + const int month = std::stoi(year_month_folder_name.right(2).toStdString()); + + const auto year_month_folder_path = records_path + QDir::separator() + year_month_folder_name; + QDir year_month_folder(year_month_folder_path); + const auto session_folders = year_month_folder.entryList(QStringList(), QDir::Dirs|QDir::NoDotAndDotDot, QDir::Name); + std::for_each(session_folders.cbegin(), session_folders.cend(), [&](const QString& day_folder){ + const auto day = std::stoi(day_folder.toStdString()); + + sessions_date.push_back(QDate(year, month, day)); + }); + }); + return sessions_date; +} + +enum class EventType { + UsageSec= 1, + UnixStart = 2, + ObstructiveApnea = 17, + CentralApnea = 18, + Hypopnea = 19, + FlowLimitation = 20, + RERA = 21, + PeriodicBreathing = 22, + Snore = 23 +}; + +struct EventData { + EventType type; + QDateTime date_time; + int duration; +}; + +struct UsageData { + QString number{}; + QDateTime start_time{}; + QDateTime end_time{}; + qint32 countAHI = 0; + qint32 countOAI = 0; + qint32 countCAI = 0; + qint32 countAI = 0; + qint32 countHI = 0; + qint32 countRERA = 0; + qint32 countSNI = 0; + qint32 countBreath = 0; +}; + +void UpdateEvents(EventType event_type, const std::unordered_map>& events, Session* session) { + static std::unordered_map mapping {{EventType::ObstructiveApnea, CPAP_Obstructive}, + {EventType::CentralApnea, CPAP_Apnea}, + {EventType::Hypopnea, CPAP_Hypopnea}, + {EventType::FlowLimitation, CPAP_FlowLimit}, + {EventType::RERA, CPAP_RERA}, + {EventType::PeriodicBreathing, CPAP_PB}, + {EventType::Snore, CPAP_Snore}}; + const auto it_events = events.find(event_type); + const auto it_mapping = mapping.find(event_type); + if (it_events == events.cend() || it_mapping == mapping.cend()) { + return; + } + + EventList* event_list = session->AddEventList(it_mapping->second, EVL_Event); + std::for_each(it_events->second.cbegin(), it_events->second.cend(), [&](const EventData& event_data){ + event_list->AddEvent(event_data.date_time.toMSecsSinceEpoch() + kDateTimeOffset, event_data.duration); + }); +} + +QString GetSessionFolder(const QString& dirpath, const QDate& session_date) { + const auto year_month_folder = QString::number(session_date.year()) + (session_date.month() > 10 ? "" : "0") + QString::number(session_date.month()); + const auto day_folder = (session_date.day() > 10 ? "" : "0") + QString::number(session_date.day()); + const auto session_folder_path = dirpath + QDir::separator() + kResventTherapyFolder + QDir::separator() + kResventRecordFolder + QDir::separator() + year_month_folder + QDir::separator() + day_folder; + return session_folder_path; +} + +void LoadEvents(const QString& session_folder_path, Session* session, const UsageData& usage) { + const auto event_file_path = session_folder_path + QDir::separator() + "EV" + usage.number; + + std::unordered_map> events; + QFile f(event_file_path); + f.open(QIODevice::ReadOnly | QIODevice::Text); + f.seek(4); + while (!f.atEnd()) { + QString line = f.readLine().trimmed(); + + const auto elems = line.split(",", Qt::SkipEmptyParts); + if (elems.size() != 4) { + continue; + } + + const auto event_type_elems = elems.at(0).split("="); + const auto date_time_elems = elems.at(1).split("="); + const auto duration_elems = elems.at(2).split("="); + + assert(event_type_elems.size() == 2); + assert(date_time_elems.size() == 2); + const auto event_type = static_cast(std::stoi(event_type_elems[1].toStdString())); + const auto date_time = QDateTime::fromTime_t(std::stoi(date_time_elems[1].toStdString())); + const auto duration = std::stoi(duration_elems[1].toStdString()); + + events[event_type].push_back(EventData{event_type, date_time, duration}); + } + + static std::vector mapping {EventType::ObstructiveApnea, + EventType::CentralApnea, + EventType::Hypopnea, + EventType::FlowLimitation, + EventType::RERA, + EventType::PeriodicBreathing, + EventType::Snore}; + + std::for_each(mapping.cbegin(), mapping.cend(), [&](EventType event_type){ + UpdateEvents(event_type, events, session); + }); +} + +template +T read_from_file(QFile& f) { + T data{}; + f.read(reinterpret_cast(&data), sizeof(T)); + return data; +} + +struct WaveFileData { + unsigned int wave_event_id; + QString file_base_name; + unsigned int sample_rate_offset; + unsigned int start_offset; +}; + +EventList* GetEventList(const QString& name, Session* session, float sample_rate = 0.0) { + if (name == "Press") { + return session->AddEventList(CPAP_Pressure, EVL_Event); + } + else if (name == "IPAP") { + return session->AddEventList(CPAP_IPAP, EVL_Event); + } + else if (name == "EPAP") { + return session->AddEventList(CPAP_EPAP, EVL_Event); + } + else if (name == "Leak") { + return session->AddEventList(CPAP_Leak, EVL_Event); + } + else if (name == "Vt") { + return session->AddEventList(CPAP_TidalVolume, EVL_Event); + } + else if (name == "MV") { + return session->AddEventList(CPAP_MinuteVent, EVL_Event); + } + else if (name == "RR") { + return session->AddEventList(CPAP_RespRate, EVL_Event); + } + else if (name == "Ti") { + return session->AddEventList(CPAP_Ti, EVL_Event); + } + else if (name == "I:E") { + return session->AddEventList(CPAP_IE, EVL_Event); + } + else if (name == "SpO2" || name == "PR") { + // Not present + return nullptr; + } + else if (name == "Pressure") { + return session->AddEventList(CPAP_MaskPressure, EVL_Waveform, 0.01, 0.0, 0.0, 0.0, 1000.0 / sample_rate); + } + else if (name == "Flow") { + return session->AddEventList(CPAP_FlowRate, EVL_Waveform, 0.01, 0.0, 0.0, 0.0, 1000.0 / sample_rate); + } + else { + // Not supported + assert(false); + return nullptr; + } +} + +struct ChunkData { + EventList* event_list; + uint16_t samples_by_chunk; + qint64 start_time; + int total_samples_by_chunk; + float sample_rate; +}; + +QString ReadDescriptionName(QFile& f) { + constexpr int kNameSize = 9; + std::array name; + const auto readed = f.read(name.data(), kNameSize - 1); + assert(readed == kNameSize - 1); + + return QString(name.data()); +} + +void ReadWaveFormsHeaders(QFile& f, std::vector& wave_forms, Session* session, const UsageData& usage) { + f.seek(kChunkDurationInSecOffset); + const auto chunk_duration_in_sec = read_from_file(f); + f.seek(kDescriptionCountOffset); + const auto description_count = read_from_file(f); + wave_forms.resize(description_count); + + for (unsigned int i = 0; i < description_count; i++) { + const auto description_header_offset = kMainHeaderSize + i * kDescriptionHeaderSize; + f.seek(description_header_offset); + const auto name = ReadDescriptionName(f); + f.seek(description_header_offset + kDescriptionSamplesByChunk); + const auto samples_by_chunk = read_from_file(f); + + wave_forms[i].sample_rate = 1.0 * samples_by_chunk / chunk_duration_in_sec; + wave_forms[i].event_list = GetEventList(name, session, wave_forms[i].sample_rate); + wave_forms[i].samples_by_chunk = samples_by_chunk; + wave_forms[i].start_time = usage.start_time.toMSecsSinceEpoch(); + } +} + +void LoadOtherWaveForms(const QString& session_folder_path, Session* session, const UsageData& usage) { + QDir session_folder(session_folder_path); + + const auto wave_files = session_folder.entryList(QStringList() << "P" + usage.number + "_*", QDir::Files, QDir::Name); + + std::vector wave_forms; + bool initialized = false; + std::for_each(wave_files.cbegin(), wave_files.cend(), [&](const QString& wave_file){ + // W01_ file + QFile f(session_folder_path + QDir::separator() + wave_file); + f.open(QIODevice::ReadOnly); + + if (!initialized) { + ReadWaveFormsHeaders(f, wave_forms, session, usage); + initialized = true; + } + f.seek(kMainHeaderSize + wave_forms.size() * kDescriptionHeaderSize); + + std::vector chunk(std::max_element(wave_forms.cbegin(), wave_forms.cend(), [](const ChunkData& lhs, const ChunkData& rhs){ + return lhs.samples_by_chunk < rhs.samples_by_chunk; + })->samples_by_chunk); + while (!f.atEnd()) { + for (unsigned int i = 0; i < wave_forms.size(); i++) { + const auto& wave_form = wave_forms[i].event_list; + const auto samples_by_chunk_actual = wave_forms[i].samples_by_chunk; + auto& start_time_current = wave_forms[i].start_time; + auto& total_samples_by_chunk = wave_forms[i].total_samples_by_chunk; + const auto sample_rate = wave_forms[i].sample_rate; + + const auto readed = f.read(reinterpret_cast(chunk.data()), chunk.size() * sizeof(qint16)); + if (wave_form) { + const auto readed_elements = readed / sizeof(qint16); + if (readed_elements != samples_by_chunk_actual) { + std::fill(std::begin(chunk) + readed_elements, std::end(chunk), 0); + } + + int offset = 0; + std::for_each(chunk.cbegin(), chunk.cend(), [&](const qint16& value){ + wave_form->AddEvent(start_time_current + offset + kDateTimeOffset, value); + offset += 1000.0 / sample_rate; + }); + } + + start_time_current += samples_by_chunk_actual * 1000.0 / sample_rate; + total_samples_by_chunk += samples_by_chunk_actual; + } + } + }); + + std::vector chunk; + for (unsigned int i = 0; i < wave_forms.size(); i++) { + const auto& wave_form = wave_forms[i]; + const auto expected_samples = usage.start_time.msecsTo(usage.end_time) / 1000.0 * wave_form.sample_rate; + if (wave_form.total_samples_by_chunk < expected_samples) { + chunk.resize(expected_samples - wave_form.total_samples_by_chunk); + if (wave_form.event_list) { + int offset = 0; + std::for_each(chunk.cbegin(), chunk.cend(), [&](const qint16& value){ + wave_form.event_list->AddEvent(wave_form.start_time + offset + kDateTimeOffset, value); + offset += 1000.0 / wave_form.sample_rate; + }); + } + } + } +} + +void LoadWaveForms(const QString& session_folder_path, Session* session, const UsageData& usage) { + QDir session_folder(session_folder_path); + + const auto wave_files = session_folder.entryList(QStringList() << "W" + usage.number + "_*", QDir::Files, QDir::Name); + + std::vector wave_forms; + bool initialized = false; + + std::for_each(wave_files.cbegin(), wave_files.cend(), [&](const QString& wave_file){ + // W01_ file + QFile f(session_folder_path + QDir::separator() + wave_file); + f.open(QIODevice::ReadOnly); + + if (!initialized) { + ReadWaveFormsHeaders(f, wave_forms, session, usage); + initialized = true; + } + f.seek(kMainHeaderSize + wave_forms.size() * kDescriptionHeaderSize); + + std::vector chunk(std::max_element(wave_forms.cbegin(), wave_forms.cend(), [](const ChunkData& lhs, const ChunkData& rhs){ + return lhs.samples_by_chunk < rhs.samples_by_chunk; + })->samples_by_chunk); + while (!f.atEnd()) { + for (unsigned int i = 0; i < wave_forms.size(); i++) { + const auto& wave_form = wave_forms[i].event_list; + const auto samples_by_chunk_actual = wave_forms[i].samples_by_chunk; + auto& start_time_current = wave_forms[i].start_time; + auto& total_samples_by_chunk = wave_forms[i].total_samples_by_chunk; + const auto sample_rate = wave_forms[i].sample_rate; + + const auto duration = samples_by_chunk_actual * 1000.0 / sample_rate; + const auto readed = f.read(reinterpret_cast(chunk.data()), chunk.size() * sizeof(qint16)); + if (wave_form) { + const auto readed_elements = readed / sizeof(qint16); + if (readed_elements != samples_by_chunk_actual) { + std::fill(std::begin(chunk) + readed_elements, std::end(chunk), 0); + } + wave_form->AddWaveform(start_time_current + kDateTimeOffset, chunk.data(), samples_by_chunk_actual, duration); + } + + start_time_current += duration; + total_samples_by_chunk += samples_by_chunk_actual; + } + } + }); + + std::vector chunk; + for (unsigned int i = 0; i < wave_forms.size(); i++) { + const auto& wave_form = wave_forms[i]; + const auto expected_samples = usage.start_time.msecsTo(usage.end_time) / 1000.0 * wave_form.sample_rate; + if (wave_form.total_samples_by_chunk < expected_samples) { + chunk.resize(expected_samples - wave_form.total_samples_by_chunk); + if (wave_form.event_list) { + const auto duration = chunk.size() * 1000.0 / wave_form.sample_rate; + wave_form.event_list->AddWaveform(wave_form.start_time + kDateTimeOffset, chunk.data(), chunk.size(), duration); + } + } + } +} + +void LoadStats(const UsageData& /*usage_data*/, Session* session) { + // session->settings[CPAP_AHI] = usage_data.countAHI; + // session->setCount(CPAP_AI, usage_data.countAI); + // session->setCount(CPAP_CAI, usage_data.countCAI); + // session->setCount(CPAP_HI, usage_data.countHI); + // session->setCount(CPAP_Obstructive, usage_data.countOAI); + // session->settings[CPAP_RERA] = usage_data.countRERA; + // session->settings[CPAP_Snore] = usage_data.countSNI; + session->settings[CPAP_Mode] = MODE_APAP; +} + +UsageData ReadUsage(const QString& session_folder_path, const QString& usage_number) { + UsageData usage_data; + usage_data.number = usage_number; + + const auto session_stat_path = session_folder_path + QDir::separator() + "STAT" + usage_number; + if (!QFile::exists(session_stat_path)) { + qDebug() << "Resvent Data card has no " << session_stat_path; + return usage_data; + } + QFile f(session_stat_path); + f.open(QIODevice::ReadOnly | QIODevice::Text); + f.seek(4); + while (!f.atEnd()) { + QString line = f.readLine().trimmed(); + + const auto elems = line.split("="); + assert(elems.size() == 2); + + if (elems[0] == "secStart") { + usage_data.start_time = QDateTime::fromTime_t(std::stoi(elems[1].toStdString())); + } + else if (elems[0] == "secUsed") { + usage_data.end_time = QDateTime::fromTime_t(usage_data.start_time.toTime_t() + std::stoi(elems[1].toStdString())); + } + else if (elems[0] == "cntAHI") { + usage_data.countAHI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntOAI") { + usage_data.countOAI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntCAI") { + usage_data.countCAI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntAI") { + usage_data.countAI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntHI") { + usage_data.countHI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntRERA") { + usage_data.countRERA = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntSNI") { + usage_data.countSNI = std::stoi(elems[1].toStdString()); + } + else if (elems[0] == "cntBreath") { + usage_data.countBreath = std::stoi(elems[1].toStdString()); + } + } + + return usage_data; +} + +std::vector GetDifferentUsage(const QString& session_folder_path) { + QDir session_folder(session_folder_path); + + const auto stat_files = session_folder.entryList(QStringList() << "STAT*", QDir::Files, QDir::Name); + std::vector usage_data; + std::for_each(stat_files.cbegin(), stat_files.cend(), [&](const QString& stat_file){ + if (stat_file.size() != 6) { + return; + } + + auto usageData = ReadUsage(session_folder_path, stat_file.right(2)); + + usage_data.push_back(usageData); + }); + return usage_data; +} + +int LoadSession(const QString& dirpath, const QDate& session_date, Machine* machine) { + const auto session_folder_path = GetSessionFolder(dirpath, session_date); + + const auto different_usage = GetDifferentUsage(session_folder_path); + + return std::accumulate(different_usage.cbegin(), different_usage.cend(), 0, [&](int base, const UsageData& usage){ + if (machine->SessionExists(usage.start_time.toMSecsSinceEpoch() + kDateTimeOffset)) { + return base; + } + Session* session = new Session(machine, usage.start_time.toMSecsSinceEpoch() + kDateTimeOffset); + session->SetChanged(true); + session->really_set_first(usage.start_time.toMSecsSinceEpoch() + kDateTimeOffset); + session->really_set_last(usage.end_time.toMSecsSinceEpoch() + kDateTimeOffset); + + LoadStats(usage, session); + LoadWaveForms(session_folder_path, session, usage); + LoadOtherWaveForms(session_folder_path, session, usage); + LoadEvents(session_folder_path, session, usage); + + session->UpdateSummaries(); + session->Store(machine->getDataPath()); + machine->AddSession(session); + return base + 1; + }); +} + + +/////////////////////////////////////////////////////////////////////////////////////////// +// Sorted EDF files that need processing into date records according to ResMed noon split +/////////////////////////////////////////////////////////////////////////////////////////// +int ResventLoader::Open(const QString & dirpath) +{ + const auto machine_info = PeekInfo(dirpath); + + // Abort if no serial number + if (machine_info.serial.isEmpty()) { + qDebug() << "Resvent Data card has no valid serial number in " << kResventSysConfigFilename; + return -1; + } + + const auto sessions_date = GetSessionsDate(dirpath); + + Machine *machine = p_profile->CreateMachine(machine_info); + + int new_sessions = 0; + std::for_each(sessions_date.cbegin(), sessions_date.cend(), [&](const QDate& session_date){ + new_sessions += LoadSession(dirpath, session_date, machine); + }); + + machine->Save(); + + return new_sessions; +} + +void ResventLoader::initChannels() +{ +} + +ChannelID ResventLoader::PresReliefMode() { return 0; } +ChannelID ResventLoader::PresReliefLevel() { return 0; } +ChannelID ResventLoader::CPAPModeChannel() { return 0; } + +bool resvent_initialized = false; +void ResventLoader::Register() +{ + if (resvent_initialized) { return; } + + qDebug() << "Registering ResventLoader"; + RegisterLoader(new ResventLoader()); + + resvent_initialized = true; +} diff --git a/oscar/SleepLib/loader_plugins/resvent_loader.h b/oscar/SleepLib/loader_plugins/resvent_loader.h new file mode 100644 index 00000000..0ccfc73f --- /dev/null +++ b/oscar/SleepLib/loader_plugins/resvent_loader.h @@ -0,0 +1,76 @@ +/* SleepLib Resvent Loader Implementation + * + * Copyright (c) 2019-2023 The OSCAR Team + * Copyright (C) 2011-2018 Mark Watkins + * + * This file is subject to the terms and conditions of the GNU General Public + * License. See the file COPYING in the main directory of the source code + * for more details. */ + +#ifndef RESVENT_LOADER_H +#define RESVENT_LOADER_H + +#include +#include "SleepLib/machine.h" // Base class: MachineLoader +#include "SleepLib/machine_loader.h" +#include "SleepLib/profiles.h" + +//******************************************************************************************** +/// IMPORTANT!!! +//******************************************************************************************** +// Please INCREMENT the following value when making changes to this loaders implementation. +// +const int resvent_data_version = 1; +// +//******************************************************************************************** + +const QString resvent_class_name = "Resvent/Hoffrichter"; + +/*! \class ResventLoader + \brief Importer for Resvent iBreezer and Hoffrichter Point 3 + */ +class ResventLoader : public CPAPLoader +{ + Q_OBJECT +public: + ResventLoader(); + virtual ~ResventLoader(); + + //! \brief Detect if the given path contains a valid Folder structure + virtual bool Detect(const QString & path); + + //! \brief Look up machine model information of ResMed file structure stored at path + virtual MachineInfo PeekInfo(const QString & path); + + //! \brief Scans for ResMed SD folder structure signature, and loads any new data if found + virtual int Open(const QString &); + + //! \brief Returns the version number of this Resvent loader + virtual int Version() { return resvent_data_version; } + + //! \brief Returns the Machine class name of this loader. ("Resvent") + virtual const QString &loaderName() { return resvent_class_name; } + + //! \brief Register the ResmedLoader with the list of other machine loaders + static void Register(); + + virtual MachineInfo newInfo() { + return MachineInfo(MT_CPAP, 0, resvent_class_name, QObject::tr("Resvent/Hoffrichter"), QString(), QString(), QString(), QObject::tr("iBreeze/Point3"), QDateTime::currentDateTime(), resvent_data_version); + } + + virtual void initChannels(); + + + //////////////////////////////////////////////////////////////////////////////////////////////////////////// + // Now for some CPAPLoader overrides + //////////////////////////////////////////////////////////////////////////////////////////////////////////// + virtual QString PresReliefLabel() { return QObject::tr("EPR: "); } + + virtual ChannelID PresReliefMode(); + virtual ChannelID PresReliefLevel(); + virtual ChannelID CPAPModeChannel(); + + //////////////////////////////////////////////////////////////////////////////////////////////////////////// +}; + +#endif // RESVENT_LOADER_H diff --git a/oscar/icons/resvent.png b/oscar/icons/resvent.png new file mode 100644 index 00000000..7e859d56 Binary files /dev/null and b/oscar/icons/resvent.png differ diff --git a/oscar/main.cpp b/oscar/main.cpp index ba2bc454..805b0858 100644 --- a/oscar/main.cpp +++ b/oscar/main.cpp @@ -47,6 +47,7 @@ #include "SleepLib/loader_plugins/weinmann_loader.h" #include "SleepLib/loader_plugins/viatom_loader.h" #include "SleepLib/loader_plugins/prisma_loader.h" +#include "SleepLib/loader_plugins/resvent_loader.h" MainWindow *mainwin = nullptr; @@ -693,6 +694,7 @@ int main(int argc, char *argv[]) { MD300W1Loader::Register(); ViatomLoader::Register(); PrismaLoader::Register(); + ResventLoader::Register(); // Begin logging device connection activity. QString connectionsLogDir = GetLogDir() + "/connections"; diff --git a/oscar/oscar.pro b/oscar/oscar.pro index 7ddf0687..a8aa8eae 100644 --- a/oscar/oscar.pro +++ b/oscar/oscar.pro @@ -320,6 +320,7 @@ SOURCES += \ SleepLib/loader_plugins/somnopose_loader.cpp \ SleepLib/loader_plugins/viatom_loader.cpp \ SleepLib/loader_plugins/zeo_loader.cpp \ + SleepLib/loader_plugins/resvent_loader.cpp \ zip.cpp \ SleepLib/thirdparty/miniz.c \ csv.cpp \ @@ -426,6 +427,7 @@ HEADERS += \ SleepLib/loader_plugins/somnopose_loader.h \ SleepLib/loader_plugins/viatom_loader.h \ SleepLib/loader_plugins/zeo_loader.h \ + SleepLib/loader_plugins/resvent_loader.h \ SleepLib/thirdparty/botan_all.h \ SleepLib/thirdparty/botan_windows.h \ SleepLib/thirdparty/botan_linux.h \