/* user graph settings Implementation * * Copyright (c) 2019-2022 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. */ #define TEST_MACROS_ENABLEDoff #include #include #include #include #include #include #include #include #include // Never include this does not work with other platforms. include #include #include #include #include #include #include "dailySearchTab.h" #include "SleepLib/day.h" #include "SleepLib/profiles.h" #include "daily.h" DailySearchTab::DailySearchTab(Daily* daily , QWidget* searchTabWidget , QTabWidget* dailyTabWidget) : daily(daily) , parent(daily) , searchTabWidget(searchTabWidget) ,dailyTabWidget(dailyTabWidget) { m_icon_selected = new QIcon(":/icons/checkmark.png"); m_icon_notSelected = new QIcon(":/icons/empty_box.png"); m_icon_configure = new QIcon(":/icons/cog.png"); #if 0 // method of find the daily tabWidgets works for english. // the question will it work for all other langauges??? // Right now they are const int in the header file. int maxIndex = dailyTabWidget->count(); for (int index=0 ; indextabText(index); if (title.contains("detail",Qt::CaseInsensitive) ) {TW_DETAILED = index; continue;}; if (title.contains("event",Qt::CaseInsensitive) ) {TW_EVENTS = index; continue;}; if (title.contains("note",Qt::CaseInsensitive) ) {TW_NOTES = index; continue;}; if (title.contains("bookmark",Qt::CaseInsensitive) ) {TW_BOOKMARK = index; continue;}; if (title.contains("search",Qt::CaseInsensitive) ) {TW_SEARCH = index; continue;}; } #endif createUi(); daily->connect(selectString, SIGNAL(textEdited(QString)), this, SLOT(on_textEdited(QString)) ); daily->connect(selectInteger, SIGNAL(valueChanged(int)), this, SLOT(on_intValueChanged(int)) ); daily->connect(selectDouble, SIGNAL(valueChanged(double)), this, SLOT(on_doubleValueChanged(double)) ); daily->connect(selectCommandCombo, SIGNAL(activated(int)), this, SLOT(on_selectCommandCombo_activated(int) )); daily->connect(selectCommandButton, SIGNAL(clicked()), this, SLOT(on_selectCommandButton_clicked()) ); daily->connect(selectOperationCombo,SIGNAL(activated(int)), this, SLOT(on_selectOperationCombo_activated(int) )); daily->connect(selectOperationButton,SIGNAL(clicked()), this, SLOT(on_selectOperationButton_clicked()) ); daily->connect(selectMatch, SIGNAL(clicked()), this, SLOT(on_selectMatch_clicked()) ); daily->connect(startButton, SIGNAL(clicked()), this, SLOT(on_startButton_clicked()) ); daily->connect(clearButton, SIGNAL(clicked()), this, SLOT(on_clearButton_clicked()) ); daily->connect(helpButton , SIGNAL(clicked()), this, SLOT(on_helpButton_clicked()) ); daily->connect(guiDisplayTable, SIGNAL(itemClicked(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->connect(guiDisplayTable, SIGNAL(itemDoubleClicked(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->connect(guiDisplayTable, SIGNAL(itemActivated(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->connect(dailyTabWidget, SIGNAL(currentChanged(int)), this, SLOT(on_dailyTabWidgetCurrentChanged(int) )); } DailySearchTab::~DailySearchTab() { daily->disconnect(selectString, SIGNAL(textEdited(QString)), this, SLOT(on_textEdited(QString)) ); daily->disconnect(dailyTabWidget, SIGNAL(currentChanged(int)), this, SLOT(on_dailyTabWidgetCurrentChanged(int) )); daily->disconnect(selectInteger, SIGNAL(valueChanged(int)), this, SLOT(on_intValueChanged(int)) ); daily->disconnect(selectDouble, SIGNAL(valueChanged(double)), this, SLOT(on_doubleValueChanged(double)) ); daily->disconnect(selectCommandCombo, SIGNAL(activated(int)), this, SLOT(on_selectCommandCombo_activated(int) )); daily->disconnect(selectCommandButton,SIGNAL(clicked()), this, SLOT(on_selectCommandButton_clicked()) ); daily->disconnect(selectOperationCombo,SIGNAL(activated(int)), this, SLOT(on_selectOperationCombo_activated(int) )); daily->disconnect(selectOperationButton,SIGNAL(clicked()), this, SLOT(on_selectOperationButton_clicked()) ); daily->disconnect(selectMatch, SIGNAL(clicked()), this, SLOT(on_selectMatch_clicked()) ); daily->disconnect(helpButton , SIGNAL(clicked()), this, SLOT(on_helpButton_clicked()) ); daily->disconnect(guiDisplayTable, SIGNAL(itemClicked(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->disconnect(guiDisplayTable, SIGNAL(itemDoubleClicked(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->disconnect(guiDisplayTable, SIGNAL(itemActivated(QTableWidgetItem*)), this, SLOT(on_dateItemClicked(QTableWidgetItem*) )); daily->disconnect(startButton, SIGNAL(clicked()), this, SLOT(on_startButton_clicked()) ); daily->connect(clearButton, SIGNAL(clicked()), this, SLOT(on_clearButton_clicked()) ); delete m_icon_selected; delete m_icon_notSelected; delete m_icon_configure ; }; void DailySearchTab::createUi() { QFont baseFont = this->font(); searchTabWidget ->setFont(baseFont); searchTabLayout = new QVBoxLayout(searchTabWidget); criteriaLayout = new QHBoxLayout(); innerCriteriaFrame = new QFrame(this); innerCriteriaLayout = new QHBoxLayout(innerCriteriaFrame); searchLayout = new QHBoxLayout(); summaryLayout = new QHBoxLayout(); searchTabLayout ->setContentsMargins(4, 4, 4, 4); helpButton = new QPushButton(this); helpText = new QTextEdit(this); selectMatch = new QPushButton(this); selectUnits = new QLabel(this); selectCommandCombo = new QComboBox(this); selectCommandButton = new QPushButton(this); selectOperationCombo = new QComboBox(this); selectOperationButton = new QPushButton(this); startButton = new QPushButton(this); clearButton = new QPushButton(this); selectDouble = new QDoubleSpinBox(this); selectInteger = new QSpinBox(this); selectString = new QLineEdit(this); statusProgress = new QLabel(this); summaryProgress = new QLabel(this); summaryFound = new QLabel(this); summaryMinMax = new QLabel(this); guiProgressBar = new QProgressBar(this); guiDisplayTable = new QTableWidget(this); searchTabLayout ->addWidget(helpButton); searchTabLayout ->addWidget(helpText); innerCriteriaLayout ->addWidget(selectCommandCombo); innerCriteriaLayout ->addWidget(selectCommandButton); innerCriteriaLayout ->addWidget(selectOperationCombo); innerCriteriaLayout ->addWidget(selectOperationButton); innerCriteriaLayout ->addWidget(selectInteger); innerCriteriaLayout ->addWidget(selectString); innerCriteriaLayout ->addWidget(selectDouble); innerCriteriaLayout ->addWidget(selectUnits); innerCriteriaLayout ->insertStretch(-1,5); // will center match command criteriaLayout ->addWidget(selectMatch); criteriaLayout ->addWidget(innerCriteriaFrame); criteriaLayout ->insertStretch(-1,5); searchTabLayout ->addLayout(criteriaLayout); searchLayout ->addWidget(clearButton); searchLayout ->addWidget(startButton); searchLayout ->insertStretch(2,5); searchLayout ->addWidget(statusProgress); searchLayout ->insertStretch(-1,5); searchTabLayout ->addLayout(searchLayout); summaryLayout ->addWidget(summaryProgress); summaryLayout ->insertStretch(1,5); summaryLayout ->addWidget(summaryFound); summaryLayout ->insertStretch(3,5); summaryLayout ->addWidget(summaryMinMax); searchTabLayout ->addLayout(summaryLayout); searchTabLayout ->addWidget(guiProgressBar); searchTabLayout ->addWidget(guiDisplayTable); // End of UI creatation // Initialize ui contents QString styleButton=QString("QPushButton { color: black; border: 1px solid black; padding: 5px ; } QPushButton:disabled { color: #606060; border: 1px solid #606060; }" ); searchTabWidget ->setFont(baseFont); helpButton ->setFont(baseFont); helpText ->setFont(baseFont); helpText ->setReadOnly(true); helpText ->setLineWrapMode(QTextEdit::NoWrap); helpMode = true; on_helpButton_clicked(); helpText ->setText(helpStr()); selectMatch->setText(tr("Match:")); selectMatch->setIcon(*m_icon_configure); selectMatch->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectMatch->setStyleSheet( styleButton ); selectOperationButton->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectOperationButton->setText(""); selectOperationButton->setStyleSheet("border:none;"); selectOperationButton->hide(); selectCommandButton->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectCommandButton->setText(tr("Select Match")); selectCommandButton->setStyleSheet("border:none;"); selectCommandCombo->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectCommandCombo->setFont(baseFont); setCommandPopupEnabled(false); selectOperationCombo->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectOperationCombo->setFont(baseFont); setOperationPopupEnabled(false); selectDouble->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectInteger->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectString->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); selectUnits->setText(""); selectUnits->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); clearButton ->setStyleSheet( styleButton ); clearButton ->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); startButton ->setStyleSheet( styleButton ); startButton ->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); helpButton ->setStyleSheet( styleButton ); summaryProgress ->setFont(baseFont); summaryFound ->setFont(baseFont); summaryMinMax ->setFont(baseFont); summaryMinMax ->setSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum); summaryProgress ->setStyleSheet("padding:4px;background-color: #ffffff;" ); summaryFound ->setStyleSheet("padding:4px;background-color: #f0f0f0;" ); summaryMinMax ->setStyleSheet("padding:4px;background-color: #ffffff;" ); searchTopic = ST_NONE; clearButton->setText(tr("Clear")); startButton->setText(tr("Start Search")); startButton->setEnabled(false); guiDisplayTable->setFont(baseFont); if (guiDisplayTable->columnCount() <2) guiDisplayTable->setColumnCount(2); horizontalHeader0 = new QTableWidgetItem(); guiDisplayTable->setHorizontalHeaderItem ( 0, horizontalHeader0); horizontalHeader1 = new QTableWidgetItem(); guiDisplayTable->setHorizontalHeaderItem ( 1, horizontalHeader1); guiDisplayTable->setObjectName(QString::fromUtf8("guiDisplayTable")); guiDisplayTable->setAlternatingRowColors(true); guiDisplayTable->setSelectionMode(QAbstractItemView::SingleSelection); guiDisplayTable->setAlternatingRowColors(true); guiDisplayTable->setSelectionBehavior(QAbstractItemView::SelectRows); guiDisplayTable->setSortingEnabled(false); guiDisplayTable->horizontalHeader()->setStretchLastSection(true); // should make the following based on a real date ei based on locale. guiDisplayTable->setColumnWidth(0, 30/*iconWidthPlus*/ + QFontMetrics(baseFont).size(Qt::TextSingleLine , "WWW MMM 99 2222").width()); horizontalHeader0->setText(tr("DATE\nJumps to Date")); horizontalHeader1->setText(""); on_clearButton_clicked(); } void DailySearchTab::delayedCreateUi() { // meed delay to insure days are populated. if (createUiFinished) return; createUiFinished = true; selectCommandCombo->clear(); selectCommandCombo->addItem(tr("Notes"),ST_NOTES); selectCommandCombo->addItem(tr("Notes containing"),ST_NOTES_STRING); selectCommandCombo->addItem(tr("Bookmarks"),ST_BOOKMARKS); selectCommandCombo->addItem(tr("Bookmarks containing"),ST_BOOKMARKS_STRING); selectCommandCombo->addItem(tr("AHI "),ST_AHI); selectCommandCombo->addItem(tr("Daily Duration"),ST_DAILY_USAGE); selectCommandCombo->addItem(tr("Session Duration" ),ST_SESSION_LENGTH); selectCommandCombo->addItem(tr("Days Skipped"),ST_DAYS_SKIPPED); selectCommandCombo->addItem(tr("Disabled Sessions"),ST_DISABLED_SESSIONS); selectCommandCombo->addItem(tr("Number of Sessions"),ST_SESSIONS_QTY); selectCommandCombo->insertSeparator(selectCommandCombo->count()); // separate from events opCodeMap.clear(); opCodeMap.insert( opCodeStr(OP_LT),OP_LT); opCodeMap.insert( opCodeStr(OP_GT),OP_GT); opCodeMap.insert( opCodeStr(OP_NE),OP_NE); opCodeMap.insert( opCodeStr(OP_LE),OP_LE); opCodeMap.insert( opCodeStr(OP_GE),OP_GE); opCodeMap.insert( opCodeStr(OP_EQ),OP_EQ); opCodeMap.insert( opCodeStr(OP_NE),OP_NE); opCodeMap.insert( opCodeStr(OP_CONTAINS),OP_CONTAINS); opCodeMap.insert( opCodeStr(OP_WILDCARD),OP_WILDCARD); // The order here is the order in the popup box selectOperationCombo->clear(); selectOperationCombo->addItem(opCodeStr(OP_LT)); selectOperationCombo->addItem(opCodeStr(OP_GT)); selectOperationCombo->addItem(opCodeStr(OP_LE)); selectOperationCombo->addItem(opCodeStr(OP_GE)); selectOperationCombo->addItem(opCodeStr(OP_EQ)); selectOperationCombo->addItem(opCodeStr(OP_NE)); // Now add events QDate date = p_profile->LastDay(MT_CPAP); if ( !date.isValid()) return; Day* day = p_profile->GetDay(date); if (!day) return; // the following is copied from daily. qint32 chans = schema::SPAN | schema::FLAG | schema::MINOR_FLAG; if (p_profile->general->showUnknownFlags()) chans |= schema::UNKNOWN; QList available; available.append(day->getSortedMachineChannels(chans)); for (int i=0; i < available.size(); ++i) { ChannelID id = available.at(i); schema::Channel chan = schema::channel[ id ]; // new stuff now QString displayName= chan.fullname(); selectCommandCombo->addItem(displayName,id); } on_clearButton_clicked(); } void DailySearchTab::on_helpButton_clicked() { helpMode = !helpMode; if (helpMode) { helpButton->setText(tr("Click HERE to close Help")); helpText ->setVisible(true); } else { helpText ->setVisible(false); helpButton->setText(tr("Help")); } } QRegExp DailySearchTab::searchPatterToRegex (QString searchPattern) { const static QChar bSlash('\\'); const static QChar asterisk('*'); const static QChar qMark('?'); const static QChar emptyChar('\0'); const QString emptyStr(""); const QString anyStr(".*"); const QString singleStr("."); searchPattern = searchPattern.simplified(); //QString wilDebug = searchPattern; // // wildcard searches uses '*' , '?' and '\' // '*' will match zero or more characters. '?' will match one character. the escape character '\' always matches the next character. // '\\' will match '\'. '\*' will match '*'. '\?' mach for '?'. otherwise the '\' is ignored // The user request will be mapped into a valid QRegularExpression. All RegExp meta characters in the request must be handled. // Most of the meta characters will be escapped. backslash, asterisk, question mark will be treated. // '\*' -> '\*' // '\?' -> '\?' // '*' -> '.*' // really asterisk followed by asterisk or questionmark -> as a single asterisk. // '?' -> '.' // '.' -> '\.' // default for all other meta characters. // '\\' -> '[\\]' // QT documentation states regex reserved characetrs are $ () * + . ? [ ] ^ {} | // seems to be missing / \ - // Regular expression reserved characters / \ [ ] () {} | + ^ . $ ? * - static const QString metaClass = QString( "[ / \\\\ \\[ \\] ( ) { } | + ^ . $ ? * - ]").replace(" ",""); // slash,bSlash,[,],(,),{,},+,^,.,$,?,*,-,| static const QRegExp metaCharRegex(metaClass); #if 0 // Verify search pattern if (!metaCharRegex.isValid()) { DEBUGFW Q(metaCharRegex.errorString()) Q(metaCharRegex) O("============================================"); return QRegExp(); } #endif // regex meta characetrs. all regex meta character must be acounts to us regular expression to make wildcard work. // they will be escaped. except for * and ? which will be treated separately searchPattern = searchPattern.simplified(); // remove witespace at ends. and multiple white space to a single space. // now handle each meta character requested. int pos=0; int len=1; QString replace; QChar metaChar; QChar nextChar; while (pos < (len = searchPattern.length()) ) { pos = searchPattern.indexOf(metaCharRegex,pos); if (pos<0) break; metaChar = searchPattern.at(pos); if (pos+1=len) break; nextChar = searchPattern.at(next); } replace = anyStr; // if asterisk then write dot asterisk } else if (metaChar == qMark ) { replace = singleStr; } else { if ((metaChar == bSlash ) ) { if ( ((nextChar == bSlash ) || (nextChar == asterisk ) || (nextChar == qMark ) ) ) { pos+=2; continue; } replace = emptyStr; //match next character. same as deleteing the backslash } else { // Now have a regex reserved character that needs escaping. // insert an escape '\' before meta characters. replace = QString("\\%1").arg(metaChar); } } searchPattern.replace(pos,replaceCount,replace); pos+=replace.length(); // skip over characters added. } // searchPattern = QString("^.*%1.*$").arg(searchPattern); // add asterisk to end end points. QRegExp convertedRegex =QRegExp(searchPattern,Qt::CaseInsensitive, QRegExp::RegExp); // verify convertedRegex to use if (!convertedRegex.isValid()) { qWarning() << QFileInfo( __FILE__).baseName() <<"[" << __LINE__ << "] " << convertedRegex.errorString() << convertedRegex ; return QRegExp(); } return convertedRegex; } bool DailySearchTab::compare(QString find , QString target) { OpCode opCode = selectOperationOpCode; bool ret=false; if (opCode==OP_CONTAINS) { ret = target.contains(find,Qt::CaseInsensitive); } else if (opCode==OP_WILDCARD) { QRegExp regex = searchPatterToRegex(find); ret = target.contains(regex); } return ret; } bool DailySearchTab::compare(int aa , int bb) { OpCode opCode = selectOperationOpCode; if (opCode>=OP_END_NUMERIC) return false; int mode=0; if (aa bb ) { mode |= OP_GT; } else { mode |= OP_EQ; } return ( (mode & (int)opCode)!=0); }; QString DailySearchTab::valueToString(int value, QString defaultValue) { switch (valueMode) { case hundredths : return QString("%1").arg( (double(value)/100.0),0,'f',2); break; case hoursToMs: case minutesToMs: return formatTime(value); break; case displayWhole: case opWhole: return QString().setNum(value); break; case displayString: case opString: return foundString; break; case invalidValueMode: case notUsed: break; } return defaultValue; } void DailySearchTab::on_selectOperationCombo_activated(int index) { QString text = selectOperationCombo->itemText(index); OpCode opCode = opCodeMap[text]; if (opCode>OP_INVALID && opCode < OP_END_NUMERIC) { selectOperationOpCode = opCode; selectOperationButton->setText(opCodeStr(selectOperationOpCode)); } else if (opCode == OP_CONTAINS || opCode == OP_WILDCARD) { selectOperationOpCode = opCode; selectOperationButton->setText(opCodeStr(selectOperationOpCode)); } else { // null case; } setOperationPopupEnabled(false); criteriaChanged(); }; void DailySearchTab::on_selectCommandCombo_activated(int index) { // here to select new search criteria // must reset all variables and label, button, etc on_clearButton_clicked() ; valueMode = notUsed; selectValue = 0; // workaround for combo box alignmnet and sizing. // copy selections to a pushbutton. hide combobox and show pushButton. Pushbutton activation can show popup. // always hide first before show. allows for best fit selectCommandButton->setText(selectCommandCombo->itemText(index)); setCommandPopupEnabled(false); selectOperationOpCode = OP_INVALID; // get item selected int itemTopic = selectCommandCombo->itemData(index).toInt(); if (itemTopic>=ST_EVENT) { channelId = itemTopic; searchTopic = ST_EVENT; } else { searchTopic = (SearchTopic)itemTopic; } switch (searchTopic) { case ST_NONE : // should never get here. horizontalHeader1->setText(""); nextTab = TW_NONE ; setSelectOperation( OP_INVALID ,notUsed); break; case ST_DAYS_SKIPPED : horizontalHeader1->setText(tr("No Data\nJumps to Date's Details ")); nextTab = TW_DETAILED ; setSelectOperation(OP_NO_PARMS,notUsed); break; case ST_DISABLED_SESSIONS : horizontalHeader1->setText(tr("Number Disabled Session\nJumps to Date's Details ")); nextTab = TW_DETAILED ; selectInteger->setValue(0); setSelectOperation(OP_NO_PARMS,displayWhole); break; case ST_NOTES : horizontalHeader1->setText(tr("Note\nJumps to Date's Notes")); nextTab = TW_NOTES ; setSelectOperation( OP_NO_PARMS ,displayString); break; case ST_BOOKMARKS : horizontalHeader1->setText(tr("Jumps to Date's Bookmark")); nextTab = TW_BOOKMARK ; setSelectOperation( OP_NO_PARMS ,displayString); break; case ST_BOOKMARKS_STRING : horizontalHeader1->setText(tr("Jumps to Date's Bookmark")); nextTab = TW_BOOKMARK ; //setSelectOperation(OP_CONTAINS,opString); setSelectOperation(OP_WILDCARD,opString); selectString->clear(); break; case ST_NOTES_STRING : horizontalHeader1->setText(tr("Note\nJumps to Date's Notes")); nextTab = TW_NOTES ; //setSelectOperation(OP_CONTAINS,opString); setSelectOperation(OP_WILDCARD,opString); selectString->clear(); break; case ST_AHI : horizontalHeader1->setText(tr("AHI\nJumps to Date's Details")); nextTab = TW_DETAILED ; setSelectOperation(OP_GT,hundredths); selectDouble->setValue(5.0); break; case ST_SESSION_LENGTH : horizontalHeader1->setText(tr("Session Duration\nJumps to Date's Details")); nextTab = TW_DETAILED ; setSelectOperation(OP_LT,minutesToMs); selectDouble->setValue(5.0); selectInteger->setValue((int)selectDouble->value()*60000.0); //convert to ms break; case ST_SESSIONS_QTY : horizontalHeader1->setText(tr("Number of Sessions\nJumps to Date's Details")); nextTab = TW_DETAILED ; setSelectOperation(OP_GT,opWhole); selectInteger->setRange(0,999); selectInteger->setValue(2); break; case ST_DAILY_USAGE : horizontalHeader1->setText(tr("Daily Duration\nJumps to Date's Details")); nextTab = TW_DETAILED ; setSelectOperation(OP_LT,hoursToMs); selectDouble->setValue(p_profile->cpap->complianceHours()); selectInteger->setValue((int)selectDouble->value()*3600000.0); //convert to ms break; case ST_EVENT: // Have an Event horizontalHeader1->setText(tr("Number of events\nJumps to Date's Events")); nextTab = TW_EVENTS ; setSelectOperation(OP_GT,opWhole); selectInteger->setValue(0); break; } criteriaChanged(); if (selectOperationOpCode == OP_NO_PARMS ) { // auto start searching startButton->setText(tr("Automatic start")); startButtonMode=true; on_startButton_clicked(); return; } return; } void DailySearchTab::updateValues(qint32 value) { foundValue = value; if (!minMaxValid ) { minMaxValid = true; minInteger = value; maxInteger = value; } else if ( value < minInteger ) { minInteger = value; } else if ( value > maxInteger ) { maxInteger = value; } } void DailySearchTab::find(QDate& date) { bool found=false; Qt::Alignment alignment=Qt::AlignCenter; Day* day = p_profile->GetDay(date); if ( (!day) && (searchTopic != ST_DAYS_SKIPPED)) { daysSkipped++; return;}; switch (searchTopic) { case ST_DAYS_SKIPPED : found=!day; break; case ST_DISABLED_SESSIONS : { qint32 numDisabled=0; QList sessions = day->getSessions(MT_CPAP,true); for (auto & sess : sessions) { if (!sess->enabled()) { numDisabled ++; found=true; } } updateValues(numDisabled); } break; case ST_NOTES : { Session* journal=daily->GetJournalSession(date); if (journal && journal->settings.contains(Journal_Notes)) { QString jcontents = convertRichText2Plain(journal->settings[Journal_Notes].toString()); foundString = jcontents.simplified().left(stringDisplayLen).simplified(); found=true; alignment=Qt::AlignLeft; } } break; case ST_BOOKMARKS : { Session* journal=daily->GetJournalSession(date); if (journal && journal->settings.contains(Bookmark_Notes)) { found=true; QStringList notes = journal->settings[Bookmark_Notes].toStringList(); for ( const auto & note : notes) { foundString = note.simplified().left(stringDisplayLen).simplified(); alignment=Qt::AlignLeft; break; } } } break; case ST_BOOKMARKS_STRING : { Session* journal=daily->GetJournalSession(date); if (journal && journal->settings.contains(Bookmark_Notes)) { QStringList notes = journal->settings[Bookmark_Notes].toStringList(); QString findStr = selectString->text(); for ( const auto & note : notes) { if (compare(findStr , note)) { found=true; foundString = note.simplified().left(stringDisplayLen).simplified(); alignment=Qt::AlignLeft; break; } } } } break; case ST_NOTES_STRING : { Session* journal=daily->GetJournalSession(date); if (journal && journal->settings.contains(Journal_Notes)) { QString jcontents = convertRichText2Plain(journal->settings[Journal_Notes].toString()); QString findStr = selectString->text(); if (jcontents.contains(findStr,Qt::CaseInsensitive) ) { found=true; foundString = jcontents.simplified().left(stringDisplayLen).simplified(); alignment=Qt::AlignLeft; } } } break; case ST_AHI : { EventDataType dahi =calculateAhi(day); dahi += 0.005; dahi *= 100.0; int ahi = (int)dahi; updateValues(ahi); found = compare (ahi , selectValue); } break; case ST_SESSION_LENGTH : { bool valid=false; qint64 value=0; QList sessions = day->getSessions(MT_CPAP); for (auto & sess : sessions) { qint64 ms = sess->length(); updateValues(ms); if (compare (ms , selectValue) ) { found =true; } if (!valid) { valid=true; value=ms; } else if (compare (ms , value) ) { value=ms; } } if (valid) updateValues(value); } break; case ST_SESSIONS_QTY : { QList sessions = day->getSessions(MT_CPAP); qint32 size = sessions.size(); updateValues(size); found=compare (size , selectValue); } break; case ST_DAILY_USAGE : { QList sessions = day->getSessions(MT_CPAP); qint64 sum = 0 ; for (auto & sess : sessions) { sum += sess->length(); } updateValues(sum); found=compare (sum , selectValue); } break; case ST_EVENT : { qint32 count = day->count(channelId); updateValues(count); found=compare (count , selectValue); } break; case ST_NONE : break; } if (found) { addItem(date , valueToString(foundValue,"------"),alignment ); passFound++; daysFound++; } return ; }; void DailySearchTab::search(QDate date) { if (!date.isValid()) { qWarning() << "DailySearchTab::find invalid date." << date; return; } guiProgressBar->show(); statusProgress->show(); guiDisplayTable->clearContents(); for (int index=0; indexrowCount();index++) { guiDisplayTable->setRowHidden(index,true); } foundString.clear(); passFound=0; while (date >= earliestDate) { nextDate = date; if (passFound >= passDisplayLimit) break; find(date); guiProgressBar->setValue(++daysProcessed); date=date.addDays(-1); } endOfPass(); return ; }; void DailySearchTab::addItem(QDate date, QString value,Qt::Alignment alignment) { int row = passFound; QTableWidgetItem *item = new QTableWidgetItem(*m_icon_notSelected,date.toString()); item->setData(dateRole,date); item->setData(valueRole,value); item->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); QTableWidgetItem *item2 = new QTableWidgetItem(*m_icon_notSelected,value); item2->setTextAlignment(alignment|Qt::AlignVCenter); item2->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); if (guiDisplayTable->rowCount()<(row+1)) { guiDisplayTable->insertRow(row); } guiDisplayTable->setItem(row,0,item); guiDisplayTable->setItem(row,1,item2); guiDisplayTable->setRowHidden(row,false); } void DailySearchTab::endOfPass() { startButtonMode=false; // display Continue; QString display; if ((passFound >= passDisplayLimit) && (daysProcessedsetText(centerLine(tr("More to Search"))); statusProgress->show(); startButton->setEnabled(true); startButton->setText(tr("Continue Search")); guiDisplayTable->horizontalHeader()->show(); } else if (daysFound>0) { statusProgress->setText(centerLine(tr("End of Search"))); statusProgress->show(); startButton->setEnabled(false); guiDisplayTable->horizontalHeader()->show(); } else { statusProgress->setText(centerLine(tr("No Matches"))); statusProgress->show(); startButton->setEnabled(false); guiDisplayTable->horizontalHeader()->hide(); } displayStatistics(); } void DailySearchTab::on_dateItemClicked(QTableWidgetItem *item) { int row = item->row(); int col = item->column(); guiDisplayTable->setCurrentItem(item,QItemSelectionModel::Clear); item->setIcon (*m_icon_selected); item=guiDisplayTable->item(row,col); if (col!=0) { item = guiDisplayTable->item(item->row(),0); } QDate date = item->data(dateRole).toDate(); daily->LoadDate( date ); if ((col!=0) && nextTab>=0 && nextTab < dailyTabWidget->count()) { dailyTabWidget->setCurrentIndex(nextTab); // 0 = details ; 1=events =2 notes ; 3=bookarks; } } void DailySearchTab::setCommandPopupEnabled(bool on) { if (on) { selectCommandButton->show(); selectCommandCombo->setEnabled(true); selectCommandCombo->showPopup(); } else { selectCommandCombo->hidePopup(); selectCommandCombo->setEnabled(false); selectCommandCombo->hide(); selectCommandButton->show(); } } void DailySearchTab::on_selectOperationButton_clicked() { if (selectOperationOpCode == OP_CONTAINS ) { selectOperationOpCode = OP_WILDCARD; } else if (selectOperationOpCode == OP_WILDCARD) { selectOperationOpCode = OP_CONTAINS ; } else { setOperationPopupEnabled(true); return; } selectOperationButton->setText(opCodeStr(selectOperationOpCode)); criteriaChanged(); }; void DailySearchTab::on_selectMatch_clicked() { setCommandPopupEnabled(true); } void DailySearchTab::on_selectCommandButton_clicked() { setCommandPopupEnabled(true); } void DailySearchTab::setOperationPopupEnabled(bool on) { //if (selectOperationOpCode= OP_END_NUMERIC) return; if (on) { selectOperationCombo->setEnabled(true); selectOperationCombo->showPopup(); selectOperationButton->show(); } else { selectOperationCombo->hidePopup(); selectOperationCombo->setEnabled(false); selectOperationCombo->hide(); selectOperationButton->show(); } } void DailySearchTab::setSelectOperation(OpCode opCode,ValueMode mode) { valueMode = mode; selectOperationOpCode = opCode; selectOperationButton->setText(opCodeStr(selectOperationOpCode)); setOperationPopupEnabled(false); if (opCode > OP_INVALID && opCode setDecimals(2); selectDouble->setRange(0,999); selectDouble->setDecimals(2); selectInteger->setRange(0,999); } switch (valueMode) { case hundredths : selectDouble->show(); break; case hoursToMs: selectUnits->setText(" Hours"); selectUnits->show(); selectDouble->show(); break; case minutesToMs: selectUnits->setText(" Minutes"); selectUnits->show(); selectDouble->setRange(0,9999); selectDouble->show(); break; case opWhole: selectInteger->show(); break; case displayWhole: selectInteger->hide(); break; case opString: selectOperationButton->show(); selectString ->show(); break; case displayString: selectString ->hide(); break; case invalidValueMode: case notUsed: break; } } void DailySearchTab::hideResults() { guiProgressBar->hide(); // clear display table && hide guiDisplayTable->horizontalHeader()->hide(); for (int index=0; indexrowCount();index++) { guiDisplayTable->setRowHidden(index,true); } guiDisplayTable->horizontalHeader()->hide(); // reset summary line summaryProgress->hide(); summaryFound->hide(); summaryMinMax->hide(); statusProgress->hide(); } void DailySearchTab::on_clearButton_clicked() { // make these button text back to start. startButton->setText(tr("Start Search")); startButtonMode=true; startButton->setEnabled( false); // hide widgets //Reset Select area selectCommandCombo->hide(); setCommandPopupEnabled(false); selectCommandButton->setText(tr("Select Match")); selectCommandButton->show(); selectOperationCombo->hide(); setOperationPopupEnabled(false); selectOperationButton->hide(); selectDouble->hide(); selectInteger->hide(); selectString->hide(); selectUnits->hide(); hideResults(); } void DailySearchTab::on_startButton_clicked() { if (startButtonMode) { search (latestDate ); startButtonMode=false; } else { search (nextDate ); } } void DailySearchTab::on_intValueChanged(int ) { selectInteger->findChild()->deselect(); criteriaChanged(); } void DailySearchTab::on_doubleValueChanged(double ) { selectDouble->findChild()->deselect(); criteriaChanged(); } void DailySearchTab::on_textEdited(QString ) { criteriaChanged(); } void DailySearchTab::on_dailyTabWidgetCurrentChanged(int ) { // Any time a tab (daily, events , notes, bookmarks, seatch) is changed // so finish updating the ui display. delayedCreateUi(); } void DailySearchTab::displayStatistics() { QString extra; summaryProgress->show(); // display days searched QString skip= daysSkipped==0?"":QString(tr(" Skip:%1")).arg(daysSkipped); summaryProgress->setText(centerLine(QString(tr("%1/%2%3 days.")).arg(daysProcessed).arg(daysTotal).arg(skip) )); // display days found summaryFound->setText(centerLine(QString(tr("Found %1.")).arg(daysFound) )); // display associated value extra =""; if (minMaxValid) { extra = QString("%1/%2").arg(valueToString(minInteger)).arg(valueToString(maxInteger)); } if (extra.size()>0) { summaryMinMax->setText(extra); summaryMinMax->show(); } else { summaryMinMax->hide(); } summaryProgress->show(); summaryFound->show(); } void DailySearchTab::criteriaChanged() { // setup before start button if (valueMode != notUsed ) { selectOperationButton->setText(opCodeStr(selectOperationOpCode)); selectOperationButton->show(); } switch (valueMode) { case hundredths : selectValue = (int)(selectDouble->value()*100.0); //convert to hundreths of AHI. break; case minutesToMs: selectValue = (int)(selectDouble->value()*60000.0); //convert to ms break; case hoursToMs: selectValue = (int)(selectDouble->value()*3600000.0); //convert to ms break; case displayWhole: case opWhole: selectValue = selectInteger->value();; break; case opString: case displayString: case invalidValueMode: case notUsed: break; } selectCommandCombo->hide(); selectCommandButton->show(); startButton->setText(tr("Start Search")); startButtonMode=true; startButton->setEnabled( true); statusProgress->setText(centerLine(" ----- ")); statusProgress->clear(); hideResults(); minMaxValid = false; minInteger = 0; maxInteger = 0; minDouble = 0.0; maxDouble = 0.0; earliestDate = p_profile->FirstDay(MT_CPAP); latestDate = p_profile->LastDay(MT_CPAP); daysTotal= 1+earliestDate.daysTo(latestDate); daysFound=0; daysSkipped=0; daysProcessed=0; startButtonMode=true; //initialize progress bar. guiProgressBar->setMinimum(0); guiProgressBar->setMaximum(daysTotal); guiProgressBar->setTextVisible(true); //guiProgressBar->setTextVisible(false); guiProgressBar->setMaximumHeight(15); guiProgressBar->reset(); } // inputs character string. // outputs cwa centered html string. // converts \n to
QString DailySearchTab::centerLine(QString line) { return QString( "
%1
").arg(line).replace("\n","
"); } QString DailySearchTab::helpStr() { QStringList str; str.append(tr("Finds days that match specified criteria.")); str.append("\n"); str.append(tr(" Searches from last day to first day.")); str.append("\n"); str.append("\n"); str.append(tr("First click on Match Button then select topic.")); str.append("\n"); str.append(tr(" Then click on the operation to modify it.")); str.append("\n"); str.append(tr(" or update the value")); str.append("\n"); str.append(tr("Topics without operations will automatically start.")); str.append("\n"); str.append("\n"); str.append(tr("Compare Operations: numberic or character. ")); str.append("\n"); str.append(tr(" Numberic Operations: ")); str.append(" > , >= , < , <= , == , != "); str.append("\n"); str.append(tr(" Character Operations: ")); str.append(" == , *? "); str.append("\n"); str.append("\n"); str.append(tr("Summary Line")); str.append("\n"); str.append(tr(" Left:Summary - Number of Day searched")); str.append("\n"); str.append(tr(" Center:Number of Items Found")); str.append("\n"); str.append(tr(" Right:Minimum/Maximum for item searched")); str.append("\n"); str.append(tr("Result Table")); str.append("\n"); str.append(tr(" Column One: Date of match. Click selects date.")); str.append("\n"); str.append(tr(" Column two: Information. Click selects date.")); str.append("\n"); str.append(tr(" Then Jumps the appropiate tab.")); str.append("\n"); str.append("\n"); str.append(tr("Wildcard Pattern Matching:")); str.append(" *? "); str.append("\n"); str.append(tr(" Wildcards use 3 characters:")); str.append("\n"); str.append(tr(" Asterisk")); str.append(" * "); str.append(" "); str.append(tr(" Question Mark")); str.append(" ? "); str.append(" "); str.append(tr(" Backslash.")); str.append(" \\ "); str.append("\n"); str.append(tr(" Asterisk matches any number of characters.")); str.append("\n"); str.append(tr(" Question Mark matches a single character.")); str.append("\n"); str.append(tr(" Backslash matches next character.")); str.append("\n"); QString result =str.join(""); return result; } QString DailySearchTab::formatTime (qint32 ms) { qint32 hours = ms / 3600000; ms = ms % 3600000; qint32 minutes = ms / 60000; ms = ms % 60000; qint32 seconds = ms /1000; //return QString(tr("%1h %2m %3s")).arg(hours).arg(minutes).arg(seconds); return QString("%1:%2:%3").arg(hours).arg(minutes,2,10,QLatin1Char('0')).arg(seconds,2,10,QLatin1Char('0')); } QString DailySearchTab::convertRichText2Plain (QString rich) { richText.setHtml(rich); QString line=richText.toPlainText().simplified(); return line.replace(QRegExp("[\\s\\r\\n]+")," ").simplified(); } QString DailySearchTab::opCodeStr(OpCode opCode) { switch (opCode) { case OP_GT : return "> "; case OP_GE : return ">="; case OP_LT : return "< "; case OP_LE : return "<="; case OP_EQ : return "=="; case OP_NE : return "!="; case OP_CONTAINS : return QChar(0x2208); // or use 0x220B case OP_WILDCARD : return "*?"; case OP_INVALID: case OP_END_NUMERIC: case OP_NO_PARMS: break; } return QString(); }; EventDataType DailySearchTab::calculateAhi(Day* day) { if (!day) return 0.0; // copied from daily.cpp double hours=day->hours(MT_CPAP); if (hours<=0) return 0; EventDataType ahi=day->count(AllAhiChannels); if (p_profile->general->calculateRDI()) ahi+=day->count(CPAP_RERA); ahi/=hours; return ahi; }