2018-04-27 04:29:03 +00:00
|
|
|
|
/* SleepLib Somnopose Loader Implementation
|
2014-04-15 13:59:24 +00:00
|
|
|
|
*
|
2018-04-02 02:08:32 +00:00
|
|
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
2014-04-15 13:59:24 +00:00
|
|
|
|
*
|
|
|
|
|
* 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. */
|
|
|
|
|
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
// IMPORTANT!!!
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
// Please INCREMENT the somnopose_data_version in somnopose_loader.h when making changes to this loader
|
|
|
|
|
// that change loader behaviour or modify channels.
|
|
|
|
|
//********************************************************************************************
|
|
|
|
|
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QTextStream>
|
|
|
|
|
#include "somnopose_loader.h"
|
|
|
|
|
#include "SleepLib/machine.h"
|
|
|
|
|
|
|
|
|
|
SomnoposeLoader::SomnoposeLoader()
|
|
|
|
|
{
|
2014-05-25 07:07:08 +00:00
|
|
|
|
m_type = MT_POSITION;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SomnoposeLoader::~SomnoposeLoader()
|
|
|
|
|
{
|
|
|
|
|
}
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int SomnoposeLoader::Open(const QString & dirpath)
|
2014-04-15 13:59:24 +00:00
|
|
|
|
{
|
|
|
|
|
QString newpath;
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QString dirtag = "somnopose";
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
|
|
|
|
// Could Scan the ZEO folder for a list of CSVs
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
QString path(dirpath);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
path = path.replace("\\", "/");
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (path.toLower().endsWith("/" + dirtag)) {
|
2014-04-15 13:59:24 +00:00
|
|
|
|
return 0;
|
|
|
|
|
//newpath=path;
|
|
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
newpath = path + "/" + dirtag.toUpper();
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//QString filename;
|
|
|
|
|
|
|
|
|
|
// Somnopose folder structure detection stuff here.
|
|
|
|
|
|
|
|
|
|
return 0; // number of machines affected
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int SomnoposeLoader::OpenFile(const QString & filename)
|
2014-04-15 13:59:24 +00:00
|
|
|
|
{
|
|
|
|
|
QFile file(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-04-15 13:59:24 +00:00
|
|
|
|
if (filename.toLower().endsWith(".csv")) {
|
|
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
|
|
|
|
qDebug() << "Couldn't open Somnopose data file" << filename;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
qDebug() << "Opening file" << filename;
|
|
|
|
|
QTextStream ts(&file);
|
|
|
|
|
|
|
|
|
|
// Read header line and determine order of fields
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QString hdr = ts.readLine();
|
|
|
|
|
QStringList headers = hdr.split(",");
|
|
|
|
|
|
|
|
|
|
int col_inclination = -1, col_orientation = -1, col_timestamp = -1;
|
|
|
|
|
|
|
|
|
|
int hdr_size = headers.size();
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < hdr_size; i++) {
|
|
|
|
|
if (headers.at(i).compare("timestamp", Qt::CaseInsensitive) == 0) {
|
|
|
|
|
col_timestamp = i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (headers.at(i).compare("inclination", Qt::CaseInsensitive) == 0) {
|
|
|
|
|
col_inclination = i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (headers.at(i).compare("orientation", Qt::CaseInsensitive) == 0) {
|
|
|
|
|
col_orientation = i;
|
|
|
|
|
}
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check we have all fields available
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if ((col_timestamp < 0) || (col_inclination < 0) || (col_orientation < 0)) {
|
2014-04-15 13:59:24 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QDateTime epoch(QDate(2001, 1, 1), QTime(0, 0, 0));
|
|
|
|
|
qint64 ep = qint64(epoch.toTime_t()) * 1000, time;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
double timestamp, orientation, inclination;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
QString data;
|
|
|
|
|
QStringList fields;
|
|
|
|
|
bool ok;
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
bool first = true;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
MachineInfo info = newInfo();
|
2018-04-22 12:06:48 +00:00
|
|
|
|
Machine *mach = p_profile->CreateMachine(info);
|
2014-04-23 13:19:56 +00:00
|
|
|
|
Session *sess = nullptr;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
SessionID sid;
|
|
|
|
|
|
2014-04-23 13:19:56 +00:00
|
|
|
|
EventList *ev_orientation = nullptr, *ev_inclination = nullptr;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
while (!(data = ts.readLine()).isEmpty()) {
|
|
|
|
|
fields = data.split(",");
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (fields.size() < hdr_size) { // missing fields.. skip this record
|
2014-04-15 13:59:24 +00:00
|
|
|
|
continue;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
}
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
timestamp = fields[col_timestamp].toDouble(&ok);
|
|
|
|
|
|
|
|
|
|
if (!ok) { continue; }
|
|
|
|
|
|
|
|
|
|
orientation = fields[col_orientation].toDouble(&ok);
|
|
|
|
|
|
|
|
|
|
if (!ok) { continue; }
|
|
|
|
|
|
|
|
|
|
inclination = fields[col_inclination].toDouble(&ok);
|
|
|
|
|
|
|
|
|
|
if (!ok) { continue; }
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
|
|
|
|
// convert to milliseconds since epoch
|
2014-04-17 05:58:57 +00:00
|
|
|
|
time = (timestamp * 1000.0) + ep;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
|
|
|
|
if (first) {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sid = time / 1000;
|
|
|
|
|
|
2014-04-15 13:59:24 +00:00
|
|
|
|
if (mach->SessionExists(sid)) {
|
|
|
|
|
return 0; // Already imported
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
sess = new Session(mach, sid);
|
2014-04-15 13:59:24 +00:00
|
|
|
|
sess->really_set_first(time);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
ev_orientation = sess->AddEventList(POS_Orientation, EVL_Event, 1, 0, 0, 0);
|
|
|
|
|
ev_inclination = sess->AddEventList(POS_Inclination, EVL_Event, 1, 0, 0, 0);
|
|
|
|
|
first = false;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-04-15 13:59:24 +00:00
|
|
|
|
sess->set_last(time);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
ev_orientation->AddEvent(time, orientation);
|
|
|
|
|
ev_inclination->AddEvent(time, inclination);
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2018-03-28 06:22:42 +00:00
|
|
|
|
if (sess) {
|
|
|
|
|
if (ev_orientation && ev_inclination) {
|
|
|
|
|
sess->setMin(POS_Orientation, ev_orientation->Min());
|
|
|
|
|
sess->setMax(POS_Orientation, ev_orientation->Max());
|
|
|
|
|
sess->setMin(POS_Inclination, ev_inclination->Min());
|
|
|
|
|
sess->setMax(POS_Inclination, ev_inclination->Max());
|
|
|
|
|
}
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2018-03-28 06:22:42 +00:00
|
|
|
|
sess->really_set_last(time);
|
|
|
|
|
sess->SetChanged(true);
|
2014-09-17 06:12:38 +00:00
|
|
|
|
|
2018-03-28 06:22:42 +00:00
|
|
|
|
mach->AddSession(sess);
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
2018-03-28 06:22:42 +00:00
|
|
|
|
mach->Save();
|
|
|
|
|
}
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
static bool somnopose_initialized = false;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
|
|
|
|
|
void SomnoposeLoader::Register()
|
|
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (somnopose_initialized) { return; }
|
|
|
|
|
|
2014-04-15 13:59:24 +00:00
|
|
|
|
qDebug("Registering SomnoposeLoader");
|
|
|
|
|
RegisterLoader(new SomnoposeLoader());
|
|
|
|
|
//InitModelMap();
|
2014-04-17 05:58:57 +00:00
|
|
|
|
somnopose_initialized = true;
|
2014-04-15 13:59:24 +00:00
|
|
|
|
}
|
|
|
|
|
|