mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Merge branch 'master' into overview
This commit is contained in:
commit
7786161a4d
@ -12,24 +12,40 @@
|
|||||||
#define MyAppVersion MyAppVersion+"-"+MyBuildNumber
|
#define MyAppVersion MyAppVersion+"-"+MyBuildNumber
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MyAppName "OSCAR"
|
|
||||||
#define MyAppPublisher "The OSCAR Team"
|
#define MyAppPublisher "The OSCAR Team"
|
||||||
#define MyAppExeName "OSCAR.exe"
|
#define MyAppExeName "OSCAR.exe"
|
||||||
|
#define MyAppName "OSCAR"
|
||||||
|
|
||||||
[Setup]
|
[Setup]
|
||||||
SetupLogging=yes
|
SetupLogging=yes
|
||||||
; NOTE: The value of AppId uniquely identifies this application.
|
; NOTE: The value of AppId uniquely identifies this application.
|
||||||
; Do not use the same AppId value in installers for other applications.
|
; Do not use the same AppId value in installers for other applications.
|
||||||
; (To generate a new GUID, click Tools | Generate GUID inside the IDE.)
|
; (To generate a new GUID, click Tools | Generate GUID inside the IDE.)
|
||||||
; Now using separate AppID for Win32 and Win64 -- GTS 4/6/2019
|
; Now using separate AppID for Win32 and Win64 and for test builds -- GTS 4/6/2019
|
||||||
#if MyPlatform == "Win64"
|
#if MyPlatform == "Win64"
|
||||||
ArchitecturesAllowed=x64
|
ArchitecturesAllowed=x64
|
||||||
ArchitecturesInstallIn64BitMode=x64
|
ArchitecturesInstallIn64BitMode=x64
|
||||||
AppId={{FC6F08E6-69BF-4469-ADE3-78199288D305}
|
#if MyReleaseStatus == "r" || MyReleaseStatus == "rc"
|
||||||
|
AppId={{FC6F08E6-69BF-4469-ADE3-78199288D305}
|
||||||
|
#define MyGroupName "OSCAR"
|
||||||
|
#define MyDirName "OSCAR"
|
||||||
|
#else
|
||||||
|
AppId={{C5E23210-4BC5-499D-A0E4-81192748D322}
|
||||||
|
#define MyGroupName "OSCAR (test)"
|
||||||
|
#define MyDirName "OSCAR-test"
|
||||||
|
#endif
|
||||||
; DefaultDirName={%PROGRAMFILES|{pf}}\OSCAR
|
; DefaultDirName={%PROGRAMFILES|{pf}}\OSCAR
|
||||||
; above doesn't work. Always returns x86 directory
|
; above doesn't work. Always returns x86 directory
|
||||||
#else // 32-bit
|
#else // 32-bit
|
||||||
AppId={{4F3EB81B-1866-4124-B388-5FB5DA34FFDD}
|
#if MyReleaseStatus == "r" || MyReleaseStatus == "rc"
|
||||||
|
AppId={{4F3EB81B-1866-4124-B388-5FB5DA34FFDD}
|
||||||
|
#define MyGroupName "OSCAR 32-bit"
|
||||||
|
#define MyDirName "OSCAR"
|
||||||
|
#else
|
||||||
|
AppId={{B0382AB3-ECB4-4F9D-ABB1-F6EF73D4E3DB}
|
||||||
|
#define MyGroupName "OSCAR 32-bit (test)"
|
||||||
|
#define MyDirName "OSCAR-test"
|
||||||
|
#endif
|
||||||
; DefaultDirName={%PROGRAMFILES(X86)|{pf}}\OSCAR
|
; DefaultDirName={%PROGRAMFILES(X86)|{pf}}\OSCAR
|
||||||
#endif
|
#endif
|
||||||
AppName={#MyAppName}
|
AppName={#MyAppName}
|
||||||
@ -38,8 +54,8 @@ AppVerName={#MyAppName} {#MyAppVersion}-{#MyPlatform}-{#MyGitRevision}{#MySuffix
|
|||||||
AppPublisher={#MyAppPublisher}
|
AppPublisher={#MyAppPublisher}
|
||||||
AppCopyright=Copyright 2019 {#MyAppPublisher}
|
AppCopyright=Copyright 2019 {#MyAppPublisher}
|
||||||
; **** AppCopyright=Copyright {GetDateTimeString('yyyy', #0, #0)} {%MyAppPublisher}
|
; **** AppCopyright=Copyright {GetDateTimeString('yyyy', #0, #0)} {%MyAppPublisher}
|
||||||
DefaultDirName={pf}\OSCAR
|
DefaultDirName={pf}\{#MyDirName}
|
||||||
DefaultGroupName={#MyAppName}
|
DefaultGroupName={#MyGroupName}
|
||||||
OutputDir=.\Installer
|
OutputDir=.\Installer
|
||||||
#if MyReleaseStatus == "r"
|
#if MyReleaseStatus == "r"
|
||||||
OutputBaseFilename={#MyAppName}-{#MyAppVersion}-{#MyPlatform}{#MySuffix}
|
OutputBaseFilename={#MyAppName}-{#MyAppVersion}-{#MyPlatform}{#MySuffix}
|
||||||
@ -51,7 +67,7 @@ Compression=lzma
|
|||||||
SolidCompression=yes
|
SolidCompression=yes
|
||||||
VersionInfoCompany={#MyAppPublisher}
|
VersionInfoCompany={#MyAppPublisher}
|
||||||
VersionInfoProductName={#MyAppName}
|
VersionInfoProductName={#MyAppName}
|
||||||
UninstallDisplayName={#MyAppName}
|
UninstallDisplayName={#MyGroupName}
|
||||||
UninstallDisplayIcon={app}\{#MyAppExeName}
|
UninstallDisplayIcon={app}\{#MyAppExeName}
|
||||||
|
|
||||||
VersionInfoVersion={#MyMajorVersion}.{#MyMinorVersion}.{#MyRevision}.{#MyBuildNumber}
|
VersionInfoVersion={#MyMajorVersion}.{#MyMinorVersion}.{#MyRevision}.{#MyBuildNumber}
|
||||||
@ -92,9 +108,9 @@ Source: ".\Release\*"; DestDir: "{app}"; Flags: ignoreversion recursesubdirs cre
|
|||||||
; NOTE: Don't use "Flags: ignoreversion" on any shared system files
|
; NOTE: Don't use "Flags: ignoreversion" on any shared system files
|
||||||
|
|
||||||
[Icons]
|
[Icons]
|
||||||
Name: "{group}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"
|
Name: "{group}\{#MyGroupName}"; Filename: "{app}\{#MyAppExeName}"
|
||||||
Name: "{group}\{cm:UninstallProgram,{#MyAppName}}"; Filename: "{uninstallexe}"
|
Name: "{group}\{cm:UninstallProgram,{#MyGroupName}}"; Filename: "{uninstallexe}"
|
||||||
Name: "{commondesktop}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: desktopicon
|
Name: "{commondesktop}\{#MyGroupName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: desktopicon
|
||||||
; Name: "{userappdata}\Microsoft\Internet Explorer\Quick Launch\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: quicklaunchicon
|
; Name: "{userappdata}\Microsoft\Internet Explorer\Quick Launch\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: quicklaunchicon
|
||||||
|
|
||||||
[Messages]
|
[Messages]
|
||||||
|
@ -1567,6 +1567,32 @@ void gGraphView::paintGL()
|
|||||||
|
|
||||||
QString gGraphView::getRangeString()
|
QString gGraphView::getRangeString()
|
||||||
{
|
{
|
||||||
|
QDateTime st = QDateTime::fromMSecsSinceEpoch(m_minx);
|
||||||
|
QDateTime et = QDateTime::fromMSecsSinceEpoch(m_maxx);
|
||||||
|
|
||||||
|
QDate std = st.date();
|
||||||
|
QDate etd = et.date();
|
||||||
|
|
||||||
|
// Format if Begin and End are on different days
|
||||||
|
if (std != etd) {
|
||||||
|
QString txt = st.toString(" d MMM [ HH:mm:ss") + " - " + et.toString("HH:mm:ss ] d MMM yyyy");
|
||||||
|
return txt;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Range is within one (local) day
|
||||||
|
qint64 diff = m_maxx - m_minx;
|
||||||
|
QString fmt;
|
||||||
|
|
||||||
|
if (diff > 60000) {
|
||||||
|
fmt = "HH:mm:ss";
|
||||||
|
} else {
|
||||||
|
fmt = "HH:mm:ss:zzz";
|
||||||
|
}
|
||||||
|
QString txt = st.toString(QObject::tr("d MMM yyyy [ %1 - %2 ]").arg(fmt).arg(et.toString(fmt))) ;
|
||||||
|
|
||||||
|
return txt;
|
||||||
|
|
||||||
|
/***** WTF is this code trying to do? Replaced by above 8/9/2019
|
||||||
// a note about time zone usage here
|
// a note about time zone usage here
|
||||||
// even though this string will be displayed to the user
|
// even though this string will be displayed to the user
|
||||||
// the graph is drawn using UTC times, so no conversion
|
// the graph is drawn using UTC times, so no conversion
|
||||||
@ -1579,7 +1605,7 @@ QString gGraphView::getRangeString()
|
|||||||
|
|
||||||
qint64 diff = m_maxx - m_minx;
|
qint64 diff = m_maxx - m_minx;
|
||||||
|
|
||||||
if (diff > 86400000) {
|
if (diff > 86400000) { // 86400000 is one day, in milliseconds
|
||||||
int days = ceil(double(m_maxx-m_minx) / 86400000.0);
|
int days = ceil(double(m_maxx-m_minx) / 86400000.0);
|
||||||
|
|
||||||
qint64 minx = floor(double(m_minx)/86400000.0);
|
qint64 minx = floor(double(m_minx)/86400000.0);
|
||||||
@ -1597,12 +1623,11 @@ QString gGraphView::getRangeString()
|
|||||||
} else {
|
} else {
|
||||||
fmt = "HH:mm:ss:zzz";
|
fmt = "HH:mm:ss:zzz";
|
||||||
}
|
}
|
||||||
QDateTime st = QDateTime::fromMSecsSinceEpoch(m_minx, Qt::UTC);
|
|
||||||
QDateTime et = QDateTime::fromMSecsSinceEpoch(m_maxx, Qt::UTC);
|
|
||||||
|
|
||||||
QString txt = st.toString(QObject::tr("d MMM [ %1 - %2 ]").arg(fmt).arg(et.toString(fmt))) ;
|
QString txt = st.toString(QObject::tr("d MMM [ %1 - %2 ]").arg(fmt).arg(et.toString(fmt))) ;
|
||||||
|
|
||||||
return txt;
|
return txt;
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
void gGraphView::leaveEvent(QEvent * event)
|
void gGraphView::leaveEvent(QEvent * event)
|
||||||
@ -3252,7 +3277,6 @@ void gGraphView::keyPressEvent(QKeyEvent *event)
|
|||||||
//qDebug() << "Keypress??";
|
//qDebug() << "Keypress??";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void gGraphView::setDay(Day *day)
|
void gGraphView::setDay(Day *day)
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -3264,6 +3288,7 @@ void gGraphView::setDay(Day *day)
|
|||||||
|
|
||||||
ResetBounds(false);
|
ResetBounds(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool gGraphView::isEmpty()
|
bool gGraphView::isEmpty()
|
||||||
{
|
{
|
||||||
bool res = true;
|
bool res = true;
|
||||||
|
@ -90,23 +90,35 @@ const QString getDeveloperDomain()
|
|||||||
const QString getAppName()
|
const QString getAppName()
|
||||||
{
|
{
|
||||||
QString name = STR_AppName;
|
QString name = STR_AppName;
|
||||||
if ((GIT_BRANCH != "master") ||
|
|
||||||
(!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) ||
|
// Append branch if there is a branch specified
|
||||||
(ReleaseStatus.compare("rc", Qt::CaseInsensitive)==0) ||
|
if (GIT_BRANCH != "master") {
|
||||||
(ReleaseStatus.compare("beta", Qt::CaseInsensitive)==0)))) {
|
|
||||||
name += "-"+GIT_BRANCH;
|
name += "-"+GIT_BRANCH;
|
||||||
|
// qDebug() << "getAppName, not master, name is" << name << "branch is" << GIT_BRANCH;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Append "-test" if not release
|
||||||
|
else if (!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) ||
|
||||||
|
(ReleaseStatus.compare("rc", Qt::CaseInsensitive)==0) )) {
|
||||||
|
name += "-test";
|
||||||
|
// qDebug() << "getAppName, not release, name is" << name << "release type is" << ReleaseStatus;
|
||||||
|
}
|
||||||
|
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
|
|
||||||
const QString getModifiedAppData()
|
const QString getModifiedAppData()
|
||||||
{
|
{
|
||||||
QString appdata = STR_AppData;
|
QString appdata = STR_AppData;
|
||||||
if ((GIT_BRANCH != "master") ||
|
|
||||||
(!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) ||
|
// Append branch if there is a branch specified
|
||||||
(ReleaseStatus.compare("rc", Qt::CaseInsensitive)==0) ||
|
if (GIT_BRANCH != "master")
|
||||||
(ReleaseStatus.compare("beta", Qt::CaseInsensitive)==0)))) {
|
|
||||||
appdata += "-"+GIT_BRANCH;
|
appdata += "-"+GIT_BRANCH;
|
||||||
|
|
||||||
|
// Append "-test" if not release
|
||||||
|
else if (!((ReleaseStatus.compare("r", Qt::CaseInsensitive)==0) ||
|
||||||
|
(ReleaseStatus.compare("rc", Qt::CaseInsensitive)==0) )) {
|
||||||
|
appdata += "-test";
|
||||||
}
|
}
|
||||||
return appdata;
|
return appdata;
|
||||||
}
|
}
|
||||||
@ -230,7 +242,7 @@ QStringList makeBuildInfo (QString relinfo, QString forcedEngine){
|
|||||||
branch = QObject::tr("Branch:") + " " + GIT_BRANCH + ", ";
|
branch = QObject::tr("Branch:") + " " + GIT_BRANCH + ", ";
|
||||||
}
|
}
|
||||||
buildInfo << branch + (QObject::tr("Revision")) + " " + GIT_REVISION;
|
buildInfo << branch + (QObject::tr("Revision")) + " " + GIT_REVISION;
|
||||||
if (GIT_BRANCH != "master")
|
if (getAppName() != STR_AppName) // Report any non-standard app key
|
||||||
buildInfo << (QObject::tr("App key:") + " " + getAppName());
|
buildInfo << (QObject::tr("App key:") + " " + getAppName());
|
||||||
buildInfo << QString("");
|
buildInfo << QString("");
|
||||||
buildInfo << (QObject::tr("Operating system:") + " " + QSysInfo::prettyProductName());
|
buildInfo << (QObject::tr("Operating system:") + " " + QSysInfo::prettyProductName());
|
||||||
|
@ -226,6 +226,7 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
|
|||||||
{ "560P", 0, 4, "REMstar Auto (System One 60 Series)" },
|
{ "560P", 0, 4, "REMstar Auto (System One 60 Series)" },
|
||||||
{ "560PBT", 0, 4, "REMstar Auto (System One 60 Series)" },
|
{ "560PBT", 0, 4, "REMstar Auto (System One 60 Series)" },
|
||||||
{ "561P", 0, 4, "REMstar Auto (System One 60 Series)" },
|
{ "561P", 0, 4, "REMstar Auto (System One 60 Series)" },
|
||||||
|
{ "562P", 0, 4, "REMstar Auto (System One 60 Series)" },
|
||||||
{ "660P", 0, 4, "BiPAP Pro (System One 60 Series)" },
|
{ "660P", 0, 4, "BiPAP Pro (System One 60 Series)" },
|
||||||
{ "760P", 0, 4, "BiPAP Auto (System One 60 Series)" },
|
{ "760P", 0, 4, "BiPAP Auto (System One 60 Series)" },
|
||||||
|
|
||||||
@ -501,7 +502,9 @@ void parseModel(MachineInfo & info, const QString & modelnum)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (series == nullptr) {
|
if (series == nullptr) {
|
||||||
qWarning() << "unknown series for" << name << modelnum;
|
if (modelnum != "100X100") { // Bogus model number seen in empty C0/Clear0 directories.
|
||||||
|
qWarning() << "unknown series for" << name << modelnum;
|
||||||
|
}
|
||||||
series = "unknown";
|
series = "unknown";
|
||||||
}
|
}
|
||||||
info.series = QObject::tr(series);
|
info.series = QObject::tr(series);
|
||||||
@ -2816,7 +2819,6 @@ void SmoothEventList(Session * session, EventList * ev, ChannelID code)
|
|||||||
|
|
||||||
|
|
||||||
// 750P is F0V2; 550P is F0V2/F0V3; 450P is F0V3; 460P, 560P[BT], 660P, 760P are F0V4
|
// 750P is F0V2; 550P is F0V2/F0V3; 450P is F0V3; 460P, 560P[BT], 660P, 760P are F0V4
|
||||||
// 200X, 400X, 400G, 500X, 502G, 600X, 700X are F0V6
|
|
||||||
bool PRS1Import::ParseF0Events()
|
bool PRS1Import::ParseF0Events()
|
||||||
{
|
{
|
||||||
// Required channels
|
// Required channels
|
||||||
@ -2984,8 +2986,12 @@ bool PRS1Import::ParseF0Events()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
bool PRS1DataChunk::ParseEventsF0V234(CPAPMode mode)
|
||||||
{
|
{
|
||||||
|
if (this->family != 0 || this->familyVersion < 2 || this->familyVersion > 4) {
|
||||||
|
qWarning() << "ParseEventsF0V234 called with family" << this->family << "familyVersion" << this->familyVersion;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
unsigned char code=0;
|
unsigned char code=0;
|
||||||
|
|
||||||
EventDataType data0, data1, data2;
|
EventDataType data0, data1, data2;
|
||||||
@ -3000,7 +3006,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
|
|
||||||
int size = this->m_data.size();
|
int size = this->m_data.size();
|
||||||
|
|
||||||
bool FV3 = (this->fileVersion == 3);
|
CHECK_VALUE(this->fileVersion, 2);
|
||||||
unsigned char * buffer = (unsigned char *)this->m_data.data();
|
unsigned char * buffer = (unsigned char *)this->m_data.data();
|
||||||
|
|
||||||
for (pos = 0; pos < size;) {
|
for (pos = 0; pos < size;) {
|
||||||
@ -3035,13 +3041,13 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
|
|
||||||
case 0x00: // Unknown 00
|
case 0x00: // Unknown 00
|
||||||
this->AddEvent(new PRS1UnknownValueEvent(code, t, buffer[pos++]));
|
this->AddEvent(new PRS1UnknownValueEvent(code, t, buffer[pos++]));
|
||||||
if (((this->family == 0) && (this->familyVersion >= 4)) || (this->fileVersion == 3)){
|
if (((this->family == 0) && (this->familyVersion == 4))){
|
||||||
pos++;
|
pos++;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x01: // Unknown
|
case 0x01: // Unknown
|
||||||
if ((this->family == 0) && (this->familyVersion >= 4)) {
|
if ((this->family == 0) && (this->familyVersion == 4)) {
|
||||||
this->AddEvent(new PRS1CPAPEvent(t, buffer[pos++]));
|
this->AddEvent(new PRS1CPAPEvent(t, buffer[pos++]));
|
||||||
} else {
|
} else {
|
||||||
this->AddEvent(new PRS1UnknownValueEvent(code, t, 0));
|
this->AddEvent(new PRS1UnknownValueEvent(code, t, 0));
|
||||||
@ -3049,7 +3055,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x02: // Pressure
|
case 0x02: // Pressure
|
||||||
if ((this->family == 0) && (this->familyVersion >= 4)) { // BiPAP Pressure
|
if ((this->family == 0) && (this->familyVersion == 4)) { // BiPAP Pressure
|
||||||
data0 = buffer[pos++];
|
data0 = buffer[pos++];
|
||||||
data1 = buffer[pos++];
|
data1 = buffer[pos++];
|
||||||
this->AddEvent(new PRS1IPAPEvent(t, data1));
|
this->AddEvent(new PRS1IPAPEvent(t, data1));
|
||||||
@ -3060,9 +3066,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x03: // BIPAP Pressure
|
case 0x03: // BIPAP Pressure
|
||||||
if (FV3) {
|
{
|
||||||
this->AddEvent(new PRS1CPAPEvent(t, buffer[pos++]));
|
|
||||||
} else {
|
|
||||||
data0 = buffer[pos++];
|
data0 = buffer[pos++];
|
||||||
data1 = buffer[pos++];
|
data1 = buffer[pos++];
|
||||||
this->AddEvent(new PRS1IPAPEvent(t, data1));
|
this->AddEvent(new PRS1IPAPEvent(t, data1));
|
||||||
@ -3105,7 +3109,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
data1 = buffer[pos+1];
|
data1 = buffer[pos+1];
|
||||||
pos += 2;
|
pos += 2;
|
||||||
|
|
||||||
if (this->familyVersion >= 4) {
|
if (this->familyVersion == 4) {
|
||||||
// might not doublerize on older machines?
|
// might not doublerize on older machines?
|
||||||
// data0 *= 2;
|
// data0 *= 2;
|
||||||
}
|
}
|
||||||
@ -3122,7 +3126,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
|
|
||||||
case 0x0e: // Unknown
|
case 0x0e: // Unknown
|
||||||
data0 = buffer[pos + 1] << 8 | buffer[pos];
|
data0 = buffer[pos + 1] << 8 | buffer[pos];
|
||||||
if (this->familyVersion >= 4) {
|
if (this->familyVersion == 4) {
|
||||||
// might not doublerize on older machines?
|
// might not doublerize on older machines?
|
||||||
data0 *= 2;
|
data0 *= 2;
|
||||||
}
|
}
|
||||||
@ -3134,7 +3138,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
|
|
||||||
case 0x0f: // Cheyne Stokes Respiration
|
case 0x0f: // Cheyne Stokes Respiration
|
||||||
data0 = (buffer[pos + 1] << 8 | buffer[pos]);
|
data0 = (buffer[pos + 1] << 8 | buffer[pos]);
|
||||||
if (this->familyVersion >= 4) {
|
if (this->familyVersion == 4) {
|
||||||
// might not doublerize on older machines
|
// might not doublerize on older machines
|
||||||
data0 *= 2;
|
data0 *= 2;
|
||||||
}
|
}
|
||||||
@ -3151,7 +3155,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
|
|
||||||
case 0x10: // Large Leak
|
case 0x10: // Large Leak
|
||||||
data0 = buffer[pos + 1] << 8 | buffer[pos];
|
data0 = buffer[pos + 1] << 8 | buffer[pos];
|
||||||
if (this->familyVersion >= 4) {
|
if (this->familyVersion == 4) {
|
||||||
// might not doublerize on older machines
|
// might not doublerize on older machines
|
||||||
data0 *= 2;
|
data0 *= 2;
|
||||||
}
|
}
|
||||||
@ -3166,7 +3170,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
this->AddEvent(new PRS1TotalLeakEvent(t, data0));
|
this->AddEvent(new PRS1TotalLeakEvent(t, data0));
|
||||||
this->AddEvent(new PRS1SnoreEvent(t, data1));
|
this->AddEvent(new PRS1SnoreEvent(t, data1));
|
||||||
|
|
||||||
if ((this->family == 0) && (this->familyVersion >= 4)) {
|
if ((this->family == 0) && (this->familyVersion == 4)) {
|
||||||
// EPAP / Flex Pressure
|
// EPAP / Flex Pressure
|
||||||
data0 = buffer[pos++];
|
data0 = buffer[pos++];
|
||||||
|
|
||||||
@ -3187,7 +3191,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
// Could end here, but I've seen data sets valid data after!!!
|
// Could end here, but I've seen data sets valid data after!!!
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
/*
|
||||||
case 0x14: // DreamStation Hypopnea
|
case 0x14: // DreamStation Hypopnea
|
||||||
data0 = buffer[pos++];
|
data0 = buffer[pos++];
|
||||||
this->AddEvent(new PRS1HypopneaEvent(t - data0, data0));
|
this->AddEvent(new PRS1HypopneaEvent(t - data0, data0));
|
||||||
@ -3197,7 +3201,7 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
data0 = buffer[pos++];
|
data0 = buffer[pos++];
|
||||||
this->AddEvent(new PRS1HypopneaEvent(t - data0, data0));
|
this->AddEvent(new PRS1HypopneaEvent(t - data0, data0));
|
||||||
break;
|
break;
|
||||||
|
*/
|
||||||
default:
|
default:
|
||||||
// ERROR!!!
|
// ERROR!!!
|
||||||
qWarning() << "Some new fandangled PRS1 code detected in" << this->sessionid << hex
|
qWarning() << "Some new fandangled PRS1 code detected in" << this->sessionid << hex
|
||||||
@ -3212,6 +3216,381 @@ bool PRS1DataChunk::ParseEventsF0(CPAPMode mode)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 200X, 400X, 400G, 500X, 502G, 600X, 700X are F0V6
|
||||||
|
// Originally a copy of PRS1Import::ParseF0Events, modified based on F5V3/F3V6 importers and comparison to reports
|
||||||
|
bool PRS1Import::ParseEventsF0V6()
|
||||||
|
{
|
||||||
|
// Required channels
|
||||||
|
EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event);
|
||||||
|
EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event);
|
||||||
|
EventList *CA = session->AddEventList(CPAP_ClearAirway, EVL_Event);
|
||||||
|
|
||||||
|
EventList *LL = session->AddEventList(CPAP_LargeLeak, EVL_Event);
|
||||||
|
EventList *TOTLEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event);
|
||||||
|
EventList *LEAK = session->AddEventList(CPAP_Leak, EVL_Event);
|
||||||
|
EventList *PB = session->AddEventList(CPAP_PB, EVL_Event);
|
||||||
|
EventList *FL = session->AddEventList(CPAP_FlowLimit, EVL_Event);
|
||||||
|
EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event);
|
||||||
|
EventList *VS = session->AddEventList(CPAP_VSnore, EVL_Event);
|
||||||
|
EventList *VS2 = session->AddEventList(CPAP_VSnore2, EVL_Event);
|
||||||
|
EventList *PP = session->AddEventList(CPAP_PressurePulse, EVL_Event);
|
||||||
|
EventList *RE = session->AddEventList(CPAP_RERA, EVL_Event);
|
||||||
|
|
||||||
|
|
||||||
|
// On-demand channels
|
||||||
|
ChannelID Codes[] = {
|
||||||
|
PRS1_00, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, PRS1_0E
|
||||||
|
};
|
||||||
|
|
||||||
|
int ncodes = sizeof(Codes) / sizeof(ChannelID);
|
||||||
|
EventList *Code[0x20] = {0};
|
||||||
|
|
||||||
|
Code[0x0e] = session->AddEventList(PRS1_0E, EVL_Event);
|
||||||
|
|
||||||
|
EventList *PRESSURE = nullptr;
|
||||||
|
EventList *EPAP = nullptr;
|
||||||
|
EventList *IPAP = nullptr;
|
||||||
|
EventList *PS = nullptr;
|
||||||
|
|
||||||
|
|
||||||
|
// Unintentional leak calculation, see zMaskProfile:calcLeak in calcs.cpp for explanation
|
||||||
|
EventDataType currentPressure=0, leak;
|
||||||
|
|
||||||
|
bool calcLeaks = p_profile->cpap->calculateUnintentionalLeaks();
|
||||||
|
EventDataType lpm4 = p_profile->cpap->custom4cmH2OLeaks();
|
||||||
|
EventDataType lpm20 = p_profile->cpap->custom20cmH2OLeaks();
|
||||||
|
|
||||||
|
EventDataType lpm = lpm20 - lpm4;
|
||||||
|
EventDataType ppm = lpm / 16.0;
|
||||||
|
|
||||||
|
CPAPMode mode = (CPAPMode) session->settings[CPAP_Mode].toInt();
|
||||||
|
|
||||||
|
qint64 duration;
|
||||||
|
qint64 t = qint64(event->timestamp) * 1000L;
|
||||||
|
session->updateFirst(t);
|
||||||
|
|
||||||
|
bool ok;
|
||||||
|
ok = event->ParseEvents(mode);
|
||||||
|
|
||||||
|
for (int i=0; i < event->m_parsedData.count(); i++) {
|
||||||
|
PRS1ParsedEvent* e = event->m_parsedData.at(i);
|
||||||
|
t = qint64(event->timestamp + e->m_start) * 1000L;
|
||||||
|
|
||||||
|
switch (e->m_type) {
|
||||||
|
case PRS1CPAPEvent::TYPE:
|
||||||
|
if (!PRESSURE) {
|
||||||
|
if (!(PRESSURE = session->AddEventList(CPAP_Pressure, EVL_Event, e->m_gain))) { return false; }
|
||||||
|
}
|
||||||
|
PRESSURE->AddEvent(t, e->m_value);
|
||||||
|
currentPressure = e->m_value;
|
||||||
|
break;
|
||||||
|
case PRS1IPAPEvent::TYPE:
|
||||||
|
if(!IPAP) {
|
||||||
|
if (!(IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, e->m_gain))) { return false; }
|
||||||
|
}
|
||||||
|
IPAP->AddEvent(t, e->m_value);
|
||||||
|
currentPressure = e->m_value;
|
||||||
|
break;
|
||||||
|
case PRS1EPAPEvent::TYPE:
|
||||||
|
if (!EPAP) {
|
||||||
|
if (!(EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, e->m_gain))) { return false; }
|
||||||
|
}
|
||||||
|
if(!PS) {
|
||||||
|
if (!(PS = session->AddEventList(CPAP_PS, EVL_Event, e->m_gain))) { return false; }
|
||||||
|
}
|
||||||
|
EPAP->AddEvent(t, e->m_value);
|
||||||
|
PS->AddEvent(t, currentPressure - e->m_value); // Pressure Support
|
||||||
|
break;
|
||||||
|
case PRS1ObstructiveApneaEvent::TYPE:
|
||||||
|
OA->AddEvent(t, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1ClearAirwayEvent::TYPE:
|
||||||
|
CA->AddEvent(t, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1HypopneaEvent::TYPE:
|
||||||
|
HY->AddEvent(t, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1FlowLimitationEvent::TYPE:
|
||||||
|
FL->AddEvent(t, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1PeriodicBreathingEvent::TYPE:
|
||||||
|
// TODO: The graphs silently treat the timestamp of a span as an end time rather than start (see gFlagsLine::paint).
|
||||||
|
// Decide whether to preserve that behavior or change it universally and update either this code or comment.
|
||||||
|
duration = e->m_duration * 1000L;
|
||||||
|
PB->AddEvent(t + duration, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1LargeLeakEvent::TYPE:
|
||||||
|
// TODO: see PB comment above.
|
||||||
|
duration = e->m_duration * 1000L;
|
||||||
|
LL->AddEvent(t + duration, e->m_duration);
|
||||||
|
break;
|
||||||
|
case PRS1TotalLeakEvent::TYPE:
|
||||||
|
TOTLEAK->AddEvent(t, e->m_value);
|
||||||
|
leak = e->m_value;
|
||||||
|
// F0V6 doesn't appear to report non-total leak
|
||||||
|
if (calcLeaks) { // Much Quicker doing this here than the recalc method.
|
||||||
|
leak -= (((currentPressure/10.0f) - 4.0) * ppm + lpm4);
|
||||||
|
if (leak < 0) leak = 0;
|
||||||
|
LEAK->AddEvent(t, leak);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PRS1SnoreEvent::TYPE: // snore count that shows up in flags but not waveform
|
||||||
|
// TODO: The numeric snore graph is the right way to present this information,
|
||||||
|
// but it needs to be shifted left 2 minutes, since it's not a starting value
|
||||||
|
// but a past statistic.
|
||||||
|
SNORE->AddEvent(t, e->m_value);
|
||||||
|
if (e->m_value > 0) {
|
||||||
|
// TODO: currently these get drawn on our waveforms, but they probably shouldn't,
|
||||||
|
// since they don't have a precise timestamp. They should continue to be drawn
|
||||||
|
// on the flags overview.
|
||||||
|
VS2->AddEvent(t, 0);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PRS1VibratorySnoreEvent::TYPE: // real VS marker on waveform
|
||||||
|
// TODO: These don't need to be drawn separately on the flag overview, since
|
||||||
|
// they're presumably included in the overall snore count statistic. They should
|
||||||
|
// continue to be drawn on the waveform, due to their precise timestamp.
|
||||||
|
VS->AddEvent(t, 0);
|
||||||
|
break;
|
||||||
|
case PRS1RERAEvent::TYPE:
|
||||||
|
RE->AddEvent(t, e->m_value);
|
||||||
|
break;
|
||||||
|
case PRS1PressurePulseEvent::TYPE:
|
||||||
|
PP->AddEvent(t, e->m_value);
|
||||||
|
break;
|
||||||
|
case PRS1UnknownValueEvent::TYPE:
|
||||||
|
{
|
||||||
|
int code = ((PRS1UnknownValueEvent*) e)->m_code;
|
||||||
|
Q_ASSERT(code < ncodes);
|
||||||
|
if (!Code[code]) {
|
||||||
|
ChannelID cpapcode = Codes[(int)code];
|
||||||
|
Q_ASSERT(cpapcode); // any unknown codes returned by chunk parser should be given a channel above
|
||||||
|
if (!(Code[code] = session->AddEventList(cpapcode, EVL_Event, e->m_gain))) { return false; }
|
||||||
|
}
|
||||||
|
Code[code]->AddEvent(t, e->m_value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
qWarning() << "Unknown PRS1 event type" << (int) e->m_type;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
t = qint64(event->timestamp + event->duration) * 1000L;
|
||||||
|
session->updateLast(t);
|
||||||
|
session->m_cnt.clear();
|
||||||
|
session->m_cph.clear();
|
||||||
|
|
||||||
|
session->m_lastchan.clear();
|
||||||
|
session->m_firstchan.clear();
|
||||||
|
session->m_valuesummary[CPAP_Pressure].clear();
|
||||||
|
session->m_valuesummary.erase(session->m_valuesummary.find(CPAP_Pressure));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// DreamStation family 0 CPAP/APAP machines (400X-700X)
|
||||||
|
// Originally derived from F5V3 parsing + (incomplete) F0V234 parsing + sample data
|
||||||
|
bool PRS1DataChunk::ParseEventsF0V6(CPAPMode /*mode*/)
|
||||||
|
{
|
||||||
|
if (this->family != 0 || this->familyVersion != 6) {
|
||||||
|
qWarning() << "ParseEventsF0V6 called with family" << this->family << "familyVersion" << this->familyVersion;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const unsigned char * data = (unsigned char *)this->m_data.constData();
|
||||||
|
int chunk_size = this->m_data.size();
|
||||||
|
static const int minimum_sizes[] = { 2, 3, 4, 3, 3, 3, 3, 3, 3, 2, 3, 4, 3, 2, 5, 5, 5, 5, 4, 3, 3, 3 };
|
||||||
|
static const int ncodes = sizeof(minimum_sizes) / sizeof(int);
|
||||||
|
|
||||||
|
if (chunk_size < 1) {
|
||||||
|
// This does occasionally happen.
|
||||||
|
qDebug() << this->sessionid << "Empty event data";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ok = true;
|
||||||
|
int pos = 0, startpos;
|
||||||
|
int code, size;
|
||||||
|
int t = 0;
|
||||||
|
int elapsed, duration;
|
||||||
|
do {
|
||||||
|
code = data[pos++];
|
||||||
|
if (!this->hblock.contains(code)) {
|
||||||
|
qWarning() << this->sessionid << "missing hblock entry for event" << code;
|
||||||
|
ok = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
size = this->hblock[code];
|
||||||
|
if (code < ncodes) {
|
||||||
|
// make sure the handlers below don't go past the end of the buffer
|
||||||
|
if (size < minimum_sizes[code]) {
|
||||||
|
qWarning() << this->sessionid << "event" << code << "too small" << size << "<" << minimum_sizes[code];
|
||||||
|
ok = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} // else if it's past ncodes, we'll log its information below (rather than handle it)
|
||||||
|
if (pos + size > chunk_size) {
|
||||||
|
qWarning() << this->sessionid << "event" << code << "@" << pos << "longer than remaining chunk";
|
||||||
|
ok = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
startpos = pos;
|
||||||
|
if (code != 0x12) { // This one event has no timestamp
|
||||||
|
t += data[pos] | (data[pos+1] << 8);
|
||||||
|
pos += 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (code) {
|
||||||
|
//case 0x00: // never seen
|
||||||
|
case 0x01: // Pressure adjustment
|
||||||
|
// Matches pressure setting, both initial and when ramp button pressed.
|
||||||
|
// TODO: Have OSCAR treat CPAP adjustment events differently than (average?) stats below.
|
||||||
|
// Based on waveform reports, it looks like the pressure graph is drawn by
|
||||||
|
// interpolating between these pressure adjustments, by 0.5 cmH2O spaced evenly between
|
||||||
|
// adjustments. E.g. 6 at 28:11 and 7.3 at 29:05 results in the following dots:
|
||||||
|
// 6 at 28:11, 6.5 around 28:30, 7.0 around 28:50, 7(.3) at 29:05. That holds until
|
||||||
|
// subsequent "adjustment" of 7.3 at 30:09 followed by 8.0 at 30:19.
|
||||||
|
this->AddEvent(new PRS1CPAPEvent(t, data[pos++]));
|
||||||
|
break;
|
||||||
|
case 0x02: // Pressure adjustment (bi-level)
|
||||||
|
// TODO: Have OSCAR treat pressure adjustment events differently than (average?) stats below.
|
||||||
|
// See notes above on interpolation.
|
||||||
|
this->AddEvent(new PRS1IPAPEvent(t, data[pos+1]));
|
||||||
|
this->AddEvent(new PRS1EPAPEvent(t, data[pos])); // EPAP needs to be added second to calculate PS
|
||||||
|
break;
|
||||||
|
case 0x03: // Auto-CPAP starting pressure
|
||||||
|
// Most of the time this occurs, it's at the start and end of a session with
|
||||||
|
// the same pressure at both. Occasionally an additional event shows up in the
|
||||||
|
// middle of a session, and then the pressure at the end matches that.
|
||||||
|
// In these cases, the new pressure corresponds to the next night's starting
|
||||||
|
// pressure for auto-CPAP. It does not appear to have any effect on the current
|
||||||
|
// night's pressure, unless there's a substantial gap between sessions, in
|
||||||
|
// which case the next session may use the new starting pressure.
|
||||||
|
//CHECK_VALUE(data[pos], 40);
|
||||||
|
break;
|
||||||
|
case 0x04: // Pressure Pulse
|
||||||
|
duration = data[pos++]; // TODO: is this a duration?
|
||||||
|
this->AddEvent(new PRS1PressurePulseEvent(t, duration));
|
||||||
|
break;
|
||||||
|
case 0x05: // RERA
|
||||||
|
elapsed = data[pos++]; // based on sample waveform, the RERA is over after this
|
||||||
|
this->AddEvent(new PRS1RERAEvent(t - elapsed, 0));
|
||||||
|
break;
|
||||||
|
case 0x06: // Obstructive Apnea
|
||||||
|
// OA events are instantaneous flags with no duration: reviewing waveforms
|
||||||
|
// shows that the time elapsed between the flag and reporting often includes
|
||||||
|
// non-apnea breathing.
|
||||||
|
elapsed = data[pos++];
|
||||||
|
this->AddEvent(new PRS1ObstructiveApneaEvent(t - elapsed, 0));
|
||||||
|
break;
|
||||||
|
case 0x07: // Clear Airway Apnea
|
||||||
|
// CA events are instantaneous flags with no duration: reviewing waveforms
|
||||||
|
// shows that the time elapsed between the flag and reporting often includes
|
||||||
|
// non-apnea breathing.
|
||||||
|
elapsed = data[pos++];
|
||||||
|
this->AddEvent(new PRS1ClearAirwayEvent(t - elapsed, 0));
|
||||||
|
break;
|
||||||
|
//case 0x08: // never seen
|
||||||
|
//case 0x09: // never seen
|
||||||
|
//case 0x0a: // Hypopnea, see 0x15
|
||||||
|
case 0x0b: // Hypopnea
|
||||||
|
// TODO: How is this hypopnea different from events 0xa, 0x14 and 0x15?
|
||||||
|
// TODO: What is the first byte?
|
||||||
|
pos++; // unknown first byte?
|
||||||
|
elapsed = data[pos++]; // based on sample waveform, the hypopnea is over after this
|
||||||
|
this->AddEvent(new PRS1HypopneaEvent(t - elapsed, 0));
|
||||||
|
break;
|
||||||
|
case 0x0c: // Flow Limitation
|
||||||
|
// TODO: We should revisit whether this is elapsed or duration once (if)
|
||||||
|
// we start calculating flow limitations ourselves. Flow limitations aren't
|
||||||
|
// as obvious as OA/CA when looking at a waveform.
|
||||||
|
elapsed = data[pos++];
|
||||||
|
this->AddEvent(new PRS1FlowLimitationEvent(t - elapsed, 0));
|
||||||
|
break;
|
||||||
|
case 0x0d: // Vibratory Snore
|
||||||
|
// VS events are instantaneous flags with no duration, drawn on the official waveform.
|
||||||
|
// The current thinking is that these are the snores that cause a change in auto-titrating
|
||||||
|
// pressure. The snoring statistics below seem to be a total count. It's unclear whether
|
||||||
|
// the trigger for pressure change is severity or count or something else.
|
||||||
|
// no data bytes
|
||||||
|
this->AddEvent(new PRS1VibratorySnoreEvent(t, 0));
|
||||||
|
break;
|
||||||
|
case 0x0e: // ???
|
||||||
|
// 5 bytes like PB and LL, but what is it?
|
||||||
|
duration = 2 * (data[pos] | (data[pos+1] << 8)); // this looks like a 16-bit value, so may be duration like PB?
|
||||||
|
pos += 2;
|
||||||
|
elapsed = data[pos++]; // this is always 60 seconds unless it's at the end, so it seems like elapsed
|
||||||
|
CHECK_VALUES(elapsed, 60, 0);
|
||||||
|
//this->AddEvent(new PRS1PeriodicBreathingEvent(t - elapsed - duration, duration));
|
||||||
|
break;
|
||||||
|
case 0x0f: // Periodic Breathing
|
||||||
|
// PB events are reported some time after they conclude, and they do have a reported duration.
|
||||||
|
duration = 2 * (data[pos] | (data[pos+1] << 8));
|
||||||
|
pos += 2;
|
||||||
|
elapsed = data[pos++];
|
||||||
|
this->AddEvent(new PRS1PeriodicBreathingEvent(t - elapsed - duration, duration));
|
||||||
|
break;
|
||||||
|
case 0x10: // Large Leak
|
||||||
|
// LL events are reported some time after they conclude, and they do have a reported duration.
|
||||||
|
duration = 2 * (data[pos] | (data[pos+1] << 8));
|
||||||
|
pos += 2;
|
||||||
|
elapsed = data[pos++];
|
||||||
|
this->AddEvent(new PRS1LargeLeakEvent(t - elapsed - duration, duration));
|
||||||
|
break;
|
||||||
|
case 0x11: // Statistics
|
||||||
|
this->AddEvent(new PRS1TotalLeakEvent(t, data[pos++]));
|
||||||
|
this->AddEvent(new PRS1SnoreEvent(t, data[pos++]));
|
||||||
|
// Average pressure: this reads lower than the current CPAP set point when
|
||||||
|
// a flex mode is on, and exactly the current CPAP set point when off. For
|
||||||
|
// bi-level it's presumably an average of the actual pressures.
|
||||||
|
// TODO: What to do with this average pressure? Actual pressure adjustments are handled above.
|
||||||
|
//this->AddEvent(new PRS1EPAPEvent(t, data[pos++]));
|
||||||
|
break;
|
||||||
|
case 0x12: // Snore count per pressure
|
||||||
|
// Some sessions (with lots of ramps) have multiple of these, each with a
|
||||||
|
// different pressure. The total snore count across all of them matches the
|
||||||
|
// total found in the stats event.
|
||||||
|
if (data[pos] != 0) {
|
||||||
|
CHECK_VALUES(data[pos], 1, 2); // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP
|
||||||
|
}
|
||||||
|
//CHECK_VALUE(data[pos+1], 0x78); // pressure
|
||||||
|
//CHECK_VALUE(data[pos+2], 1); // 16-bit snore count
|
||||||
|
//CHECK_VALUE(data[pos+3], 0);
|
||||||
|
break;
|
||||||
|
//case 0x13: // never seen
|
||||||
|
case 0x0a: // Hypopnea
|
||||||
|
// TODO: Why does this hypopnea have a different event code?
|
||||||
|
// fall through
|
||||||
|
case 0x14: // Hypopnea
|
||||||
|
// TODO: Why does this hypopnea have a different event code?
|
||||||
|
// fall through
|
||||||
|
case 0x15: // Hypopnea
|
||||||
|
// TODO: We should revisit whether this is elapsed or duration once (if)
|
||||||
|
// we start calculating hypopneas ourselves. Their official definition
|
||||||
|
// is 40% reduction in flow lasting at least 10s.
|
||||||
|
duration = data[pos++];
|
||||||
|
this->AddEvent(new PRS1HypopneaEvent(t - duration, 0));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
qWarning() << "Unknown event:" << code << "in" << this->sessionid << "at" << startpos-1;
|
||||||
|
this->AddEvent(new PRS1UnknownDataEvent(m_data, startpos-1, size+1));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pos = startpos + size;
|
||||||
|
} while (ok && pos < chunk_size);
|
||||||
|
|
||||||
|
this->duration = t;
|
||||||
|
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
bool PRS1Import::ImportCompliance()
|
bool PRS1Import::ImportCompliance()
|
||||||
{
|
{
|
||||||
bool ok;
|
bool ok;
|
||||||
@ -5130,7 +5509,11 @@ bool PRS1DataChunk::ParseEvents(CPAPMode mode)
|
|||||||
bool ok = false;
|
bool ok = false;
|
||||||
switch (this->family) {
|
switch (this->family) {
|
||||||
case 0:
|
case 0:
|
||||||
ok = this->ParseEventsF0(mode);
|
if (this->familyVersion == 6) {
|
||||||
|
ok = this->ParseEventsF0V6(mode);
|
||||||
|
} else if (this->familyVersion >= 2 && this->familyVersion <= 4) {
|
||||||
|
ok = this->ParseEventsF0V234(mode);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
if (this->familyVersion == 6) {
|
if (this->familyVersion == 6) {
|
||||||
@ -5160,7 +5543,11 @@ bool PRS1Import::ParseEvents()
|
|||||||
if (!event) return false;
|
if (!event) return false;
|
||||||
switch (event->family) {
|
switch (event->family) {
|
||||||
case 0:
|
case 0:
|
||||||
res = ParseF0Events();
|
if (event->familyVersion == 6) {
|
||||||
|
res = this->ParseEventsF0V6();
|
||||||
|
} else {
|
||||||
|
res = this->ParseF0Events();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
// NOTE: The original comment in the header for ParseF3EventsV3 said there was a 1060P with fileVersion 3.
|
// NOTE: The original comment in the header for ParseF3EventsV3 said there was a 1060P with fileVersion 3.
|
||||||
|
@ -174,8 +174,11 @@ public:
|
|||||||
bool ParseEvents(CPAPMode mode);
|
bool ParseEvents(CPAPMode mode);
|
||||||
|
|
||||||
//! \brief Parse a single data chunk from a .002 file containing event data for a family 0 CPAP/APAP machine
|
//! \brief Parse a single data chunk from a .002 file containing event data for a family 0 CPAP/APAP machine
|
||||||
bool ParseEventsF0(CPAPMode mode);
|
bool ParseEventsF0V234(CPAPMode mode);
|
||||||
|
|
||||||
|
//! \brief Parse a single data chunk from a .002 file containing event data for a DreamStation family 0 CPAP/APAP machine
|
||||||
|
bool ParseEventsF0V6(CPAPMode mode);
|
||||||
|
|
||||||
//! \brief Parse a single data chunk from a .002 file containing event data for a family 3 ventilator family version 3 machine
|
//! \brief Parse a single data chunk from a .002 file containing event data for a family 3 ventilator family version 3 machine
|
||||||
bool ParseEventsF3V3(void);
|
bool ParseEventsF3V3(void);
|
||||||
|
|
||||||
@ -275,6 +278,8 @@ public:
|
|||||||
|
|
||||||
//! \brief Parse a single data chunk from a .002 file containing event data for a standard system one machine
|
//! \brief Parse a single data chunk from a .002 file containing event data for a standard system one machine
|
||||||
bool ParseF0Events();
|
bool ParseF0Events();
|
||||||
|
//! \brief Parse a single data chunk from a .002 file containing event data for a standard system one machine (family version 6)
|
||||||
|
bool ParseEventsF0V6();
|
||||||
//! \brief Parse a single data chunk from a .002 file containing event data for a AVAPS 1060P machine
|
//! \brief Parse a single data chunk from a .002 file containing event data for a AVAPS 1060P machine
|
||||||
bool ParseF3Events();
|
bool ParseF3Events();
|
||||||
//! \brief Parse a single data chunk from a .002 file containing event data for a family 3 ventilator machine (family version 6)
|
//! \brief Parse a single data chunk from a .002 file containing event data for a family 3 ventilator machine (family version 6)
|
||||||
|
@ -852,6 +852,28 @@ void Daily::ResetGraphLayout()
|
|||||||
void Daily::ResetGraphOrder()
|
void Daily::ResetGraphOrder()
|
||||||
{
|
{
|
||||||
GraphView->resetGraphOrder(true);
|
GraphView->resetGraphOrder(true);
|
||||||
|
|
||||||
|
// Enable all graphs (make them not hidden)
|
||||||
|
for (int i=0;i<ui->graphCombo->count();i++) {
|
||||||
|
// If disabled, emulate a click to enable the graph
|
||||||
|
if (!ui->graphCombo->itemData(i,Qt::UserRole).toBool()) {
|
||||||
|
qDebug() << "resetting graph" << i;
|
||||||
|
Daily::on_graphCombo_activated(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark all events as active
|
||||||
|
for (int i=0;i<ui->eventsCombo->count();i++) {
|
||||||
|
// If disabled, emulate a click to enable the event
|
||||||
|
ChannelID code = ui->eventsCombo->itemData(i, Qt::UserRole).toUInt();
|
||||||
|
schema::Channel * chan = &schema::channel[code];
|
||||||
|
if (!chan->enabled()) {
|
||||||
|
qDebug() << "resetting event" << i;
|
||||||
|
Daily::on_eventsCombo_activated(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset graph heights (and repaint)
|
||||||
ResetGraphLayout();
|
ResetGraphLayout();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2416,6 +2438,7 @@ void Daily::on_graphCombo_activated(int index)
|
|||||||
GraphView->updateScale();
|
GraphView->updateScale();
|
||||||
GraphView->redraw();
|
GraphView->redraw();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Daily::updateCube()
|
void Daily::updateCube()
|
||||||
{
|
{
|
||||||
//brick..
|
//brick..
|
||||||
@ -2495,7 +2518,6 @@ void Daily::updateGraphCombo()
|
|||||||
}
|
}
|
||||||
ui->graphCombo->setCurrentIndex(0);
|
ui->graphCombo->setCurrentIndex(0);
|
||||||
|
|
||||||
|
|
||||||
updateCube();
|
updateCube();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2504,7 +2526,6 @@ void Daily::on_eventsCombo_activated(int index)
|
|||||||
if (index<0)
|
if (index<0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|
||||||
ChannelID code = ui->eventsCombo->itemData(index, Qt::UserRole).toUInt();
|
ChannelID code = ui->eventsCombo->itemData(index, Qt::UserRole).toUInt();
|
||||||
schema::Channel * chan = &schema::channel[code];
|
schema::Channel * chan = &schema::channel[code];
|
||||||
|
|
||||||
|
@ -10,8 +10,10 @@ Which was written and copyright 2011-2018 © Mark Watkins
|
|||||||
<b>Changes and fixes in OSCAR <u>AFTER</u> v1.1.0-testing-3</b>
|
<b>Changes and fixes in OSCAR <u>AFTER</u> v1.1.0-testing-3</b>
|
||||||
<ul>
|
<ul>
|
||||||
<li>Portions of OSCAR are © 2019 by The OSCAR Team</li>
|
<li>Portions of OSCAR are © 2019 by The OSCAR Team</li>
|
||||||
<li>[new] </li>
|
<li>[new] Windows installers support Oscar, Oscar 32-bit, Oscar (test) and Oscar 32-bit (test)</li>
|
||||||
<li>[fix] </li>
|
<li>[fix] Release builds use a Settings key of OSCAR, Test builds use OSCAR-test, and Branch builds use OSCAR-branch. Default data directories are similarly named.</li>
|
||||||
|
<li>[fix] Date bar on bottom of Daily graph now in local time when no line cursor displayed, and formatting improved</li>
|
||||||
|
<li>[fix] View/Reset Graphs now enables all graphs and all event flags</li>
|
||||||
<li>[fix] Calendar date now formatted per national settings</li>
|
<li>[fix] Calendar date now formatted per national settings</li>
|
||||||
</ul>
|
</ul>
|
||||||
</p>
|
</p>
|
||||||
|
@ -50,7 +50,7 @@ void initTranslations()
|
|||||||
langNames["en_UK"] = "English (UK)";
|
langNames["en_UK"] = "English (UK)";
|
||||||
langNames["nl"] = "Nederlands";
|
langNames["nl"] = "Nederlands";
|
||||||
langNames["pt_BR"] = "Portugues (BR)";
|
langNames["pt_BR"] = "Portugues (BR)";
|
||||||
langNames["ro"] = "Romanian";
|
langNames["ro"] = "Românește";
|
||||||
|
|
||||||
langNames[DefaultLanguage]="English (US)";
|
langNames[DefaultLanguage]="English (US)";
|
||||||
|
|
||||||
@ -176,7 +176,7 @@ void initTranslations()
|
|||||||
QString qtLang = language.left(2);
|
QString qtLang = language.left(2);
|
||||||
if ( qtLang.compare("zh") == 0 )
|
if ( qtLang.compare("zh") == 0 )
|
||||||
qtLang.append("_CN");
|
qtLang.append("_CN");
|
||||||
qDebug() << "Loading" << langname << "translation" << "qt_" + qtLang + ".qm" << "from" << qtLangPath.toLocal8Bit().data();
|
qDebug() << "Loading" << langname << "QT translation" << "qt_" + qtLang + ".qm" << "from" << qtLangPath.toLocal8Bit().data();
|
||||||
QTranslator * qtranslator = new QTranslator();
|
QTranslator * qtranslator = new QTranslator();
|
||||||
|
|
||||||
if (!langfile.isEmpty() && !qtranslator->load("qt_" + qtLang + ".qm", qtLangPath)) {
|
if (!langfile.isEmpty() && !qtranslator->load("qt_" + qtLang + ".qm", qtLangPath)) {
|
||||||
@ -186,7 +186,7 @@ void initTranslations()
|
|||||||
qApp->installTranslator(qtranslator);
|
qApp->installTranslator(qtranslator);
|
||||||
|
|
||||||
// Install OSCAR translation files
|
// Install OSCAR translation files
|
||||||
qDebug() << "Loading" << langname << "translation" << langfile.toLocal8Bit().data() << "from" << langpath.toLocal8Bit().data();
|
qDebug() << "Loading" << langname << "OSCAR translation" << langfile.toLocal8Bit().data() << "from" << langpath.toLocal8Bit().data();
|
||||||
QTranslator * translator = new QTranslator();
|
QTranslator * translator = new QTranslator();
|
||||||
|
|
||||||
if (!langfile.isEmpty() && !translator->load(langfile, langpath)) {
|
if (!langfile.isEmpty() && !translator->load(langfile, langpath)) {
|
||||||
|
Loading…
Reference in New Issue
Block a user