Renamed Purge menu to Rebuild, automatically reimport from backups if present, fixed vertical scroll bug

This commit is contained in:
Mark Watkins 2014-07-26 22:46:11 +10:00
parent ff0e87843e
commit 936a21cefd
9 changed files with 94 additions and 51 deletions

View File

@ -1405,7 +1405,7 @@ void gGraphView::mousePressEvent(QMouseEvent *event)
m_sizer_point.setX(x); m_sizer_point.setX(x);
m_sizer_point.setY(py); // point at top of graph.. m_sizer_point.setY(py); // point at top of graph..
this->setCursor(Qt::ClosedHandCursor); this->setCursor(Qt::ClosedHandCursor);
} } else
{ {
if (m_metaselect) { if (m_metaselect) {
@ -1508,6 +1508,11 @@ void gGraphView::mouseReleaseEvent(QMouseEvent *event)
float h, py = 0, pinned_height = 0; float h, py = 0, pinned_height = 0;
bool done = false; bool done = false;
// Copy to a local variable to make sure this gets cleared
bool button_down = m_button_down;
m_button_down = false;
// Handle pinned graphs first // Handle pinned graphs first
for (int i = 0; i < m_graphs.size(); i++) { for (int i = 0; i < m_graphs.size(); i++) {
gGraph *g = m_graphs[i]; gGraph *g = m_graphs[i];
@ -1599,8 +1604,8 @@ void gGraphView::mouseReleaseEvent(QMouseEvent *event)
} }
// The graph that got the button press gets the release event // The graph that got the button press gets the release event
if (m_button_down) { if (button_down) {
m_button_down = false; // m_button_down = false;
m_metaselect = event->modifiers() & Qt::AltModifier; m_metaselect = event->modifiers() & Qt::AltModifier;
saveHistory(); saveHistory();
@ -1739,7 +1744,9 @@ void gGraphView::wheelEvent(QWheelEvent *event)
{ {
// Hmm.. I could optionalize this to change mousewheel behaviour without affecting the scrollbar now.. // Hmm.. I could optionalize this to change mousewheel behaviour without affecting the scrollbar now..
if (m_button_down) return; if (m_button_down)
return;
if ((event->modifiers() & Qt::ControlModifier)) { if ((event->modifiers() & Qt::ControlModifier)) {
int x = event->x(); int x = event->x();
int y = event->y(); int y = event->y();
@ -1763,9 +1770,7 @@ void gGraphView::wheelEvent(QWheelEvent *event)
// What to do when ctrl+wheel is used on the graph title ?? // What to do when ctrl+wheel is used on the graph title ??
} else { } else {
// send event to graph.. // send event to graph..
if (!m_button_down) { g->wheelEvent(event);
g->wheelEvent(event);
}
} }
} else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) { } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) {
// What to do when the wheel is used on the resize handle? // What to do when the wheel is used on the resize handle?

View File

@ -503,7 +503,8 @@ bool EDFParser::Parse()
long recs = sig.nr * num_data_records; long recs = sig.nr * num_data_records;
if (num_data_records < 0) { if (num_data_records < 0) {
return false; sig.data = nullptr;
continue;
} }
sig.data = new qint16 [recs]; sig.data = new qint16 [recs];
@ -622,7 +623,6 @@ void ResmedImport::run()
if (sess->summaryOnly()) { if (sess->summaryOnly()) {
sess->Destroy(); sess->Destroy();
//mach->sessionlist.remove(sess->session());
delete sess; delete sess;
} }
} }
@ -646,11 +646,14 @@ void ResmedImport::run()
} }
if (sess->first() == 0) { if (sess->first() == 0) {
// loader->saveMutex.lock();
//if (mach->sessionlist.contains(sess->session())) { //if (mach->sessionlist.contains(sess->session())) {
sess->Destroy(); // sess->Destroy();
//mach->sessionlist.remove(sess->session()); //mach->sessionlist.remove(sess->session());
//} //}
delete sess; delete sess;
//loader->saveMutex.unlock();
return; return;
} }
@ -1681,6 +1684,8 @@ bool ResmedLoader::LoadBRP(Session *sess, const QString & path)
EDFSignal &es = edf.edfsignals[s]; EDFSignal &es = edf.edfsignals[s];
long recs = es.nr * edf.GetNumDataRecords(); long recs = es.nr * edf.GetNumDataRecords();
if (recs < 0)
continue;
ChannelID code; ChannelID code;
if (matchSignal(CPAP_FlowRate, es.label)) { if (matchSignal(CPAP_FlowRate, es.label)) {

View File

@ -201,6 +201,10 @@ bool Machine::AddSession(Session *s)
for (QList<Session *>::iterator i = nextday.value()->begin(); i != nextday.value()->end(); i++) { for (QList<Session *>::iterator i = nextday.value()->begin(); i != nextday.value()->end(); i++) {
// i may need to do something here // i may need to do something here
unlinkSession(*i); unlinkSession(*i);
// Add it back
sessionlist[(*i)->session()] = *i;
dd->AddSession(*i); dd->AddSession(*i);
} }

View File

@ -79,6 +79,7 @@ Profile::~Profile()
delete session; delete session;
delete general; delete general;
for (QHash<MachineID, Machine *>::iterator it = machlist.begin(); it != machlist.end(); it++) { for (QHash<MachineID, Machine *>::iterator it = machlist.begin(); it != machlist.end(); it++) {
delete it.value(); delete it.value();
} }

View File

@ -785,6 +785,10 @@ class UserSettings : public ProfileSettings
void setStatReportMode(int i) { setPref(STR_US_StatReportMode, i); } void setStatReportMode(int i) { setPref(STR_US_StatReportMode, i); }
}; };
//! \brief Returns a count of all files & directories in a supplied folder
int dirCount(QString path);
namespace Profiles { namespace Profiles {
extern QMap<QString, Profile *> profiles; extern QMap<QString, Profile *> profiles;

View File

@ -117,8 +117,6 @@ bool Session::Destroy()
{ {
QString path = p_profile->Get(s_machine->properties[STR_PROP_Path]); QString path = p_profile->Get(s_machine->properties[STR_PROP_Path]);
s_machine->unlinkSession(this);
QDir dir(path); QDir dir(path);
QString base; QString base;
base.sprintf("%08lx", s_session); base.sprintf("%08lx", s_session);
@ -127,7 +125,7 @@ bool Session::Destroy()
dir.remove(base + ".000"); dir.remove(base + ".000");
dir.remove(base + ".001"); dir.remove(base + ".001");
return !dir.exists(base + ".000") && !dir.exists(base + ".001"); return s_machine->unlinkSession(this); //!dir.exists(base + ".000") && !dir.exists(base + ".001");
} }
bool Session::Store(QString path) bool Session::Store(QString path)

View File

@ -430,11 +430,11 @@ void MyStatsPage::javaScriptAlert(QWebFrame *frame, const QString &msg)
void MainWindow::PopulatePurgeMenu() void MainWindow::PopulatePurgeMenu()
{ {
QList<QAction *> actions = ui->menu_Purge_CPAP_Data->actions(); QList<QAction *> actions = ui->menu_Rebuild_CPAP_Data->actions();
ui->menu_Purge_CPAP_Data->disconnect(ui->menu_Purge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction *))); ui->menu_Rebuild_CPAP_Data->disconnect(ui->menu_Rebuild_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction *)));
ui->menu_Purge_CPAP_Data->clear(); ui->menu_Rebuild_CPAP_Data->clear();
QList<Machine *> machines = p_profile->GetMachines(MT_CPAP); QList<Machine *> machines = p_profile->GetMachines(MT_CPAP);
for (int i=0; i < machines.size(); ++i) { for (int i=0; i < machines.size(); ++i) {
@ -443,12 +443,13 @@ void MainWindow::PopulatePurgeMenu()
mach->properties[STR_PROP_Model]+" "+ mach->properties[STR_PROP_Model]+" "+
mach->properties[STR_PROP_Serial]; mach->properties[STR_PROP_Serial];
QAction * action = new QAction(name.replace("&","&&"), ui->menu_Purge_CPAP_Data); QAction * action = new QAction(name.replace("&","&&"), ui->menu_Rebuild_CPAP_Data);
action->setData(mach->GetClass()+":"+mach->properties[STR_PROP_Serial]); action->setData(mach->GetClass()+":"+mach->properties[STR_PROP_Serial]);
ui->menu_Purge_CPAP_Data->addAction(action); ui->menu_Rebuild_CPAP_Data->addAction(action);
} }
ui->menu_Purge_CPAP_Data->connect(ui->menu_Purge_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction*))); ui->menu_Rebuild_CPAP_Data->connect(ui->menu_Rebuild_CPAP_Data, SIGNAL(triggered(QAction*)), this, SLOT(on_actionPurgeMachine(QAction*)));
} }
QString GenerateWelcomeHTML(); QString GenerateWelcomeHTML();
void MainWindow::Startup() void MainWindow::Startup()
@ -525,7 +526,6 @@ void MainWindow::finishCPAPImport()
void MainWindow::importCPAPBackups() void MainWindow::importCPAPBackups()
{ {
// Get BackupPaths for all CPAP machines // Get BackupPaths for all CPAP machines
QList<Machine *> machlist = p_profile->GetMachines(MT_CPAP); QList<Machine *> machlist = p_profile->GetMachines(MT_CPAP);
QStringList paths; QStringList paths;
@ -1822,15 +1822,37 @@ void MainWindow::on_actionPurgeMachine(QAction *action)
void MainWindow::purgeMachine(Machine * mach) void MainWindow::purgeMachine(Machine * mach)
{ {
// detect backups
bool backups = false;
if (mach->properties.contains(STR_PROP_BackupPath)) {
QString bpath = p_profile->Get(mach->properties[STR_PROP_BackupPath]);
int cnt = dirCount(bpath);
if (cnt > 0) backups = true;
}
if (QMessageBox::question(this, if (backups) {
if (QMessageBox::question(this,
STR_MessageBox_Question, STR_MessageBox_Question,
tr("Are you sure you want to purge all CPAP data for the following machine:")+ "\n\n" + tr("Are you sure you want to rebuild all CPAP data for the following machine:")+ "\n\n" +
mach->properties[STR_PROP_Brand] + " " + mach->properties[STR_PROP_Model] + " " + mach->properties[STR_PROP_Brand] + " " + mach->properties[STR_PROP_Model] + " " +
mach->properties[STR_PROP_ModelNumber] + " (" + mach->properties[STR_PROP_Serial] + ")", mach->properties[STR_PROP_ModelNumber] + " (" + mach->properties[STR_PROP_Serial] + ")" + "\n\n"+
tr("Please note, that this could result in loss of graph data if SleepyHead's internal backups have been disabled or interfered with in any way."),
QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes | QMessageBox::No,
QMessageBox::No) == QMessageBox::No) { QMessageBox::No) == QMessageBox::No) {
return; return;
}
} else {
if (QMessageBox::question(this,
STR_MessageBox_Warning,
"<p><b>"+STR_MessageBox_Warning+": </b>"+tr("For some reason, SleepyHead does not have internal backups for the following machine:")+ "</p>" +
"<p>"+mach->properties[STR_PROP_Brand] + " " + mach->properties[STR_PROP_Model] + " " +
mach->properties[STR_PROP_ModelNumber] + " (" + mach->properties[STR_PROP_Serial] + ")" + "</p>"+
"<p>"+tr("Provided you have made <i>your <b>own</b> backups for ALL of your CPAP data</i>, you can still complete this operation, but you will have to restore from your backups manually.")+"</p>"
"<p><b>"+tr("Are you really sure you want to do this?")+"</b></p>",
QMessageBox::Yes | QMessageBox::No,
QMessageBox::No) == QMessageBox::No) {
return;
}
} }
daily->Unload(daily->getDate()); daily->Unload(daily->getDate());
@ -1840,7 +1862,7 @@ void MainWindow::purgeMachine(Machine * mach)
mach->day.clear(); mach->day.clear();
} else { } else {
QMessageBox::warning(this, STR_MessageBox_Error, QMessageBox::warning(this, STR_MessageBox_Error,
tr("Not all session data could be removed, you have to delete the following folder manually.") tr("A file permission error or simillar screwed up the purge process, you will have to delete the following folder manually:")
+"\n\n"+ +"\n\n"+
QDir::toNativeSeparators(p_profile->Get(mach->properties[STR_PROP_Path])), QMessageBox::Ok, QMessageBox::Ok); QDir::toNativeSeparators(p_profile->Get(mach->properties[STR_PROP_Path])), QMessageBox::Ok, QMessageBox::Ok);
if (overview) overview->ReloadGraphs(); if (overview) overview->ReloadGraphs();
@ -1862,23 +1884,23 @@ void MainWindow::purgeMachine(Machine * mach)
GenerateStatistics(); GenerateStatistics();
QApplication::processEvents(); QApplication::processEvents();
if (QMessageBox::question(this,
STR_MessageBox_Question, if (backups) {
tr("Machine data has been successfully purged.") + "\n\n" +
tr("Would you like to reimport from the backup folder?") + "\n\n" +
QDir::toNativeSeparators(p_profile->Get(mach->properties[STR_PROP_BackupPath])),
QMessageBox::Yes | QMessageBox::No,
QMessageBox::Yes) == QMessageBox::No) {
p_profile->machlist.erase(p_profile->machlist.find(mach->id()));
delete mach;
} else {
importCPAP(p_profile->Get(mach->properties[STR_PROP_BackupPath]),tr("Please wait, importing...")); importCPAP(p_profile->Get(mach->properties[STR_PROP_BackupPath]),tr("Please wait, importing..."));
if (overview) overview->ReloadGraphs(); } else {
if (daily) { if (QMessageBox::information(this, STR_MessageBox_Warning,
daily->clearLastDay(); // otherwise Daily will crash tr("Because there are no internal backups to rebuild from, you will have to restore from your own.")+"\n\n"+
daily->ReloadGraphs(); tr("Would you like to import from your own backups now? (you will have no data visible for this machine until you do)"),
QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes) == QMessageBox::Yes) {
} else {
on_action_Import_Data_triggered();
} }
} }
if (overview) overview->ReloadGraphs();
if (daily) {
daily->clearLastDay(); // otherwise Daily will crash
daily->ReloadGraphs();
}
GenerateStatistics(); GenerateStatistics();
p_profile->Save(); p_profile->Save();
} }

View File

@ -3185,23 +3185,21 @@ border-radius: 10px;
<property name="title"> <property name="title">
<string>&amp;Advanced</string> <string>&amp;Advanced</string>
</property> </property>
<widget class="QMenu" name="menu_Purge_CPAP_Data">
<property name="title">
<string>&amp;Purge CPAP Data</string>
</property>
<addaction name="separator"/>
</widget>
<widget class="QMenu" name="menuPurge_Oximetery_Data"> <widget class="QMenu" name="menuPurge_Oximetery_Data">
<property name="title"> <property name="title">
<string>Purge Oximetery Data</string> <string>Purge Oximetery Data</string>
</property> </property>
<addaction name="actionPurgeCurrentDaysOximetry"/> <addaction name="actionPurgeCurrentDaysOximetry"/>
</widget> </widget>
<addaction name="menu_Purge_CPAP_Data"/>
<addaction name="actionPurge_Current_Day"/> <addaction name="actionPurge_Current_Day"/>
<addaction name="separator"/> <addaction name="separator"/>
<addaction name="menuPurge_Oximetery_Data"/> <addaction name="menuPurge_Oximetery_Data"/>
</widget> </widget>
<widget class="QMenu" name="menu_Rebuild_CPAP_Data">
<property name="title">
<string>Rebuild CPAP Data</string>
</property>
</widget>
<addaction name="actionView_Oximetry"/> <addaction name="actionView_Oximetry"/>
<addaction name="separator"/> <addaction name="separator"/>
<addaction name="actionImport_ZEO_Data"/> <addaction name="actionImport_ZEO_Data"/>
@ -3209,8 +3207,9 @@ border-radius: 10px;
<addaction name="actionImport_RemStar_MSeries_Data"/> <addaction name="actionImport_RemStar_MSeries_Data"/>
<addaction name="separator"/> <addaction name="separator"/>
<addaction name="action_Rebuild_Oximetry_Index"/> <addaction name="action_Rebuild_Oximetry_Index"/>
<addaction name="menu_Advanced"/> <addaction name="menu_Rebuild_CPAP_Data"/>
<addaction name="separator"/> <addaction name="separator"/>
<addaction name="menu_Advanced"/>
</widget> </widget>
<addaction name="menu_File"/> <addaction name="menu_File"/>
<addaction name="menu_View"/> <addaction name="menu_View"/>
@ -3366,7 +3365,7 @@ border-radius: 10px;
</action> </action>
<action name="action_Rebuild_Oximetry_Index"> <action name="action_Rebuild_Oximetry_Index">
<property name="text"> <property name="text">
<string>&amp;Rebuild Oximetry Indices</string> <string>&amp;Automatic Oximetry Cleanup</string>
</property> </property>
</action> </action>
<action name="actionChange_User"> <action name="actionChange_User">
@ -3463,6 +3462,11 @@ border-radius: 10px;
<string>Ctrl+S</string> <string>Ctrl+S</string>
</property> </property>
</action> </action>
<action name="actionFoo">
<property name="text">
<string>Foo</string>
</property>
</action>
</widget> </widget>
<customwidgets> <customwidgets>
<customwidget> <customwidget>

View File

@ -99,7 +99,7 @@ QString GenerateWelcomeHTML()
html+=QString("<tr><td colspan=%1 align=center>").arg(cols)+ html+=QString("<tr><td colspan=%1 align=center>").arg(cols)+
"<font size=+1>"+((havecpapdata || haveoximeterdata) ? QObject::tr("Would you like to import more data?") : QObject::tr("Please Import Some Data")) +"</font></td>" "<font size=+1>"+((havecpapdata || haveoximeterdata) ? QObject::tr("What would you like to do?") : QObject::tr("Please Import Some Data")) +"</font></td>"
"</tr>" "</tr>"
"<tr>" "<tr>"
"<td align=center>" "<td align=center>"
@ -134,9 +134,9 @@ QString GenerateWelcomeHTML()
else if (day->machine->GetClass() == STR_MACH_PRS1) cpapimage = "qrc:/icons/prs1.png"; else if (day->machine->GetClass() == STR_MACH_PRS1) cpapimage = "qrc:/icons/prs1.png";
else if (day->machine->GetClass() == STR_MACH_Intellipap) cpapimage = "qrc:/icons/intellipap.png"; else if (day->machine->GetClass() == STR_MACH_Intellipap) cpapimage = "qrc:/icons/intellipap.png";
} }
html += "<table cellpadding=4><tr><td><img src='"+cpapimage+"' width=128px><br/>"; html += "<table cellpadding=4><tr><td><img src='"+cpapimage+"' width=160px><br/>";
html+="</td><td><table cellpadding=4 class=curved2><tr>"+ html+="</td><td align=center><table cellpadding=4 class=curved2 title=\""+QObject::tr("Click this box to see this in daily view.")+"\"><tr>"+
QString("<td align=center onmouseover='ChangeColor(this, \"#efefa0\");' onmouseout='ChangeColor(this, \"#ffffc0\");' onclick='alert(\"daily=%1\");'>").arg(date.toString(Qt::ISODate))+"<b>"+ QString("<td align=center onmouseover='ChangeColor(this, \"#efefa0\");' onmouseout='ChangeColor(this, \"#ffffc0\");' onclick='alert(\"daily=%1\");'>").arg(date.toString(Qt::ISODate))+"<b>"+
QObject::tr("The last time you used your %1...").arg(day->machine->properties[STR_PROP_Brand]+" "+day->machine->properties[STR_PROP_Model])+"</b><br/>"; QObject::tr("The last time you used your %1...").arg(day->machine->properties[STR_PROP_Brand]+" "+day->machine->properties[STR_PROP_Model])+"</b><br/>";