OSCAR-code/sleepyhead/exportcsv.cpp

402 lines
15 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim: set ts=8 sts=4 et sw=4 tw=99:
*
* ExportCSV module implementation
*
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.net>
*
* 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 Linux
* distribution for more details. */
2011-11-14 14:10:46 +00:00
#include <QFileDialog>
#include <QLocale>
#include <QMessageBox>
#include <QCalendarWidget>
#include <QTextCharFormat>
2011-11-14 14:10:46 +00:00
#include "SleepLib/profiles.h"
2011-11-14 15:43:56 +00:00
#include "SleepLib/day.h"
#include "common_gui.h"
2011-11-14 14:10:46 +00:00
#include "exportcsv.h"
#include "ui_exportcsv.h"
#include "mainwindow.h"
extern MainWindow *mainwin;
2011-11-14 14:10:46 +00:00
ExportCSV::ExportCSV(QWidget *parent) :
QDialog(parent),
ui(new Ui::ExportCSV)
{
ui->setupUi(this);
ui->rb1_Summary->setChecked(true);
ui->quickRangeCombo->setCurrentIndex(0);
// Set Date controls locale to 4 digit years
QLocale locale = QLocale::system();
QString shortformat = locale.dateFormat(QLocale::ShortFormat);
2011-11-14 14:10:46 +00:00
if (!shortformat.toLower().contains("yyyy")) {
shortformat.replace("yy", "yyyy");
2011-11-14 14:10:46 +00:00
}
2011-11-14 14:10:46 +00:00
ui->startDate->setDisplayFormat(shortformat);
ui->endDate->setDisplayFormat(shortformat);
// Stop both calendar drop downs highlighting weekends in red
QTextCharFormat format = ui->startDate->calendarWidget()->weekdayTextFormat(Qt::Saturday);
format.setForeground(QBrush(Qt::black, Qt::SolidPattern));
ui->startDate->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format);
ui->startDate->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format);
ui->endDate->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format);
ui->endDate->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format);
Qt::DayOfWeek dow = firstDayOfWeekFromLocale();
ui->startDate->calendarWidget()->setFirstDayOfWeek(dow);
ui->endDate->calendarWidget()->setFirstDayOfWeek(dow);
// Connect the signals to update which days have CPAP data when the month is changed
connect(ui->startDate->calendarWidget(), SIGNAL(currentPageChanged(int, int)),
SLOT(startDate_currentPageChanged(int, int)));
connect(ui->endDate->calendarWidget(), SIGNAL(currentPageChanged(int, int)),
SLOT(endDate_currentPageChanged(int, int)));
2011-11-14 14:10:46 +00:00
on_quickRangeCombo_activated(tr("Most Recent Day"));
2011-11-14 14:10:46 +00:00
ui->rb1_details->clearFocus();
ui->quickRangeCombo->setFocus();
ui->exportButton->setEnabled(false);
}
ExportCSV::~ExportCSV()
{
delete ui;
}
void ExportCSV::on_filenameBrowseButton_clicked()
{
QString timestamp = tr("SleepyHead_");
timestamp += PROFILE.Get("Username") + "_";
if (ui->rb1_details->isChecked()) { timestamp += tr("Details_"); }
if (ui->rb1_Sessions->isChecked()) { timestamp += tr("Sessions_"); }
if (ui->rb1_Summary->isChecked()) { timestamp += tr("Summary_"); }
2011-11-14 14:10:46 +00:00
timestamp += ui->startDate->date().toString(Qt::ISODate);
if (ui->startDate->date() != ui->endDate->date()) { timestamp += "_" + ui->endDate->date().toString(Qt::ISODate); }
timestamp += ".csv";
QString name = QFileDialog::getSaveFileName(this, tr("Select file to export to"),
PREF.Get("{home}/") + timestamp, tr("CSV Files (*.csv)"));
2011-11-14 14:10:46 +00:00
if (name.isEmpty()) {
ui->exportButton->setEnabled(false);
return;
}
2011-11-14 14:10:46 +00:00
if (!name.toLower().endsWith(".csv")) {
name += ".csv";
2011-11-14 14:10:46 +00:00
}
ui->filenameEdit->setText(name);
ui->exportButton->setEnabled(true);
}
void ExportCSV::on_quickRangeCombo_activated(const QString &arg1)
{
QDate first = PROFILE.FirstDay();
QDate last = PROFILE.LastDay();
if (arg1 == tr("Custom")) {
2011-11-14 14:10:46 +00:00
ui->startDate->setEnabled(true);
ui->endDate->setEnabled(true);
ui->startLabel->setEnabled(true);
ui->endLabel->setEnabled(true);
} else {
ui->startDate->setEnabled(false);
ui->endDate->setEnabled(false);
ui->startLabel->setEnabled(false);
ui->endLabel->setEnabled(false);
if (arg1 == tr("Everything")) {
ui->startDate->setDate(first);
ui->endDate->setDate(last);
} else if (arg1 == tr("Most Recent Day")) {
ui->startDate->setDate(last);
ui->endDate->setDate(last);
} else if (arg1 == tr("Last Week")) {
ui->startDate->setDate(last.addDays(-7));
ui->endDate->setDate(last);
} else if (arg1 == tr("Last Fortnight")) {
ui->startDate->setDate(last.addDays(-14));
ui->endDate->setDate(last);
} else if (arg1 == tr("Last Month")) {
ui->startDate->setDate(last.addMonths(-1));
ui->endDate->setDate(last);
} else if (arg1 == tr("Last 6 Months")) {
ui->startDate->setDate(last.addMonths(-6));
ui->endDate->setDate(last);
} else if (arg1 == tr("Last Year")) {
ui->startDate->setDate(last.addYears(-1));
ui->endDate->setDate(last);
2011-11-14 14:10:46 +00:00
}
}
}
void ExportCSV::on_exportButton_clicked()
{
2011-11-14 15:43:56 +00:00
QFile file(ui->filenameEdit->text());
file.open(QFile::WriteOnly);
QString header;
const QString sep = ",";
const QString newline = "\n";
// if (ui->rb1_details->isChecked()) {
// fields.append(DumpField(NoChannel,MT_CPAP,ST_DATE));
// } else {
// header=tr("DateTime")+sep+tr("Session")+sep+tr("Event")+sep+tr("Data/Duration");
// } else {
// if (ui->rb1_Summary->isChecked()) {
// header=tr("Date")+sep+tr("Session Count")+sep+tr("Start")+sep+tr("End")+sep+tr("Total Time")+sep+tr("AHI");
// } else if (ui->rb1_Sessions->isChecked()) {
// header=tr("Date")+sep+tr("Session")+sep+tr("Start")+sep+tr("End")+sep+tr("Total Time")+sep+tr("AHI");
// }
// }
// fields.append(DumpField(NoChannel,MT_CPAP,ST_SESSIONS));
QList<ChannelID> countlist, avglist, p90list;
2011-11-14 15:43:56 +00:00
countlist.append(CPAP_Hypopnea);
countlist.append(CPAP_Obstructive);
countlist.append(CPAP_Apnea);
countlist.append(CPAP_ClearAirway);
countlist.append(CPAP_VSnore);
countlist.append(CPAP_VSnore2);
countlist.append(CPAP_RERA);
countlist.append(CPAP_FlowLimit);
countlist.append(CPAP_NRI);
countlist.append(CPAP_ExP);
countlist.append(CPAP_LeakFlag);
countlist.append(CPAP_UserFlag1);
countlist.append(CPAP_UserFlag2);
2011-11-14 15:43:56 +00:00
countlist.append(CPAP_PressurePulse);
2011-11-14 15:43:56 +00:00
avglist.append(CPAP_Pressure);
avglist.append(CPAP_IPAP);
avglist.append(CPAP_EPAP);
p90list.append(CPAP_Pressure);
p90list.append(CPAP_IPAP);
p90list.append(CPAP_EPAP);
EventDataType percent = 0.90F;
2011-11-14 15:43:56 +00:00
// Not sure this section should be translateable.. :-/
2011-11-14 15:43:56 +00:00
if (ui->rb1_details->isChecked()) {
header = tr("DateTime") + sep + tr("Session") + sep + tr("Event") + sep + tr("Data/Duration");
2011-11-14 15:43:56 +00:00
} else {
if (ui->rb1_Summary->isChecked()) {
header = tr("Date") + sep + tr("Session Count") + sep + tr("Start") + sep + tr("End") + sep +
tr("Total Time") + sep + tr("AHI");
2011-11-14 15:43:56 +00:00
} else if (ui->rb1_Sessions->isChecked()) {
header = tr("Date") + sep + tr("Session") + sep + tr("Start") + sep + tr("End") + sep +
tr("Total Time") + sep + tr("AHI");
}
for (int i = 0; i < countlist.size(); i++) {
header += sep + schema::channel[countlist[i]].label() + tr(" Count");
}
for (int i = 0; i < avglist.size(); i++) {
header += sep + schema::channel[avglist[i]].label() + " " + tr(" Avg");
}
for (int i = 0; i < p90list.size(); i++) {
header += sep + schema::channel[p90list[i]].label() + tr(" %1%").arg(percent, 0, 'f', 0);
2011-11-14 15:43:56 +00:00
}
}
header += newline;
file.write(header.toLatin1());
QDate date = ui->startDate->date();
Daily *daily = mainwin->getDaily();
QDate daily_date = daily->getDate();
ui->progressBar->setValue(0);
ui->progressBar->setMaximum(PROFILE.daylist.count());
2011-11-14 15:43:56 +00:00
do {
ui->progressBar->setValue(ui->progressBar->value() + 1);
QApplication::processEvents();
Day *day = PROFILE.GetDay(date, MT_CPAP);
2011-11-14 15:43:56 +00:00
if (day) {
QString data;
2011-11-14 15:43:56 +00:00
if (ui->rb1_Summary->isChecked()) {
QDateTime start = QDateTime::fromTime_t(day->first() / 1000L);
QDateTime end = QDateTime::fromTime_t(day->last() / 1000L);
data = date.toString(Qt::ISODate);
data += sep + QString::number(day->size(), 10);
data += sep + start.toString(Qt::ISODate);
data += sep + end.toString(Qt::ISODate);
int time = day->total_time() / 1000L;
int h = time / 3600;
int m = int(time / 60) % 60;
int s = int(time) % 60;
data += sep + QString().sprintf("%02i:%02i:%02i", h, m, s);
float ahi = day->count(CPAP_Obstructive) + day->count(CPAP_Hypopnea) + day->count(
CPAP_Apnea) + day->count(CPAP_ClearAirway);
ahi /= day->hours();
data += sep + QString::number(ahi, 'f', 3);
for (int i = 0; i < countlist.size(); i++) {
data += sep + QString::number(day->count(countlist.at(i)));
}
for (int i = 0; i < avglist.size(); i++) {
data += sep + QString::number(day->wavg(avglist.at(i)));
}
for (int i = 0; i < p90list.size(); i++) {
data += sep + QString::number(day->p90(p90list.at(i)));
}
data += newline;
file.write(data.toLatin1());
2011-11-14 15:43:56 +00:00
} else if (ui->rb1_Sessions->isChecked()) {
for (int i = 0; i < day->size(); i++) {
Session *sess = (*day)[i];
QDateTime start = QDateTime::fromTime_t(sess->first() / 1000L);
QDateTime end = QDateTime::fromTime_t(sess->last() / 1000L);
data = date.toString(Qt::ISODate);
data += sep + QString::number(sess->session(), 10);
data += sep + start.toString(Qt::ISODate);
data += sep + end.toString(Qt::ISODate);
int time = sess->length() / 1000L;
int h = time / 3600;
int m = int(time / 60) % 60;
int s = int(time) % 60;
data += sep + QString().sprintf("%02i:%02i:%02i", h, m, s);
float ahi = sess->count(CPAP_Obstructive) + sess->count(CPAP_Hypopnea) + sess->count(
CPAP_Apnea) + sess->count(CPAP_ClearAirway);
ahi /= sess->hours();
data += sep + QString::number(ahi, 'f', 3);
for (int j = 0; j < countlist.size(); j++) {
data += sep + QString::number(sess->count(countlist.at(j)));
}
for (int j = 0; j < avglist.size(); j++) {
data += sep + QString::number(day->wavg(avglist.at(j)));
}
for (int j = 0; j < p90list.size(); j++) {
data += sep + QString::number(day->p90(p90list.at(j)));
}
data += newline;
file.write(data.toLatin1());
2011-11-14 15:43:56 +00:00
}
} else if (ui->rb1_details->isChecked()) {
QList<ChannelID> all = countlist;
2011-11-14 15:43:56 +00:00
all.append(avglist);
for (int i = 0; i < day->size(); i++) {
Session *sess = (*day)[i];
2011-11-14 15:43:56 +00:00
sess->OpenEvents();
QHash<ChannelID, QVector<EventList *> >::iterator fnd;
for (int j = 0; j < all.size(); j++) {
ChannelID key = all.at(j);
fnd = sess->eventlist.find(key);
if (fnd != sess->eventlist.end()) {
2011-11-14 15:43:56 +00:00
//header="DateTime"+sep+"Session"+sep+"Event"+sep+"Data/Duration";
for (int e = 0; e < fnd.value().size(); e++) {
EventList *ev = fnd.value()[e];
for (quint32 q = 0; q < ev->count(); q++) {
data = QDateTime::fromTime_t(ev->time(q) / 1000L).toString(Qt::ISODate);
data += sep + QString::number(sess->session());
data += sep + schema::channel[key].code();
data += sep + QString::number(ev->data(q), 'f', 2);
data += newline;
file.write(data.toLatin1());
2011-11-14 15:43:56 +00:00
}
}
}
}
if (daily_date != date) {
sess->TrashEvents();
}
2011-11-14 15:43:56 +00:00
}
}
}
date = date.addDays(1);
} while (date <= ui->endDate->date());
2011-11-14 15:43:56 +00:00
file.close();
2011-11-14 14:10:46 +00:00
ExportCSV::accept();
}
void ExportCSV::UpdateCalendarDay(QDateEdit *dateedit, QDate date)
{
QCalendarWidget *calendar = dateedit->calendarWidget();
QTextCharFormat bold;
QTextCharFormat cpapcol;
QTextCharFormat normal;
QTextCharFormat oxiday;
bold.setFontWeight(QFont::Bold);
cpapcol.setForeground(QBrush(Qt::blue, Qt::SolidPattern));
cpapcol.setFontWeight(QFont::Bold);
oxiday.setForeground(QBrush(Qt::red, Qt::SolidPattern));
oxiday.setFontWeight(QFont::Bold);
bool hascpap = p_profile->GetDay(date, MT_CPAP) != NULL;
bool hasoxi = p_profile->GetDay(date, MT_OXIMETER) != NULL;
//bool hasjournal=p_profile->GetDay(date,MT_JOURNAL)!=NULL;
if (hascpap) {
if (hasoxi) {
calendar->setDateTextFormat(date, oxiday);
} else {
calendar->setDateTextFormat(date, cpapcol);
}
} else if (p_profile->GetDay(date)) {
calendar->setDateTextFormat(date, bold);
} else {
calendar->setDateTextFormat(date, normal);
}
calendar->setHorizontalHeaderFormat(QCalendarWidget::ShortDayNames);
}
void ExportCSV::startDate_currentPageChanged(int year, int month)
{
QDate d(year, month, 1);
int dom = d.daysInMonth();
for (int i = 1; i <= dom; i++) {
d = QDate(year, month, i);
UpdateCalendarDay(ui->startDate, d);
}
}
void ExportCSV::endDate_currentPageChanged(int year, int month)
{
QDate d(year, month, 1);
int dom = d.daysInMonth();
for (int i = 1; i <= dom; i++) {
d = QDate(year, month, i);
UpdateCalendarDay(ui->endDate, d);
}
}