OSCAR-code/oscar/SleepLib/journal.cpp

427 lines
15 KiB
C++
Raw Permalink Normal View History

2014-08-17 12:56:05 +00:00
/* SleepLib Journal Implementation
*
* Copyright (c) 2019-2024 The OSCAR Team
* Copyright (c) 2011-2018 Mark Watkins
2014-08-17 12:56:05 +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 source code
* for more details. */
2014-08-17 12:56:05 +00:00
#define TEST_MACROS_ENABLED
#include <test_macros.h>
#include <QMessageBox>
2014-08-17 12:56:05 +00:00
#include <QDomDocument>
#include <QDomElement>
#include <QFile>
#include <QTextStream>
#include <QXmlStreamWriter>
2014-08-17 12:56:05 +00:00
#include <QDir>
#include "journal.h"
#include "daily.h"
#include "SleepLib/common.h"
#include "SleepLib/machine_common.h"
#include "SleepLib/session.h"
#include "SleepLib/profiles.h"
#include "mainwindow.h"
extern MainWindow * mainwin;
const QString OLD_ZOMBIE = QString("zombie");
const QString ZOMBIE = QString("Feelings");
const QString WEIGHT = QString("weight");
bool Journal::BackupJournal(QString filename)
{
QString outBuf;
QXmlStreamWriter stream(&outBuf);
stream.setAutoFormatting(true);
stream.setAutoFormattingIndent(2);
stream.writeStartDocument();
// stream.writeProcessingInstruction("xml", "version=\"1.0\" encoding=\"UTF-8\"");
stream.writeStartElement("OSCAR");
stream.writeAttribute("created", QDateTime::currentDateTime().toString(Qt::ISODate));
stream.writeStartElement("Journal");
stream.writeAttribute("username", p_profile->user->userName());
stream.writeAttribute("height_cm", QString::number(p_profile->user->height()));
if (p_profile->general->unitSystem()!=US_Undefined) {
stream.writeAttribute("systemUnits", ( p_profile->general->unitSystem()==US_Metric?"Metric":"Englsh") );
}
QDate first = p_profile->FirstDay(MT_JOURNAL);
QDate last = p_profile->LastDay(MT_JOURNAL);
DEBUGFC Q(first) Q(last);
QDate date = first.addDays(-1);
int days_saved = 0 ;
do {
date = date.addDays(1);
Day * journal = p_profile->GetDay(date, MT_JOURNAL);
if (!journal) continue;
2014-08-17 12:56:05 +00:00
Session * sess = journal->firstSession(MT_JOURNAL);
if (!sess) continue;
if ( !journal->settingExists(Journal_Notes)
&& !journal->settingExists(Journal_Weight)
&& !journal->settingExists(Journal_ZombieMeter)
&& !journal->settingExists(LastUpdated)
&& !journal->settingExists(Bookmark_Start)) {
continue;
}
QString weight;
QString zombie;
QString notes;
QString lastupdated;
QVariantList start;
int havedata=0;
if (journal->settingExists(Journal_Weight)) {
weight = sess->settings[Journal_Weight].toString();
havedata |= JRNL_Weight ;
}
if (journal->settingExists(Journal_ZombieMeter)) {
zombie = sess->settings[Journal_ZombieMeter].toString();
havedata |= JRNL_Zombie ;
}
if (journal->settingExists(Journal_Notes)) {
notes = sess->settings[Journal_Notes].toString();
//notes = Daily::convertHtmlToPlainText(notes).trimmed();
havedata |= JRNL_Notes ;
}
if (journal->settingExists(Bookmark_Start)) {
start=sess->settings[Bookmark_Start].toList();
if (start.size()>0) havedata |= JRNL_Bookmarks ;
}
if (journal->settingExists(LastUpdated)) {
QDateTime dt = sess->settings[LastUpdated].toDateTime();
#if QT_VERSION < QT_VERSION_CHECK(5,8,0)
qint64 dtx = dt.toMSecsSinceEpoch()/1000L;
2020-05-26 19:34:22 +00:00
#else
qint64 dtx = dt.toSecsSinceEpoch();
#endif
lastupdated = QString::number(dtx);
}
if (!havedata) {
// No data to archive.
continue ;
}
//QString dateStr = ((++count & 1)==0) ? date.toString(/*Qt::ISODate*/) : date.toString(Qt::ISODate) ;
QString dateStr =date.toString(Qt::ISODate) ;
stream.writeStartElement("day");
stream.writeAttribute("date", dateStr);
if(!weight.isEmpty()) stream.writeAttribute(WEIGHT, weight);
if(!zombie.isEmpty()) stream.writeAttribute(ZOMBIE, zombie);
stream.writeAttribute("lastupdated", lastupdated);
if (!notes.isEmpty() ) {
stream.writeStartElement("note");
stream.writeTextElement("text", notes);
stream.writeEndElement(); // notes
}
if (start.size()>0) {
QVariantList end=sess->settings[Bookmark_End].toList();
QStringList notes=sess->settings[Bookmark_Notes].toStringList();
stream.writeStartElement("bookmarks");
int size = start.size();
for (int i=0; i< size; i++) {
stream.writeStartElement("bookmark");
stream.writeAttribute("notes",notes.at(i));
stream.writeAttribute("start",start.at(i).toString());
stream.writeAttribute("end",end.at(i).toString());
stream.writeEndElement(); // bookmark
}
stream.writeEndElement(); // bookmarks
}
days_saved++;
stream.writeEndElement(); // day
} while (date <= last);
// //stream.writeAttribute("DaysSaved", QString::number(days_saved));
stream.writeEndElement(); // Journal
stream.writeEndElement(); // OSCAR
stream.writeEndDocument();
QFile file(filename);
if (!file.open(QIODevice::WriteOnly)) {
qWarning() << "Couldn't open journal file" << filename << "error code" << file.error() << file.errorString();
return false;
}
QTextStream ts(&file);
ts.setCodec("UTF-8");
ts.setGenerateByteOrderMark(true);
ts << outBuf;
file.close();
QMessageBox::information(nullptr, STR_MessageBox_Information,
QString(QObject::tr("%1 days Journal Data was saved in file %2")).arg(days_saved).arg(filename) ,
QMessageBox::Ok);
return true;
}
2014-08-17 12:56:05 +00:00
void Journal::getJournal(Daily*& daily,QDate& date,Session* & journal) {
if (journal) return;
journal = daily->GetJournalSession(date,true);
if (journal) return;
journal = daily->CreateJournalSession(date);
2014-08-17 12:56:05 +00:00
}
bool Journal::RestoreDay (QDomElement& dayElement,QDate& date,QString& filename) {
Daily* daily = mainwin->getDaily();
if (!daily) return false;
Session* journal = nullptr;
bool changed = false;
// handle zombie - feelings
bool ok = false;
int zombie = 0;
zombie = (dayElement.attribute(ZOMBIE)).toInt(&ok);
if (!ok) { zombie=0; }
if (zombie == 0 ) {
zombie = (dayElement.attribute(OLD_ZOMBIE)).toInt(&ok);
if (!ok) { zombie=0; }
}
DEBUGFC O("\n\n") ;
DEBUGFC O(date) O(filename); Q_UNUSED(filename);
if (zombie>0) {
int jvalue = 0 ;
getJournal(daily,date,journal);
if (journal->settings.contains(Journal_ZombieMeter)) {
jvalue = journal->settings[Journal_ZombieMeter].toInt();
}
if (jvalue == 0) {
DEBUGFC O(date) Q(zombie);
daily->set_JournalZombie(date,zombie);
if (date == daily->getDate()) daily->set_ZombieUI(zombie);
changed = true;
}
}
// handle Weight
double weight = (dayElement.attribute(WEIGHT)).toDouble(&ok);
if (weight>zeroD) {
getJournal(daily,date,journal);
double jvalue = 0.0 ;
if (journal->settings.contains(Journal_Weight)) {
jvalue = journal->settings[Journal_Weight].toDouble();
}
if (jvalue < zeroD ) {
DEBUGFC O(date) Q(weight);
daily->set_JournalWeightValue(date,weight);
if (date == daily->getDate()) daily->set_WeightUI(weight);
changed = true;
}
}
// Handle Notes.
QDomElement noteText = dayElement.elementsByTagName("note").at(0).toElement().elementsByTagName("text").at(0).toElement();
if (!noteText.text().isEmpty() ) {
getJournal(daily,date,journal);
// there are characters in notes. maybe just spaces. Ignore spaces.
QString plainTextToAdd = Daily::convertHtmlToPlainText(noteText.text());
// get existing note if any.
QString currNoteHtml = journal->settings[Journal_Notes].toString();
QString currNotePlainText = Daily::convertHtmlToPlainText(currNoteHtml);
if (currNotePlainText.contains(plainTextToAdd) ) {
// plainText to add is equal to curr or is it a subset of the current Note
// use the current notes.. ignore text in backup.
} else {
// plainText is not equal to curr nor is it a subset of the current Note
if (plainTextToAdd.contains(currNotePlainText) ) {
//curr note is a subset of the new - so use new.
currNoteHtml = noteText.text();
} else {
// ToAdd text and Current text are different.
// use previous verson append with current
currNoteHtml.prepend(noteText.text());
}
daily->set_JournalNotesHtml(date,currNoteHtml);
if (date == daily->getDate()) daily->set_NotesUI(noteText.text());
DEBUGFC O(date) Q((void*)journal) O( )Daily::convertHtmlToPlainText(currNoteHtml);
changed = true;
}
}
QDomNodeList bookmarks = dayElement.elementsByTagName("bookmarks").at(0).toElement().elementsByTagName("bookmark");
if (bookmarks.size()>0) {
DEBUGFC Q(bookmarks.size());
getJournal(daily,date,journal);
// get list of bookmarks for journal. These will not be removed.
QVariantList start;
QVariantList end;
QStringList notes;
if (journal->settings.contains(Bookmark_Start)) {
//DEBUGFC;
start=journal->settings[Bookmark_Start].toList();
end=journal->settings[Bookmark_End].toList();
notes=journal->settings[Bookmark_Notes].toStringList();
}
// check if arcived bookmark is current list
bool bmChanged = false;
for (int idx=0 ; idx < bookmarks.size() ; idx++) {
// get archived bookmark
//DEBUGFC Q(idx) Q(bookmarks.size()) Q(start.size()) ;
QDomElement bookmark = bookmarks.at(idx).toElement();
qint64 archiveStart = bookmark.attribute("start").toLongLong();
qint64 archiveEnd = bookmark.attribute("end").toLongLong();
QString archiveNote = bookmark.attribute("notes");
// check if bookmark already exists.
bool duplicate = false;
bool bNoteChanged = false;
for (int idy=0 ; idy < start.size() ; idy++) {
//DEBUGFC Q(idy);
qint64 bmStart = start.at(idy).toLongLong();
qint64 bmEnd = end.at(idy).toLongLong();
if ( (bmStart == archiveStart) && (bmEnd == archiveEnd) ) {
duplicate = true;
//DEBUGFC Q(idx) Q(idy);
// have same bookmark - new check if notes need merging
QString aNote = archiveNote.simplified();
QString bmNote = notes.at(idy);
QString bNote = bmNote.simplified();
DEBUGFC Q(aNote) Q(bNote) Q(bNote.contains(aNote)) Q(aNote.contains(bNote));
if (bNote.contains(aNote) ) {
// no action needed.
//DEBUGFC Q(idx) Q(idy) Q(bNote);;
break;
} else {
// updated existing bookmark label
//DEBUGFC Q(idx) Q(idy);
// if an append is needed.
if (aNote.contains(bNote)) {
// use archived note
//DEBUGFC Q(idx) Q(idy);
} else {
//DEBUGFC Q(idx) Q(idy);
// prepend archive note to bmNote
archiveNote.append(" :: ").append(bmNote);
}
// use archiveNote.
notes[idy]= archiveNote;
bmChanged = true;
bNoteChanged = true;
}
}
}
if (!duplicate) {
DEBUGFC Q(date) Q(idx) Q(archiveNote);
// here if need to add archive bookmark.
// add archive bookmark to currrent list
start.push_back(archiveStart);
end.push_back(archiveEnd);
notes.push_back(archiveNote);
bmChanged = true;
} else if (bNoteChanged) {
DEBUGFC Q(date) Q(idx) Q(archiveNote);
}
}
if (bmChanged) {
//DEBUGFC Q(bmChanged);
getJournal(daily,date,journal);
journal->settings[Bookmark_Start]=start;
journal->settings[Bookmark_End]=end;
journal->settings[Bookmark_Notes]=notes;
if (date == daily->getDate()) daily->set_BookmarksUI(start,end,notes,0);
} else {
DEBUGFC O("bookmark ignored") ;
}
}
if (!journal) {
return true;
}
if (!journal->machine()) {
return true;
}
if (changed) {
journal->SetChanged(true);
journal->settings[LastUpdated] = QDateTime::currentDateTime();
journal->machine()->SaveSummaryCache();
journal->SetChanged(false);
}
DEBUGFC O(date) O("\n\n");
return true;
2014-08-17 12:56:05 +00:00
}
bool Journal::RestoreJournal(QString filename)
2014-08-17 12:56:05 +00:00
{
QFile file(filename);
if (!file.open(QFile::ReadOnly)) {
qWarning() << "Could not open" << filename.toLocal8Bit().data() << "for reading, error code" << file.error() << file.errorString();
return false;
}
QDomDocument doc("machines.xml");
if (!doc.setContent(&file)) {
qWarning() << "Invalid XML Content in" << filename.toLocal8Bit().data();
return false;
}
file.close();
QDomElement root = doc.documentElement(); // Get the root element of the document
if (root.isNull()) {
return false;
}
QDomNodeList days = doc.elementsByTagName("day");
if (days.isEmpty()) {
return false;
2014-08-17 12:56:05 +00:00
}
QDate first = p_profile->FirstDay();
QDate last = p_profile->LastDay();
QDate current = QDate::currentDate();
//int used=0;
for (int idx=0 ; idx < days.size() ; idx++) {
QDomElement dayElement = days.at(idx).toElement();
QString dateStr = dayElement.attribute("date") ;
QDate date = QDate::fromString(dateStr,Qt::ISODate) ;
if (!date.isValid()) {
QDate newdate = QDate::fromString(dateStr) ; // read original date format
date = newdate;
}
if (!date.isValid()) {
continue;
}
if (date < first) {
continue;
}
if (date > last) {
continue;
}
if (date > current) {
continue;
}
if ( RestoreDay(dayElement,date,filename) ) {
//used++;
}
}
double user_height_cm = p_profile->user->height();
if (user_height_cm<zeroD) {
QDomElement journal = root.elementsByTagName("Journal").at(0).toElement();
double height_cm = journal.attribute("height_cm").toDouble();
DEBUGFC Q(journal.tagName()) Q(height_cm);
if (height_cm>=zeroD) p_profile->user->setHeight(height_cm);
}
return true;
2014-08-17 12:56:05 +00:00
}