New Feature:Daily,Search: added ability to search for any apnea by length (seconds)

This commit is contained in:
LoudSnorer 2023-12-12 08:26:27 -05:00
parent 5f7c5650fb
commit a35a54fb56
2 changed files with 180 additions and 25 deletions

View File

@ -6,8 +6,10 @@
* License. See the file COPYING in the main directory of the source code * License. See the file COPYING in the main directory of the source code
* for more details. */ * for more details. */
// Issues no clear of results between runs or between passes.
// automatically open event. in graph view for events TAB
#define TEST_MACROS_ENABLEDoff #define TEST_MACROS_ENABLED
#include <test_macros.h> #include <test_macros.h>
#include <QWidget> #include <QWidget>
@ -24,10 +26,12 @@
#include <QDoubleSpinBox> #include <QDoubleSpinBox>
#include <QtGlobal> #include <QtGlobal>
#include <QHeaderView> #include <QHeaderView>
#include <QCoreApplication>
#include "dailySearchTab.h" #include "dailySearchTab.h"
#include "SleepLib/day.h" #include "SleepLib/day.h"
#include "SleepLib/profiles.h" #include "SleepLib/profiles.h"
#include "daily.h" #include "daily.h"
#include "SleepLib/machine_common.h"
enum DS_COL { DS_COL_LEFT=0, DS_COL_RIGHT, DS_COL_MAX }; enum DS_COL { DS_COL_LEFT=0, DS_COL_RIGHT, DS_COL_MAX };
enum DS_ROW{ DS_ROW_HEADER, DS_ROW_DATA }; enum DS_ROW{ DS_ROW_HEADER, DS_ROW_DATA };
@ -44,7 +48,7 @@ enum DS_ROW{ DS_ROW_HEADER, DS_ROW_DATA };
|------------------------| |------------------------|
| control:cmd op value | | control:cmd op value |
+========================+ +========================+
| Progress | | Progress Bar |
+========================+ +========================+
| Summary | | Summary |
+========================+ +========================+
@ -187,7 +191,7 @@ void DailySearchTab::createUi() {
matchButton->setIcon(*m_icon_configure); matchButton->setIcon(*m_icon_configure);
matchButton->setStyleSheet( styleButton ); matchButton->setStyleSheet( styleButton );
clearButton->setStyleSheet( styleButton ); clearButton->setStyleSheet( styleButton );
startButton->setStyleSheet( styleButton ); //startButton->setStyleSheet( styleButton );
//matchButton->setSizePolicy( sizePolicyEE); //matchButton->setSizePolicy( sizePolicyEE);
//clearButton->setSizePolicy( sizePolicyEE); //clearButton->setSizePolicy( sizePolicyEE);
//startButton->setSizePolicy( sizePolicyEE); //startButton->setSizePolicy( sizePolicyEE);
@ -303,7 +307,6 @@ void DailySearchTab::updateEvents(ChannelID id,QString fullname) {
} }
void DailySearchTab::populateControl() { void DailySearchTab::populateControl() {
commandList->clear(); commandList->clear();
commandList->addItem(calculateMaxSize(tr("Notes"),ST_NOTES)); commandList->addItem(calculateMaxSize(tr("Notes"),ST_NOTES));
commandList->addItem(calculateMaxSize(tr("Notes containing"),ST_NOTES_STRING)); commandList->addItem(calculateMaxSize(tr("Notes containing"),ST_NOTES_STRING));
@ -313,6 +316,7 @@ void DailySearchTab::populateControl() {
commandList->addItem(calculateMaxSize(tr("Daily Duration"),ST_DAILY_USAGE)); commandList->addItem(calculateMaxSize(tr("Daily Duration"),ST_DAILY_USAGE));
commandList->addItem(calculateMaxSize(tr("Session Duration" ),ST_SESSION_LENGTH)); commandList->addItem(calculateMaxSize(tr("Session Duration" ),ST_SESSION_LENGTH));
commandList->addItem(calculateMaxSize(tr("Days Skipped"),ST_DAYS_SKIPPED)); commandList->addItem(calculateMaxSize(tr("Days Skipped"),ST_DAYS_SKIPPED));
commandList->addItem(calculateMaxSize(tr("Apnea Length"),ST_APNEA_LENGTH));
if ( !p_profile->cpap->clinicalMode() ) { if ( !p_profile->cpap->clinicalMode() ) {
commandList->addItem(calculateMaxSize(tr("Disabled Sessions"),ST_DISABLED_SESSIONS)); commandList->addItem(calculateMaxSize(tr("Disabled Sessions"),ST_DISABLED_SESSIONS));
} }
@ -514,6 +518,7 @@ QString DailySearchTab::valueToString(int value, QString defaultValue) {
case opWhole: case opWhole:
return QString().setNum(value); return QString().setNum(value);
break; break;
case secondsDisplayString:
case displayString: case displayString:
case opString: case opString:
return foundString; return foundString;
@ -638,6 +643,15 @@ void DailySearchTab::on_commandList_activated(QListWidgetItem* item) {
selectDouble->setValue(5.0); selectDouble->setValue(5.0);
selectDouble->setSingleStep(0.1); selectDouble->setSingleStep(0.1);
break; break;
case ST_APNEA_LENGTH :
DaysWithFileErrors = 0;
setResult(DS_ROW_HEADER,1,QDate(),tr("Set of Apnea:Length\nJumps to Date's Events"));
nextTab = TW_EVENTS ;
setoperation(OP_GE,secondsDisplayString);
selectInteger->setRange(0,999);
selectInteger->setValue(25);
setText(selectUnits,tr(" Seconds"));
break;
case ST_SESSION_LENGTH : case ST_SESSION_LENGTH :
setResult(DS_ROW_HEADER,1,QDate(),tr("Session Duration\nJumps to Date's Details")); setResult(DS_ROW_HEADER,1,QDate(),tr("Session Duration\nJumps to Date's Details"));
nextTab = TW_DETAILED ; nextTab = TW_DETAILED ;
@ -677,10 +691,12 @@ void DailySearchTab::on_commandList_activated(QListWidgetItem* item) {
if (operationOpCode == OP_NO_PARMS ) { if (operationOpCode == OP_NO_PARMS ) {
// auto start searching // auto start searching
setText(startButton,tr("Automatic start")); setText(startButton,tr("Automatic start"));
setColor(startButton,red);
startButtonMode=true; startButtonMode=true;
on_startButton_clicked(); on_startButton_clicked();
return; return;
} }
setColor(startButton,green);
return; return;
} }
@ -740,8 +756,8 @@ void DailySearchTab::updateValues(qint32 value) {
} }
void DailySearchTab::find(QDate& date) void DailySearchTab::find(QDate& date) {
{ QCoreApplication::processEvents();
bool found=false; bool found=false;
Qt::Alignment alignment=Qt::AlignCenter; Qt::Alignment alignment=Qt::AlignCenter;
Day* day = p_profile->GetDay(date); Day* day = p_profile->GetDay(date);
@ -830,6 +846,88 @@ void DailySearchTab::find(QDate& date)
found = compare (ahi , selectValue); found = compare (ahi , selectValue);
} }
break; break;
case ST_APNEA_LENGTH :
{
QList<Session *> sessions = day->getSessions(MT_CPAP);
QMap<ChannelID,int> values;
// find possible channeld to use
QList<ChannelID> apneaLikeChannels(ahiChannels.begin(),ahiChannels.end());
#if 1
if (p_profile->cpap->userEventFlagging()) {
apneaLikeChannels.push_back(CPAP_UserFlag1);
apneaLikeChannels.push_back(CPAP_UserFlag2);
}
#endif
apneaLikeChannels.push_back(CPAP_RERA);
QList<ChannelID> chans;
for( auto code : apneaLikeChannels) {
if (day->count(code)) { chans.push_back(code); }
}
bool errorFound = false;
QString result;
if (!chans.isEmpty()) {
for (Session* sess : sessions ) {
if (!sess->enabled()) continue;
auto keys = sess->eventlist.keys();
if (keys.size() <= 0) {
bool ok = sess->LoadSummary(false);
bool ok1 = sess->OpenEvents(false);
keys = sess->eventlist.keys();
if ((keys.size() <= 0) || !ok || !ok1 ) {
if ((keys.size() <= 0) || !ok || !ok1 ) {
errorFound |= true;
DEBUGFC O(day->date()) O(keys.size()) Q(daysSkipped) O("NO KEYS STILL") ;
// skip this channel
continue;
}
}
}
for( ChannelID code : chans) {
auto evlist = sess->eventlist.find(code);
if (evlist == sess->eventlist.end()) {
continue;
}
// Now we go through the event list for the *session* (not for the day)
for (int z=0;z<evlist.value().size();z++) {
EventList & ev=*(evlist.value()[z]);
for (quint32 o=0;o<ev.count();o++) {
int sec = evlist.value()[z]->raw(o);
if (compare (sec , selectValue)) {
// save value in map
auto it = values.find(code);
if (it == values.end() ) {
values.insert(code,sec);
} else {
// save max or min value in map
int saved_sec = it.value();
// save highest or lowest value.
if (compare (sec ,saved_sec) ) {
*it = sec;
}
}
}
}
}
}
}
}
for ( auto it= values.begin() ; it != values.end() ; it++) {
ChannelID code = it.key();
int value = it.value();
result += QString("%1:%2 ").arg(schema::channel[ code ].label()).arg(value);
}
if (errorFound) {
daysSkipped++;
return;
};
found = !result.isEmpty();
if (found) {
foundString = result;
alignment=Qt::AlignLeft;
}
}
break;
case ST_SESSION_LENGTH : case ST_SESSION_LENGTH :
{ {
bool valid=false; bool valid=false;
@ -888,8 +986,7 @@ void DailySearchTab::find(QDate& date)
return ; return ;
}; };
void DailySearchTab::search(QDate date) void DailySearchTab::search(QDate date) {
{
if (!date.isValid()) { if (!date.isValid()) {
qWarning() << "DailySearchTab::find invalid date." << date; qWarning() << "DailySearchTab::find invalid date." << date;
return; return;
@ -919,15 +1016,20 @@ void DailySearchTab::addItem(QDate date, QString value,Qt::Alignment alignment)
void DailySearchTab::endOfPass() { void DailySearchTab::endOfPass() {
startButtonMode=false; // display Continue; startButtonMode=false; // display Continue;
QString display; QString display;
if (DaysWithFileErrors) {
}
if ((passFound >= passDisplayLimit) && (daysProcessed<daysTotal)) { if ((passFound >= passDisplayLimit) && (daysProcessed<daysTotal)) {
startButton->setEnabled(true); startButton->setEnabled(true);
setText(startButton,(tr("Continue Search"))); setText(startButton,(tr("Continue Search")));
setColor(startButton,green);
} else if (daysFound>0) { } else if (daysFound>0) {
startButton->setEnabled(false); startButton->setEnabled(false);
setText(startButton,tr("End of Search")); setText(startButton,tr("End of Search"));
setColor(startButton,red);
} else { } else {
startButton->setEnabled(false); startButton->setEnabled(false);
setText(startButton,tr("No Matches")); setText(startButton,tr("No Matches"));
setColor(startButton,red);
} }
displayStatistics(); displayStatistics();
@ -974,11 +1076,11 @@ void DailySearchTab::on_operationButton_clicked() {
void DailySearchTab::on_matchButton_clicked() { void DailySearchTab::on_matchButton_clicked() {
setColor(startButton,grey);
setCommandPopupEnabled(!commandPopupEnabled); setCommandPopupEnabled(!commandPopupEnabled);
} }
void DailySearchTab::on_commandButton_clicked() void DailySearchTab::on_commandButton_clicked() {
{
setCommandPopupEnabled(true); setCommandPopupEnabled(true);
} }
@ -1021,6 +1123,11 @@ void DailySearchTab::setoperation(OpCode opCode,ValueMode mode) {
selectUnits->show(); selectUnits->show();
selectDouble->show(); selectDouble->show();
break; break;
case secondsDisplayString:
setText(selectUnits,tr(" Seconds"));
selectUnits->show();
selectInteger->show();
break;
case minutesToMs: case minutesToMs:
setText(selectUnits,tr(" Minutes")); setText(selectUnits,tr(" Minutes"));
selectUnits->show(); selectUnits->show();
@ -1065,9 +1172,11 @@ QSize DailySearchTab::textsize(QFont font ,QString text) {
void DailySearchTab::on_clearButton_clicked() void DailySearchTab::on_clearButton_clicked()
{ {
DaysWithFileErrors = 0 ;
searchTopic = ST_NONE; searchTopic = ST_NONE;
// make these button text back to start. // make these button text back to start.
startButton->setText(tr("Start Search")); startButton->setText(tr("Start Search"));
setColor(startButton,grey);
startButtonMode=true; startButtonMode=true;
startButton->setEnabled( false); startButton->setEnabled( false);
@ -1092,9 +1201,9 @@ void DailySearchTab::on_clearButton_clicked()
//controlWidget->show(); //controlWidget->show();
} }
void DailySearchTab::on_startButton_clicked() void DailySearchTab::on_startButton_clicked() {
{
hideResults(false); hideResults(false);
setColor(startButton,grey);
if (startButtonMode) { if (startButtonMode) {
search (latestDate ); search (latestDate );
startButtonMode=false; startButtonMode=false;
@ -1117,6 +1226,22 @@ void DailySearchTab::on_textEdited(QString ) {
criteriaChanged(); criteriaChanged();
} }
void DailySearchTab::setColor(QPushButton* button,QString color) {
#if 0
QPalette pal = button->palette();
pal.setColor(QPalette::Button, color);
button->setPalette(pal);
button->setAutoFillBackground(true);
#else
QString style=QString(
"QPushButton { color: black; border: 1px solid black; padding: 5px ;background-color:%1; }"
).arg(color);
//"QPushButton:disabled { background-color:#EEEEEE ;}"
button->setStyleSheet(style);
#endif
QCoreApplication::processEvents();
}
void DailySearchTab::displayStatistics() { void DailySearchTab::displayStatistics() {
QString extra; QString extra;
summaryProgress->show(); summaryProgress->show();
@ -1136,9 +1261,15 @@ void DailySearchTab::displayStatistics() {
if (extra.size()>0) { if (extra.size()>0) {
setText(summaryMinMax,extra); setText(summaryMinMax,extra);
summaryMinMax->show(); summaryMinMax->show();
} else {
if (DaysWithFileErrors) {
QString msg = tr("File errors:%1");
setText(summaryMinMax, QString(msg).arg(DaysWithFileErrors));
summaryMinMax->show();
} else { } else {
summaryMinMax->hide(); summaryMinMax->hide();
} }
}
summaryProgress->show(); summaryProgress->show();
summaryFound->show(); summaryFound->show();
} }
@ -1160,6 +1291,7 @@ void DailySearchTab::criteriaChanged() {
case hoursToMs: case hoursToMs:
selectValue = (int)(selectDouble->value()*3600000.0); //convert to ms selectValue = (int)(selectDouble->value()*3600000.0); //convert to ms
break; break;
case secondsDisplayString:
case displayWhole: case displayWhole:
case opWhole: case opWhole:
selectValue = selectInteger->value();; selectValue = selectInteger->value();;
@ -1175,6 +1307,7 @@ void DailySearchTab::criteriaChanged() {
commandButton->show(); commandButton->show();
setText(startButton,tr("Start Search")); setText(startButton,tr("Start Search"));
setColor(startButton,green);
startButtonMode=true; startButtonMode=true;
startButton->setEnabled( true); startButton->setEnabled( true);
@ -1215,22 +1348,29 @@ QString DailySearchTab::helpStr()
{ {
QStringList str; str QStringList str; str
<<tr("Finds days that match specified criteria.") <<"\n" <<tr("Finds days that match specified criteria.") <<"\n"
<<tr(" Searches from last day to first day.") <<"\n" <<"\n" <<tr(" Searches from last day to first day.") <<"\n"
<<tr(" Skips Days with no graphing data.") <<"\n"
<<"\n"
<<tr("First click on Match Button then select topic.") <<"\n" <<tr("First click on Match Button then select topic.") <<"\n"
<<tr(" Then click on the operation to modify it.") <<"\n" <<tr(" Then click on the operation to modify it.") <<"\n"
<<tr(" or update the value") <<"\n" <<tr(" or update the value") <<"\n"
<<tr("Topics without operations will automatically start.") <<"\n" <<"\n" <<"\n"
<<tr("Topics without operations will automatically start.") <<"\n"
<<"\n"
<<tr("Compare Operations: numberic or character. ") <<"\n" <<tr("Compare Operations: numberic or character. ") <<"\n"
<<tr(" Numberic Operations: ") <<" > , >= , < , <= , == , != " <<"\n" <<tr(" Numberic Operations: ") <<" > , >= , < , <= , == , != " <<"\n"
<<tr(" Character Operations: ") <<" == , *? " <<"\n" <<"\n" <<tr(" Character Operations: ") <<" == , *? " <<"\n"
<<"\n"
<<tr("Summary Line") <<"\n" <<tr("Summary Line") <<"\n"
<<tr(" Left:Summary - Number of Day searched") <<"\n" <<tr(" Left:Summary - Number of Day searched") <<"\n"
<<tr(" Center:Number of Items Found") <<"\n" <<tr(" Center:Number of Items Found") <<"\n"
<<tr(" Right:Minimum/Maximum for item searched") <<"\n" <<tr(" Right:Minimum/Maximum for item searched") <<"\n"
<<"\n"
<<tr("Result Table") <<"\n" <<tr("Result Table") <<"\n"
<<tr(" Column One: Date of match. Click selects date.") <<"\n" <<tr(" Column One: Date of match. Click selects date.") <<"\n"
<<tr(" Column two: Information. Click selects date.") <<"\n" <<tr(" Column two: Information. Click selects date.") <<"\n"
<<tr(" Then Jumps the appropiate tab.") <<"\n" <<"\n" <<tr(" Then Jumps the appropiate tab.") <<"\n"
<<"\n"
<<tr("Wildcard Pattern Matching:") <<" *? " <<"\n" <<tr("Wildcard Pattern Matching:") <<" *? " <<"\n"
<<tr(" Wildcards use 3 characters:") <<"\n" <<tr(" Wildcards use 3 characters:") <<"\n"
<<tr(" Asterisk") <<" * " <<" " <<tr(" Asterisk") <<" * " <<" "
@ -1238,7 +1378,8 @@ QString DailySearchTab::helpStr()
<<tr(" Backslash.") <<" \\ " <<"\n" <<tr(" Backslash.") <<" \\ " <<"\n"
<<tr(" Asterisk matches any number of characters.") <<"\n" <<tr(" Asterisk matches any number of characters.") <<"\n"
<<tr(" Question Mark matches a single character.") <<"\n" <<tr(" Question Mark matches a single character.") <<"\n"
<<tr(" Backslash matches next character.") <<"\n"; <<tr(" Backslash matches next character.") <<"\n\n"
;
return str.join(""); return str.join("");
} }
@ -1296,12 +1437,14 @@ void DailySearchTab::on_activated(GPushButton* item ) {
item->setIcon (*m_icon_selected); item->setIcon (*m_icon_selected);
daily->LoadDate( item->date() ); daily->LoadDate( item->date() );
if ((col!=0) && nextTab>=0 && nextTab < dailyTabWidget->count()) { if ((col!=0) && nextTab>=0 && nextTab < dailyTabWidget->count()) {
dailyTabWidget->setCurrentIndex(nextTab); // 0 = details ; 1=events =2 notes ; 3=bookarks; dailyTabWidget->setCurrentIndex(nextTab); // 0 = details ; 1=events =2 notes ; 3=bookarks;
} }
} }
GPushButton::GPushButton (int row,int column,QDate date,DailySearchTab* parent) : QPushButton(parent), _parent(parent), _row(row), _column(column), _date(date) GPushButton::GPushButton (int row,int column,QDate date,DailySearchTab* parent , ChannelID code, quint32 offset) :
QPushButton(parent), _parent(parent), _row(row), _column(column), _date(date), _code(code) , _offset(offset)
{ {
connect(this, SIGNAL(clicked()), this, SLOT(on_clicked())); connect(this, SIGNAL(clicked()), this, SLOT(on_clicked()));
connect(this, SIGNAL(activated(GPushButton*)), _parent, SLOT(on_activated(GPushButton*))); connect(this, SIGNAL(activated(GPushButton*)), _parent, SLOT(on_activated(GPushButton*)));

View File

@ -49,6 +49,10 @@ public:
void updateEvents(ChannelID id,QString fullname); void updateEvents(ChannelID id,QString fullname);
private: private:
QString red = "#ff8080";
QString green="#80ff80";
QString grey= "#c0c0c0";
QString blue= "#8080ff";
// these values are hard coded. because dynamic translation might not do the proper assignment. // these values are hard coded. because dynamic translation might not do the proper assignment.
// Dynamic code is commented out using c preprocess #if #endif // Dynamic code is commented out using c preprocess #if #endif
@ -64,9 +68,9 @@ private:
const int passDisplayLimit = 30; const int passDisplayLimit = 30;
const int stringDisplayLen = 80; const int stringDisplayLen = 80;
enum ValueMode { invalidValueMode , notUsed , minutesToMs , hoursToMs , hundredths , opWhole , displayWhole , opString , displayString}; enum ValueMode { invalidValueMode , notUsed , minutesToMs , hoursToMs , hundredths , opWhole , displayWhole , opString , displayString, secondsDisplayString};
enum SearchTopic { ST_NONE = 0 , ST_DAYS_SKIPPED = 1 , ST_DISABLED_SESSIONS = 2 , ST_NOTES = 3 , ST_NOTES_STRING , ST_BOOKMARKS , ST_BOOKMARKS_STRING , ST_AHI , ST_SESSION_LENGTH , ST_SESSIONS_QTY , ST_DAILY_USAGE, ST_EVENT }; enum SearchTopic { ST_NONE = 0 , ST_DAYS_SKIPPED = 1 , ST_DISABLED_SESSIONS = 2 , ST_NOTES = 3 , ST_NOTES_STRING , ST_BOOKMARKS , ST_BOOKMARKS_STRING , ST_AHI , ST_SESSION_LENGTH , ST_SESSIONS_QTY , ST_DAILY_USAGE , ST_APNEA_LENGTH , ST_EVENT };
enum OpCode { enum OpCode {
//DO NOT CHANGE NUMERIC OP CODES because THESE VALUES impact compare operations. //DO NOT CHANGE NUMERIC OP CODES because THESE VALUES impact compare operations.
@ -136,6 +140,7 @@ enum OpCode {
QSize setText(QPushButton*,QString); QSize setText(QPushButton*,QString);
QSize setText(QLabel*,QString); QSize setText(QLabel*,QString);
QSize textsize(QFont font ,QString text); QSize textsize(QFont font ,QString text);
void setColor(QPushButton*,QString);
void search(QDate date); void search(QDate date);
void find(QDate&); void find(QDate&);
@ -185,6 +190,7 @@ enum OpCode {
int daysProcessed; int daysProcessed;
int daysFound; int daysFound;
int passFound; int passFound;
int DaysWithFileErrors;
void setoperation(OpCode opCode,ValueMode mode) ; void setoperation(OpCode opCode,ValueMode mode) ;
@ -231,17 +237,23 @@ class GPushButton : public QPushButton
{ {
Q_OBJECT Q_OBJECT
public: public:
GPushButton (int,int,QDate,DailySearchTab* parent); GPushButton (int,int,QDate,DailySearchTab* parent, ChannelID code=0, quint32 offset=0);
virtual ~GPushButton(); virtual ~GPushButton();
int row() { return _row;}; int row() { return _row;};
int column() { return _column;}; int column() { return _column;};
QDate date() { return _date;}; QDate date() { return _date;};
ChannelID code() {return _code;};
quint32 offset() {return _offset;};
void setDate(QDate date) {_date=date;}; void setDate(QDate date) {_date=date;};
void setChannelId(ChannelID code) {_code=code;};
void setOffset(quint32 offset) {_offset=offset;};
private: private:
const DailySearchTab* _parent; const DailySearchTab* _parent;
const int _row; const int _row;
const int _column; const int _column;
QDate _date; QDate _date;
ChannelID _code;
quint32 _offset;
signals: signals:
void activated(GPushButton*); void activated(GPushButton*);
public slots: public slots: