From 9ed886e4bea6c88a427a8fce50e2cb226d24afcc Mon Sep 17 00:00:00 2001 From: Berthold Stoeger Date: Mon, 3 Feb 2020 19:33:06 +0100 Subject: Cleanup: lower-case filenames in core/subsurface-qt/ We tend to use lower-case filenames. Let's do it for these files as well. Simple search & replace. Signed-off-by: Berthold Stoeger --- core/subsurface-qt/CylinderObjectHelper.cpp | 18 -- core/subsurface-qt/CylinderObjectHelper.h | 29 --- core/subsurface-qt/DiveListNotifier.cpp | 4 - core/subsurface-qt/DiveListNotifier.h | 208 ------------------ core/subsurface-qt/DiveObjectHelper.cpp | 330 ---------------------------- core/subsurface-qt/DiveObjectHelper.h | 105 --------- core/subsurface-qt/cylinderobjecthelper.cpp | 18 ++ core/subsurface-qt/cylinderobjecthelper.h | 29 +++ core/subsurface-qt/divelistnotifier.cpp | 4 + core/subsurface-qt/divelistnotifier.h | 208 ++++++++++++++++++ core/subsurface-qt/diveobjecthelper.cpp | 330 ++++++++++++++++++++++++++++ core/subsurface-qt/diveobjecthelper.h | 105 +++++++++ 12 files changed, 694 insertions(+), 694 deletions(-) delete mode 100644 core/subsurface-qt/CylinderObjectHelper.cpp delete mode 100644 core/subsurface-qt/CylinderObjectHelper.h delete mode 100644 core/subsurface-qt/DiveListNotifier.cpp delete mode 100644 core/subsurface-qt/DiveListNotifier.h delete mode 100644 core/subsurface-qt/DiveObjectHelper.cpp delete mode 100644 core/subsurface-qt/DiveObjectHelper.h create mode 100644 core/subsurface-qt/cylinderobjecthelper.cpp create mode 100644 core/subsurface-qt/cylinderobjecthelper.h create mode 100644 core/subsurface-qt/divelistnotifier.cpp create mode 100644 core/subsurface-qt/divelistnotifier.h create mode 100644 core/subsurface-qt/diveobjecthelper.cpp create mode 100644 core/subsurface-qt/diveobjecthelper.h (limited to 'core/subsurface-qt') diff --git a/core/subsurface-qt/CylinderObjectHelper.cpp b/core/subsurface-qt/CylinderObjectHelper.cpp deleted file mode 100644 index 28c14e10a..000000000 --- a/core/subsurface-qt/CylinderObjectHelper.cpp +++ /dev/null @@ -1,18 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -#include "CylinderObjectHelper.h" -#include "../qthelper.h" - -static QString EMPTY_CYLINDER_STRING = QStringLiteral(""); -CylinderObjectHelper::CylinderObjectHelper(const cylinder_t *cylinder) -{ - if (!cylinder) - return; - - description = cylinder->type.description ? cylinder->type.description: - EMPTY_CYLINDER_STRING; - size = get_volume_string(cylinder->type.size, true); - workingPressure = get_pressure_string(cylinder->type.workingpressure, true); - startPressure = get_pressure_string(cylinder->start, true); - endPressure = get_pressure_string(cylinder->end, true); - gasMix = get_gas_string(cylinder->gasmix); -} diff --git a/core/subsurface-qt/CylinderObjectHelper.h b/core/subsurface-qt/CylinderObjectHelper.h deleted file mode 100644 index e1e2490e4..000000000 --- a/core/subsurface-qt/CylinderObjectHelper.h +++ /dev/null @@ -1,29 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -#ifndef CYLINDER_QOBJECT_H -#define CYLINDER_QOBJECT_H - -#include "../equipment.h" -#include -#include - -class CylinderObjectHelper { - Q_GADGET - Q_PROPERTY(QString description MEMBER description CONSTANT) - Q_PROPERTY(QString size MEMBER size CONSTANT) - Q_PROPERTY(QString workingPressure MEMBER workingPressure CONSTANT) - Q_PROPERTY(QString startPressure MEMBER startPressure CONSTANT) - Q_PROPERTY(QString endPressure MEMBER endPressure CONSTANT) - Q_PROPERTY(QString gasMix MEMBER gasMix CONSTANT) -public: - CylinderObjectHelper(const cylinder_t *cylinder = NULL); - QString description; - QString size; - QString workingPressure; - QString startPressure; - QString endPressure; - QString gasMix; -}; - -Q_DECLARE_METATYPE(CylinderObjectHelper) - -#endif diff --git a/core/subsurface-qt/DiveListNotifier.cpp b/core/subsurface-qt/DiveListNotifier.cpp deleted file mode 100644 index 2009defcd..000000000 --- a/core/subsurface-qt/DiveListNotifier.cpp +++ /dev/null @@ -1,4 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -#include "DiveListNotifier.h" - -DiveListNotifier diveListNotifier; diff --git a/core/subsurface-qt/DiveListNotifier.h b/core/subsurface-qt/DiveListNotifier.h deleted file mode 100644 index fc7115a99..000000000 --- a/core/subsurface-qt/DiveListNotifier.h +++ /dev/null @@ -1,208 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 - -// The DiveListNotifier emits signals when the dive-list changes (dives/trips/divesites created/deleted/moved/edited) - -#ifndef DIVELISTNOTIFIER_H -#define DIVELISTNOTIFIER_H - -#include "core/dive.h" - -#include - -// Dive and trip fields that can be edited. Use bit fields so that we can pass multiple fields at once. -// Provides an inlined flag-based constructur because sadly C-style designated initializers are only supported since C++20. -struct DiveField { - // Note: using int instead of the more natural bool, because gcc produces significantly worse code with - // bool. clang, on the other hand, does fine. - unsigned int nr : 1; - unsigned int datetime : 1; - unsigned int depth : 1; - unsigned int duration : 1; - unsigned int air_temp : 1; - unsigned int water_temp : 1; - unsigned int atm_press : 1; - unsigned int divesite : 1; - unsigned int divemaster : 1; - unsigned int buddy : 1; - unsigned int rating : 1; - unsigned int visibility : 1; - unsigned int wavesize : 1; - unsigned int current : 1; - unsigned int surge : 1; - unsigned int chill : 1; - unsigned int suit : 1; - unsigned int tags : 1; - unsigned int mode : 1; - unsigned int notes : 1; - unsigned int salinity : 1; - enum Flags { - NONE = 0, - NR = 1 << 0, - DATETIME = 1 << 1, - DEPTH = 1 << 2, - DURATION = 1 << 3, - AIR_TEMP = 1 << 4, - WATER_TEMP = 1 << 5, - ATM_PRESS = 1 << 6, - DIVESITE = 1 << 7, - DIVEMASTER = 1 << 8, - BUDDY = 1 << 9, - RATING = 1 << 10, - VISIBILITY = 1 << 11, - WAVESIZE = 1 << 12, - CURRENT = 1 << 13, - SURGE = 1 << 14, - CHILL = 1 << 15, - SUIT = 1 << 16, - TAGS = 1 << 17, - MODE = 1 << 18, - NOTES = 1 << 19, - SALINITY = 1 << 20 - }; - DiveField(int flags); -}; -struct TripField { - unsigned int location : 1; - unsigned int notes : 1; - enum Flags { - NONE = 0, - LOCATION = 1 << 0, - NOTES = 1 << 1 - }; - TripField(int flags); -}; - -class DiveListNotifier : public QObject { - Q_OBJECT -signals: - // Note that there are no signals for trips being added and created - // because these events never happen without a dive being added, removed or moved. - // The dives are always sorted according to the dives_less_than() function of the core. - void divesAdded(dive_trip *trip, bool addTrip, const QVector &dives); - void divesDeleted(dive_trip *trip, bool deleteTrip, const QVector &dives); - void divesMovedBetweenTrips(dive_trip *from, dive_trip *to, bool deleteFrom, bool createTo, const QVector &dives); - void divesChanged(const QVector &dives, DiveField field); - void divesTimeChanged(timestamp_t delta, const QVector &dives); - - void cylindersReset(const QVector &dives); - void weightsystemsReset(const QVector &dives); - void weightAdded(dive *d, int pos); - void weightRemoved(dive *d, int pos); - void weightEdited(dive *d, int pos); - - // Trip edited signal - void tripChanged(dive_trip *trip, TripField field); - - // Selection changes - void divesSelected(const QVector &dives, dive *current); - - // Dive site signals. Add and delete events are sent per dive site and - // provide an index into the global dive site table. - void diveSiteAdded(dive_site *ds, int idx); - void diveSiteDeleted(dive_site *ds, int idx); - void diveSiteDiveCountChanged(dive_site *ds); - void diveSiteChanged(dive_site *ds, int field); // field according to LocationInformationModel - void diveSiteDivesChanged(dive_site *ds); // The dives associated with that site changed - - // Filter-related signals - void numShownChanged(); - void filterReset(); - - // This signal is emited every time a command is executed. - // This is used to hide an old multi-dives-edited warning message. - // This is necessary, so that the user can't click on the "undo" button and undo - // an unrelated command. - void commandExecuted(); -public: - // Desktop uses the QTreeView class to present the list of dives. The layout - // of this class gives us a very fundamental problem, as we can not easily - // distinguish between user-initiated changes of the selection and changes - // that are due to actions of the Command-classes. To solve this problem, - // the frontend can use this function to query whether a dive list-modifying - // command is currently executed. If this function returns true, the - // frontend is supposed to not modify the selection. - bool inCommand() const; - - // The following class and function are used by divelist-modifying commands - // to signal that they are in-flight. If the returned object goes out of scope, - // the command-in-flight status is reset to its previous value. Thus, the - // function can be called recursively. - class InCommandMarker { - DiveListNotifier ¬ifier; - bool oldValue; - InCommandMarker(DiveListNotifier &); - friend DiveListNotifier; - public: - ~InCommandMarker(); - }; - - // Usage: - // void doWork() - // { - // auto marker = diveListNotifier.enterCommand(); - // ... do work ... - // } - InCommandMarker enterCommand(); -private: - friend InCommandMarker; - bool commandExecuting; -}; - -// The DiveListNotifier class has only trivial state. -// We can simply define it as a global object. -extern DiveListNotifier diveListNotifier; - -// InCommandMarker is so trivial that the functions can be inlined. -// TODO: perhaps move this into own header-file. -inline DiveListNotifier::InCommandMarker::InCommandMarker(DiveListNotifier ¬ifierIn) : notifier(notifierIn), - oldValue(notifier.commandExecuting) -{ - notifier.commandExecuting = true; -} - -inline DiveListNotifier::InCommandMarker::~InCommandMarker() -{ - notifier.commandExecuting = oldValue; -} - -inline bool DiveListNotifier::inCommand() const -{ - return commandExecuting; -} - -inline DiveListNotifier::InCommandMarker DiveListNotifier::enterCommand() -{ - return InCommandMarker(*this); -} - -inline DiveField::DiveField(int flags) : - nr((flags & NR) != 0), - datetime((flags & DATETIME) != 0), - depth((flags & DEPTH) != 0), - duration((flags & DURATION) != 0), - air_temp((flags & AIR_TEMP) != 0), - water_temp((flags & WATER_TEMP) != 0), - atm_press((flags & ATM_PRESS) != 0), - divesite((flags & DIVESITE) != 0), - divemaster((flags & DIVEMASTER) != 0), - buddy((flags & BUDDY) != 0), - rating((flags & RATING) != 0), - visibility((flags & VISIBILITY) != 0), - wavesize((flags & WAVESIZE) != 0), - current((flags & CURRENT) != 0), - surge((flags & SURGE) != 0), - chill((flags & CHILL) != 0), - suit((flags & SUIT) != 0), - tags((flags & TAGS) != 0), - mode((flags & MODE) != 0), - notes((flags & NOTES) != 0), - salinity((flags & SALINITY) != 0) -{ -} - -inline TripField::TripField(int flags) : - location((flags & LOCATION) != 0), - notes((flags & NOTES) != 0) -{ -} -#endif diff --git a/core/subsurface-qt/DiveObjectHelper.cpp b/core/subsurface-qt/DiveObjectHelper.cpp deleted file mode 100644 index 606614b05..000000000 --- a/core/subsurface-qt/DiveObjectHelper.cpp +++ /dev/null @@ -1,330 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -#include "DiveObjectHelper.h" - -#include -#include - -#include "core/qthelper.h" -#include "core/divesite.h" -#include "core/trip.h" -#include "core/subsurface-string.h" -#include "qt-models/tankinfomodel.h" - -#if defined(DEBUG_DOH) -#include -#include -#include -static int callCounter = 0; -#endif /* defined(DEBUG_DOH) */ - - -enum returnPressureSelector {START_PRESSURE, END_PRESSURE}; - -static QString getFormattedWeight(const struct dive *dive, int idx) -{ - const weightsystem_t *weight = &dive->weightsystems.weightsystems[idx]; - if (!weight->description) - return QString(); - QString fmt = QString(weight->description); - fmt += ", " + get_weight_string(weight->weight, true); - return fmt; -} - -static QString getFormattedCylinder(const struct dive *dive, int idx) -{ - const cylinder_t *cyl = get_cylinder(dive, idx); - const char *desc = cyl->type.description; - if (!desc && idx > 0) - return QString(); - QString fmt = desc ? QString(desc) : gettextFromC::tr("unknown"); - fmt += ", " + get_volume_string(cyl->type.size, true); - fmt += ", " + get_pressure_string(cyl->type.workingpressure, true); - fmt += ", " + get_pressure_string(cyl->start, false) + " - " + get_pressure_string(cyl->end, true); - fmt += ", " + get_gas_string(cyl->gasmix); - return fmt; -} - -static QString getPressures(const struct dive *dive, int i, enum returnPressureSelector ret) -{ - const cylinder_t *cyl = get_cylinder(dive, i); - QString fmt; - if (ret == START_PRESSURE) { - if (cyl->start.mbar) - fmt = get_pressure_string(cyl->start, true); - else if (cyl->sample_start.mbar) - fmt = get_pressure_string(cyl->sample_start, true); - } - if (ret == END_PRESSURE) { - if (cyl->end.mbar) - fmt = get_pressure_string(cyl->end, true); - else if(cyl->sample_end.mbar) - fmt = get_pressure_string(cyl->sample_end, true); - } - return fmt; -} - -QString format_gps_decimal(const dive *d) -{ - bool savep = prefs.coordinates_traditional; - - prefs.coordinates_traditional = false; - QString val = d->dive_site ? printGPSCoords(&d->dive_site->location) : QString(); - prefs.coordinates_traditional = savep; - return val; -} - -QString formatNotes(const dive *d) -{ - QString tmp = d->notes ? QString::fromUtf8(d->notes) : QString(); - if (is_dc_planner(&d->dc)) { - QTextDocument notes; - #define _NOTES_BR "\n" - tmp.replace("", "" _NOTES_BR) - .replace("
", "
" _NOTES_BR) - .replace("
", "
" _NOTES_BR) - .replace("
", "
" _NOTES_BR) - .replace("", "" _NOTES_BR) - .replace("", "" _NOTES_BR); - notes.setHtml(tmp); - tmp = notes.toPlainText(); - tmp.replace(_NOTES_BR, "
"); - #undef _NOTES_BR - } else { - tmp.replace("\n", "
"); - } - return tmp; -} - -static QString formatGas(const dive *d) -{ - /*WARNING: here should be the gastlist, returned - * from the get_gas_string function or this is correct? - */ - QString gas, gases; - for (int i = 0; i < d->cylinders.nr; i++) { - if (!is_cylinder_used(d, i)) - continue; - gas = get_cylinder(d, i)->type.description; - if (!gas.isEmpty()) - gas += QChar(' '); - gas += gasname(get_cylinder(d, i)->gasmix); - // if has a description and if such gas is not already present - if (!gas.isEmpty() && gases.indexOf(gas) == -1) { - if (!gases.isEmpty()) - gases += QString(" / "); - gases += gas; - } - } - return gases; -} - -QString formatSac(const dive *d) -{ - if (!d->sac) - return QString(); - const char *unit; - int decimal; - double value = get_volume_units(d->sac, &decimal, &unit); - return QString::number(value, 'f', decimal).append(unit); -} - -static QString formatWeightList(const dive *d) -{ - QString weights; - for (int i = 0; i < d->weightsystems.nr; i++) { - QString w = getFormattedWeight(d, i); - if (w.isEmpty()) - continue; - weights += w + "; "; - } - return weights; -} - -static QStringList formatWeights(const dive *d) -{ - QStringList weights; - for (int i = 0; i < d->weightsystems.nr; i++) { - QString w = getFormattedWeight(d, i); - if (w.isEmpty()) - continue; - weights << w; - } - return weights; -} - -QStringList formatCylinders(const dive *d) -{ - QStringList cylinders; - for (int i = 0; i < d->cylinders.nr; i++) { - QString cyl = getFormattedCylinder(d, i); - cylinders << cyl; - } - return cylinders; -} - -static QVector makeCylinderObjects(const dive *d) -{ - QVector res; - for (int i = 0; i < d->cylinders.nr; i++) { - //Don't add blank cylinders, only those that have been defined. - if (get_cylinder(d, i)->type.description) - res.append(CylinderObjectHelper(get_cylinder(d, i))); // no emplace for QVector. :( - } - return res; -} - -QStringList formatGetCylinder(const dive *d) -{ - QStringList getCylinder; - for (int i = 0; i < d->cylinders.nr; i++) { - if (is_cylinder_used(d, i)) - getCylinder << get_cylinder(d, i)->type.description; - } - return getCylinder; -} - -QStringList getStartPressure(const dive *d) -{ - QStringList startPressure; - for (int i = 0; i < d->cylinders.nr; i++) { - if (is_cylinder_used(d, i)) - startPressure << getPressures(d, i, START_PRESSURE); - } - return startPressure; -} - -QStringList getEndPressure(const dive *d) -{ - QStringList endPressure; - for (int i = 0; i < d->cylinders.nr; i++) { - if (is_cylinder_used(d, i)) - endPressure << getPressures(d, i, END_PRESSURE); - } - return endPressure; -} - -QStringList getFirstGas(const dive *d) -{ - QStringList gas; - for (int i = 0; i < d->cylinders.nr; i++) { - if (is_cylinder_used(d, i)) - gas << get_gas_string(get_cylinder(d, i)->gasmix); - } - return gas; -} - -// Add string to sorted QStringList, if it doesn't already exist and -// it isn't the empty string. -static void addStringToSortedList(QStringList &l, const char *s) -{ - if (empty_string(s)) - return; - - // Do a binary search for the string. lower_bound() returns an iterator - // to either the searched-for element or the next higher element if it - // doesn't exist. - QString qs(s); - auto it = std::lower_bound(l.begin(), l.end(), qs); // TODO: use locale-aware sorting - if (it != l.end() && *it == s) - return; - - // Add new string at sorted position - l.insert(it, s); -} - -QStringList getFullCylinderList() -{ - QStringList cylinders; - struct dive *d; - int i = 0; - for_each_dive (i, d) { - for (int j = 0; j < d->cylinders.nr; j++) - addStringToSortedList(cylinders, get_cylinder(d, j)->type.description); - } - - for (int ti = 0; ti < MAX_TANK_INFO; ti++) - addStringToSortedList(cylinders, tank_info[ti].name); - - return cylinders; -} - -// Qt's metatype system insists on generating a default constructed object, even if that makes no sense. -DiveObjectHelper::DiveObjectHelper() -{ -} - -DiveObjectHelper::DiveObjectHelper(const struct dive *d) : - number(d->number), - id(d->id), - rating(d->rating), - visibility(d->visibility), - timestamp(d->when), - location(get_dive_location(d) ? QString::fromUtf8(get_dive_location(d)) : QString()), - gps(d->dive_site ? printGPSCoords(&d->dive_site->location) : QString()), - gps_decimal(format_gps_decimal(d)), - dive_site(QVariant::fromValue(d->dive_site)), - duration(get_dive_duration_string(d->duration.seconds, gettextFromC::tr("h"), gettextFromC::tr("min"))), - noDive(d->duration.seconds == 0 && d->dc.duration.seconds == 0), - depth(get_depth_string(d->dc.maxdepth.mm, true, true)), - divemaster(d->divemaster ? d->divemaster : QString()), - buddy(d->buddy ? d->buddy : QString()), - airTemp(get_temperature_string(d->airtemp, true)), - waterTemp(get_temperature_string(d->watertemp, true)), - notes(formatNotes(d)), - tags(get_taglist_string(d->tag_list)), - gas(formatGas(d)), - sac(formatSac(d)), - weightList(formatWeightList(d)), - weights(formatWeights(d)), - singleWeight(d->weightsystems.nr <= 1), - suit(d->suit ? d->suit : QString()), - cylinders(formatCylinders(d)), - maxcns(d->maxcns), - otu(d->otu), - sumWeight(get_weight_string(weight_t { total_weight(d) }, true)), - getCylinder(formatGetCylinder(d)), - startPressure(getStartPressure(d)), - endPressure(getEndPressure(d)), - firstGas(getFirstGas(d)) -{ -#if defined(DEBUG_DOH) - void *array[4]; - size_t size; - - // get void*'s for all entries on the stack - size = backtrace(array, 4); - - // print out all the frames to stderr - fprintf(stderr, "\n\nCalling DiveObjectHelper constructor for dive %d - call #%d\n", d->number, ++callCounter); - backtrace_symbols_fd(array, size, STDERR_FILENO); -#endif /* defined(DEBUG_DOH) */ -} - -DiveObjectHelperGrantlee::DiveObjectHelperGrantlee() -{ -} - -DiveObjectHelperGrantlee::DiveObjectHelperGrantlee(const struct dive *d) : - DiveObjectHelper(d), - cylinderObjects(makeCylinderObjects(d)) -{ -} - -QString DiveObjectHelper::date() const -{ - QDateTime localTime = QDateTime::fromMSecsSinceEpoch(1000 * timestamp, Qt::UTC); - localTime.setTimeSpec(Qt::UTC); - return localTime.date().toString(prefs.date_format_short); -} - -QString DiveObjectHelper::time() const -{ - QDateTime localTime = QDateTime::fromMSecsSinceEpoch(1000 * timestamp, Qt::UTC); - localTime.setTimeSpec(Qt::UTC); - return localTime.time().toString(prefs.time_format); -} - -QStringList DiveObjectHelper::cylinderList() const -{ - return getFullCylinderList(); -} diff --git a/core/subsurface-qt/DiveObjectHelper.h b/core/subsurface-qt/DiveObjectHelper.h deleted file mode 100644 index 639dae696..000000000 --- a/core/subsurface-qt/DiveObjectHelper.h +++ /dev/null @@ -1,105 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -#ifndef DIVE_QOBJECT_H -#define DIVE_QOBJECT_H - -#include "CylinderObjectHelper.h" -#include -#include -#include -#include -#include - -class DiveObjectHelper { - Q_GADGET - Q_PROPERTY(int number MEMBER number CONSTANT) - Q_PROPERTY(int id MEMBER id CONSTANT) - Q_PROPERTY(int rating MEMBER rating CONSTANT) - Q_PROPERTY(int visibility MEMBER visibility CONSTANT) - Q_PROPERTY(QString date READ date CONSTANT) - Q_PROPERTY(QString time READ time CONSTANT) - Q_PROPERTY(int timestamp MEMBER timestamp CONSTANT) - Q_PROPERTY(QString location MEMBER location CONSTANT) - Q_PROPERTY(QString gps MEMBER gps CONSTANT) - Q_PROPERTY(QString gps_decimal MEMBER gps_decimal CONSTANT) - Q_PROPERTY(QVariant dive_site MEMBER dive_site CONSTANT) - Q_PROPERTY(QString duration MEMBER duration CONSTANT) - Q_PROPERTY(bool noDive MEMBER noDive CONSTANT) - Q_PROPERTY(QString depth MEMBER depth CONSTANT) - Q_PROPERTY(QString divemaster MEMBER divemaster CONSTANT) - Q_PROPERTY(QString buddy MEMBER buddy CONSTANT) - Q_PROPERTY(QString airTemp MEMBER airTemp CONSTANT) - Q_PROPERTY(QString waterTemp MEMBER waterTemp CONSTANT) - Q_PROPERTY(QString notes MEMBER notes CONSTANT) - Q_PROPERTY(QString tags MEMBER tags CONSTANT) - Q_PROPERTY(QString gas MEMBER gas CONSTANT) - Q_PROPERTY(QString sac MEMBER sac CONSTANT) - Q_PROPERTY(QString weightList MEMBER weightList CONSTANT) - Q_PROPERTY(QStringList weights MEMBER weights CONSTANT) - Q_PROPERTY(bool singleWeight MEMBER singleWeight CONSTANT) - Q_PROPERTY(QString suit MEMBER suit CONSTANT) - Q_PROPERTY(QStringList cylinderList READ cylinderList CONSTANT) - Q_PROPERTY(QStringList cylinders MEMBER cylinders CONSTANT) - Q_PROPERTY(int maxcns MEMBER maxcns CONSTANT) - Q_PROPERTY(int otu MEMBER otu CONSTANT) - Q_PROPERTY(QString sumWeight MEMBER sumWeight CONSTANT) - Q_PROPERTY(QStringList getCylinder MEMBER getCylinder CONSTANT) - Q_PROPERTY(QStringList startPressure MEMBER startPressure CONSTANT) - Q_PROPERTY(QStringList endPressure MEMBER endPressure CONSTANT) - Q_PROPERTY(QStringList firstGas MEMBER firstGas CONSTANT) -public: - DiveObjectHelper(); // This is only to be used by Qt's metatype system! - DiveObjectHelper(const struct dive *dive); - int number; - int id; - int rating; - int visibility; - QString date() const; - timestamp_t timestamp; - QString time() const; - QString location; - QString gps; - QString gps_decimal; - QVariant dive_site; - QString duration; - bool noDive; - QString depth; - QString divemaster; - QString buddy; - QString airTemp; - QString waterTemp; - QString notes; - QString tags; - QString gas; - QString sac; - QString weightList; - QStringList weights; - bool singleWeight; - QString suit; - QStringList cylinderList() const; - QStringList cylinders; - int maxcns; - int otu; - QString sumWeight; - QStringList getCylinder; - QStringList startPressure; - QStringList endPressure; - QStringList firstGas; -}; - -// This is an extended version of DiveObjectHelper that also keeps track of cylinder data. -// It is used by grantlee to display structured cylinder data. -// Note: this grantlee feature is undocumented. If there turns out to be no users, we might -// want to remove this class. -class DiveObjectHelperGrantlee : public DiveObjectHelper { - Q_GADGET - Q_PROPERTY(QVector cylinderObjects MEMBER cylinderObjects CONSTANT) -public: - DiveObjectHelperGrantlee(); - DiveObjectHelperGrantlee(const struct dive *dive); - QVector cylinderObjects; -}; - -Q_DECLARE_METATYPE(DiveObjectHelper) -Q_DECLARE_METATYPE(DiveObjectHelperGrantlee) - -#endif diff --git a/core/subsurface-qt/cylinderobjecthelper.cpp b/core/subsurface-qt/cylinderobjecthelper.cpp new file mode 100644 index 000000000..897746952 --- /dev/null +++ b/core/subsurface-qt/cylinderobjecthelper.cpp @@ -0,0 +1,18 @@ +// SPDX-License-Identifier: GPL-2.0 +#include "cylinderobjecthelper.h" +#include "../qthelper.h" + +static QString EMPTY_CYLINDER_STRING = QStringLiteral(""); +CylinderObjectHelper::CylinderObjectHelper(const cylinder_t *cylinder) +{ + if (!cylinder) + return; + + description = cylinder->type.description ? cylinder->type.description: + EMPTY_CYLINDER_STRING; + size = get_volume_string(cylinder->type.size, true); + workingPressure = get_pressure_string(cylinder->type.workingpressure, true); + startPressure = get_pressure_string(cylinder->start, true); + endPressure = get_pressure_string(cylinder->end, true); + gasMix = get_gas_string(cylinder->gasmix); +} diff --git a/core/subsurface-qt/cylinderobjecthelper.h b/core/subsurface-qt/cylinderobjecthelper.h new file mode 100644 index 000000000..e1e2490e4 --- /dev/null +++ b/core/subsurface-qt/cylinderobjecthelper.h @@ -0,0 +1,29 @@ +// SPDX-License-Identifier: GPL-2.0 +#ifndef CYLINDER_QOBJECT_H +#define CYLINDER_QOBJECT_H + +#include "../equipment.h" +#include +#include + +class CylinderObjectHelper { + Q_GADGET + Q_PROPERTY(QString description MEMBER description CONSTANT) + Q_PROPERTY(QString size MEMBER size CONSTANT) + Q_PROPERTY(QString workingPressure MEMBER workingPressure CONSTANT) + Q_PROPERTY(QString startPressure MEMBER startPressure CONSTANT) + Q_PROPERTY(QString endPressure MEMBER endPressure CONSTANT) + Q_PROPERTY(QString gasMix MEMBER gasMix CONSTANT) +public: + CylinderObjectHelper(const cylinder_t *cylinder = NULL); + QString description; + QString size; + QString workingPressure; + QString startPressure; + QString endPressure; + QString gasMix; +}; + +Q_DECLARE_METATYPE(CylinderObjectHelper) + +#endif diff --git a/core/subsurface-qt/divelistnotifier.cpp b/core/subsurface-qt/divelistnotifier.cpp new file mode 100644 index 000000000..8753c79a4 --- /dev/null +++ b/core/subsurface-qt/divelistnotifier.cpp @@ -0,0 +1,4 @@ +// SPDX-License-Identifier: GPL-2.0 +#include "divelistnotifier.h" + +DiveListNotifier diveListNotifier; diff --git a/core/subsurface-qt/divelistnotifier.h b/core/subsurface-qt/divelistnotifier.h new file mode 100644 index 000000000..fc7115a99 --- /dev/null +++ b/core/subsurface-qt/divelistnotifier.h @@ -0,0 +1,208 @@ +// SPDX-License-Identifier: GPL-2.0 + +// The DiveListNotifier emits signals when the dive-list changes (dives/trips/divesites created/deleted/moved/edited) + +#ifndef DIVELISTNOTIFIER_H +#define DIVELISTNOTIFIER_H + +#include "core/dive.h" + +#include + +// Dive and trip fields that can be edited. Use bit fields so that we can pass multiple fields at once. +// Provides an inlined flag-based constructur because sadly C-style designated initializers are only supported since C++20. +struct DiveField { + // Note: using int instead of the more natural bool, because gcc produces significantly worse code with + // bool. clang, on the other hand, does fine. + unsigned int nr : 1; + unsigned int datetime : 1; + unsigned int depth : 1; + unsigned int duration : 1; + unsigned int air_temp : 1; + unsigned int water_temp : 1; + unsigned int atm_press : 1; + unsigned int divesite : 1; + unsigned int divemaster : 1; + unsigned int buddy : 1; + unsigned int rating : 1; + unsigned int visibility : 1; + unsigned int wavesize : 1; + unsigned int current : 1; + unsigned int surge : 1; + unsigned int chill : 1; + unsigned int suit : 1; + unsigned int tags : 1; + unsigned int mode : 1; + unsigned int notes : 1; + unsigned int salinity : 1; + enum Flags { + NONE = 0, + NR = 1 << 0, + DATETIME = 1 << 1, + DEPTH = 1 << 2, + DURATION = 1 << 3, + AIR_TEMP = 1 << 4, + WATER_TEMP = 1 << 5, + ATM_PRESS = 1 << 6, + DIVESITE = 1 << 7, + DIVEMASTER = 1 << 8, + BUDDY = 1 << 9, + RATING = 1 << 10, + VISIBILITY = 1 << 11, + WAVESIZE = 1 << 12, + CURRENT = 1 << 13, + SURGE = 1 << 14, + CHILL = 1 << 15, + SUIT = 1 << 16, + TAGS = 1 << 17, + MODE = 1 << 18, + NOTES = 1 << 19, + SALINITY = 1 << 20 + }; + DiveField(int flags); +}; +struct TripField { + unsigned int location : 1; + unsigned int notes : 1; + enum Flags { + NONE = 0, + LOCATION = 1 << 0, + NOTES = 1 << 1 + }; + TripField(int flags); +}; + +class DiveListNotifier : public QObject { + Q_OBJECT +signals: + // Note that there are no signals for trips being added and created + // because these events never happen without a dive being added, removed or moved. + // The dives are always sorted according to the dives_less_than() function of the core. + void divesAdded(dive_trip *trip, bool addTrip, const QVector &dives); + void divesDeleted(dive_trip *trip, bool deleteTrip, const QVector &dives); + void divesMovedBetweenTrips(dive_trip *from, dive_trip *to, bool deleteFrom, bool createTo, const QVector &dives); + void divesChanged(const QVector &dives, DiveField field); + void divesTimeChanged(timestamp_t delta, const QVector &dives); + + void cylindersReset(const QVector &dives); + void weightsystemsReset(const QVector &dives); + void weightAdded(dive *d, int pos); + void weightRemoved(dive *d, int pos); + void weightEdited(dive *d, int pos); + + // Trip edited signal + void tripChanged(dive_trip *trip, TripField field); + + // Selection changes + void divesSelected(const QVector &dives, dive *current); + + // Dive site signals. Add and delete events are sent per dive site and + // provide an index into the global dive site table. + void diveSiteAdded(dive_site *ds, int idx); + void diveSiteDeleted(dive_site *ds, int idx); + void diveSiteDiveCountChanged(dive_site *ds); + void diveSiteChanged(dive_site *ds, int field); // field according to LocationInformationModel + void diveSiteDivesChanged(dive_site *ds); // The dives associated with that site changed + + // Filter-related signals + void numShownChanged(); + void filterReset(); + + // This signal is emited every time a command is executed. + // This is used to hide an old multi-dives-edited warning message. + // This is necessary, so that the user can't click on the "undo" button and undo + // an unrelated command. + void commandExecuted(); +public: + // Desktop uses the QTreeView class to present the list of dives. The layout + // of this class gives us a very fundamental problem, as we can not easily + // distinguish between user-initiated changes of the selection and changes + // that are due to actions of the Command-classes. To solve this problem, + // the frontend can use this function to query whether a dive list-modifying + // command is currently executed. If this function returns true, the + // frontend is supposed to not modify the selection. + bool inCommand() const; + + // The following class and function are used by divelist-modifying commands + // to signal that they are in-flight. If the returned object goes out of scope, + // the command-in-flight status is reset to its previous value. Thus, the + // function can be called recursively. + class InCommandMarker { + DiveListNotifier ¬ifier; + bool oldValue; + InCommandMarker(DiveListNotifier &); + friend DiveListNotifier; + public: + ~InCommandMarker(); + }; + + // Usage: + // void doWork() + // { + // auto marker = diveListNotifier.enterCommand(); + // ... do work ... + // } + InCommandMarker enterCommand(); +private: + friend InCommandMarker; + bool commandExecuting; +}; + +// The DiveListNotifier class has only trivial state. +// We can simply define it as a global object. +extern DiveListNotifier diveListNotifier; + +// InCommandMarker is so trivial that the functions can be inlined. +// TODO: perhaps move this into own header-file. +inline DiveListNotifier::InCommandMarker::InCommandMarker(DiveListNotifier ¬ifierIn) : notifier(notifierIn), + oldValue(notifier.commandExecuting) +{ + notifier.commandExecuting = true; +} + +inline DiveListNotifier::InCommandMarker::~InCommandMarker() +{ + notifier.commandExecuting = oldValue; +} + +inline bool DiveListNotifier::inCommand() const +{ + return commandExecuting; +} + +inline DiveListNotifier::InCommandMarker DiveListNotifier::enterCommand() +{ + return InCommandMarker(*this); +} + +inline DiveField::DiveField(int flags) : + nr((flags & NR) != 0), + datetime((flags & DATETIME) != 0), + depth((flags & DEPTH) != 0), + duration((flags & DURATION) != 0), + air_temp((flags & AIR_TEMP) != 0), + water_temp((flags & WATER_TEMP) != 0), + atm_press((flags & ATM_PRESS) != 0), + divesite((flags & DIVESITE) != 0), + divemaster((flags & DIVEMASTER) != 0), + buddy((flags & BUDDY) != 0), + rating((flags & RATING) != 0), + visibility((flags & VISIBILITY) != 0), + wavesize((flags & WAVESIZE) != 0), + current((flags & CURRENT) != 0), + surge((flags & SURGE) != 0), + chill((flags & CHILL) != 0), + suit((flags & SUIT) != 0), + tags((flags & TAGS) != 0), + mode((flags & MODE) != 0), + notes((flags & NOTES) != 0), + salinity((flags & SALINITY) != 0) +{ +} + +inline TripField::TripField(int flags) : + location((flags & LOCATION) != 0), + notes((flags & NOTES) != 0) +{ +} +#endif diff --git a/core/subsurface-qt/diveobjecthelper.cpp b/core/subsurface-qt/diveobjecthelper.cpp new file mode 100644 index 000000000..aec5984a8 --- /dev/null +++ b/core/subsurface-qt/diveobjecthelper.cpp @@ -0,0 +1,330 @@ +// SPDX-License-Identifier: GPL-2.0 +#include "diveobjecthelper.h" + +#include +#include + +#include "core/qthelper.h" +#include "core/divesite.h" +#include "core/trip.h" +#include "core/subsurface-string.h" +#include "qt-models/tankinfomodel.h" + +#if defined(DEBUG_DOH) +#include +#include +#include +static int callCounter = 0; +#endif /* defined(DEBUG_DOH) */ + + +enum returnPressureSelector {START_PRESSURE, END_PRESSURE}; + +static QString getFormattedWeight(const struct dive *dive, int idx) +{ + const weightsystem_t *weight = &dive->weightsystems.weightsystems[idx]; + if (!weight->description) + return QString(); + QString fmt = QString(weight->description); + fmt += ", " + get_weight_string(weight->weight, true); + return fmt; +} + +static QString getFormattedCylinder(const struct dive *dive, int idx) +{ + const cylinder_t *cyl = get_cylinder(dive, idx); + const char *desc = cyl->type.description; + if (!desc && idx > 0) + return QString(); + QString fmt = desc ? QString(desc) : gettextFromC::tr("unknown"); + fmt += ", " + get_volume_string(cyl->type.size, true); + fmt += ", " + get_pressure_string(cyl->type.workingpressure, true); + fmt += ", " + get_pressure_string(cyl->start, false) + " - " + get_pressure_string(cyl->end, true); + fmt += ", " + get_gas_string(cyl->gasmix); + return fmt; +} + +static QString getPressures(const struct dive *dive, int i, enum returnPressureSelector ret) +{ + const cylinder_t *cyl = get_cylinder(dive, i); + QString fmt; + if (ret == START_PRESSURE) { + if (cyl->start.mbar) + fmt = get_pressure_string(cyl->start, true); + else if (cyl->sample_start.mbar) + fmt = get_pressure_string(cyl->sample_start, true); + } + if (ret == END_PRESSURE) { + if (cyl->end.mbar) + fmt = get_pressure_string(cyl->end, true); + else if(cyl->sample_end.mbar) + fmt = get_pressure_string(cyl->sample_end, true); + } + return fmt; +} + +QString format_gps_decimal(const dive *d) +{ + bool savep = prefs.coordinates_traditional; + + prefs.coordinates_traditional = false; + QString val = d->dive_site ? printGPSCoords(&d->dive_site->location) : QString(); + prefs.coordinates_traditional = savep; + return val; +} + +QString formatNotes(const dive *d) +{ + QString tmp = d->notes ? QString::fromUtf8(d->notes) : QString(); + if (is_dc_planner(&d->dc)) { + QTextDocument notes; + #define _NOTES_BR "\n" + tmp.replace("", "" _NOTES_BR) + .replace("
", "
" _NOTES_BR) + .replace("
", "
" _NOTES_BR) + .replace("
", "
" _NOTES_BR) + .replace("", "" _NOTES_BR) + .replace("", "" _NOTES_BR); + notes.setHtml(tmp); + tmp = notes.toPlainText(); + tmp.replace(_NOTES_BR, "
"); + #undef _NOTES_BR + } else { + tmp.replace("\n", "
"); + } + return tmp; +} + +static QString formatGas(const dive *d) +{ + /*WARNING: here should be the gastlist, returned + * from the get_gas_string function or this is correct? + */ + QString gas, gases; + for (int i = 0; i < d->cylinders.nr; i++) { + if (!is_cylinder_used(d, i)) + continue; + gas = get_cylinder(d, i)->type.description; + if (!gas.isEmpty()) + gas += QChar(' '); + gas += gasname(get_cylinder(d, i)->gasmix); + // if has a description and if such gas is not already present + if (!gas.isEmpty() && gases.indexOf(gas) == -1) { + if (!gases.isEmpty()) + gases += QString(" / "); + gases += gas; + } + } + return gases; +} + +QString formatSac(const dive *d) +{ + if (!d->sac) + return QString(); + const char *unit; + int decimal; + double value = get_volume_units(d->sac, &decimal, &unit); + return QString::number(value, 'f', decimal).append(unit); +} + +static QString formatWeightList(const dive *d) +{ + QString weights; + for (int i = 0; i < d->weightsystems.nr; i++) { + QString w = getFormattedWeight(d, i); + if (w.isEmpty()) + continue; + weights += w + "; "; + } + return weights; +} + +static QStringList formatWeights(const dive *d) +{ + QStringList weights; + for (int i = 0; i < d->weightsystems.nr; i++) { + QString w = getFormattedWeight(d, i); + if (w.isEmpty()) + continue; + weights << w; + } + return weights; +} + +QStringList formatCylinders(const dive *d) +{ + QStringList cylinders; + for (int i = 0; i < d->cylinders.nr; i++) { + QString cyl = getFormattedCylinder(d, i); + cylinders << cyl; + } + return cylinders; +} + +static QVector makeCylinderObjects(const dive *d) +{ + QVector res; + for (int i = 0; i < d->cylinders.nr; i++) { + //Don't add blank cylinders, only those that have been defined. + if (get_cylinder(d, i)->type.description) + res.append(CylinderObjectHelper(get_cylinder(d, i))); // no emplace for QVector. :( + } + return res; +} + +QStringList formatGetCylinder(const dive *d) +{ + QStringList getCylinder; + for (int i = 0; i < d->cylinders.nr; i++) { + if (is_cylinder_used(d, i)) + getCylinder << get_cylinder(d, i)->type.description; + } + return getCylinder; +} + +QStringList getStartPressure(const dive *d) +{ + QStringList startPressure; + for (int i = 0; i < d->cylinders.nr; i++) { + if (is_cylinder_used(d, i)) + startPressure << getPressures(d, i, START_PRESSURE); + } + return startPressure; +} + +QStringList getEndPressure(const dive *d) +{ + QStringList endPressure; + for (int i = 0; i < d->cylinders.nr; i++) { + if (is_cylinder_used(d, i)) + endPressure << getPressures(d, i, END_PRESSURE); + } + return endPressure; +} + +QStringList getFirstGas(const dive *d) +{ + QStringList gas; + for (int i = 0; i < d->cylinders.nr; i++) { + if (is_cylinder_used(d, i)) + gas << get_gas_string(get_cylinder(d, i)->gasmix); + } + return gas; +} + +// Add string to sorted QStringList, if it doesn't already exist and +// it isn't the empty string. +static void addStringToSortedList(QStringList &l, const char *s) +{ + if (empty_string(s)) + return; + + // Do a binary search for the string. lower_bound() returns an iterator + // to either the searched-for element or the next higher element if it + // doesn't exist. + QString qs(s); + auto it = std::lower_bound(l.begin(), l.end(), qs); // TODO: use locale-aware sorting + if (it != l.end() && *it == s) + return; + + // Add new string at sorted position + l.insert(it, s); +} + +QStringList getFullCylinderList() +{ + QStringList cylinders; + struct dive *d; + int i = 0; + for_each_dive (i, d) { + for (int j = 0; j < d->cylinders.nr; j++) + addStringToSortedList(cylinders, get_cylinder(d, j)->type.description); + } + + for (int ti = 0; ti < MAX_TANK_INFO; ti++) + addStringToSortedList(cylinders, tank_info[ti].name); + + return cylinders; +} + +// Qt's metatype system insists on generating a default constructed object, even if that makes no sense. +DiveObjectHelper::DiveObjectHelper() +{ +} + +DiveObjectHelper::DiveObjectHelper(const struct dive *d) : + number(d->number), + id(d->id), + rating(d->rating), + visibility(d->visibility), + timestamp(d->when), + location(get_dive_location(d) ? QString::fromUtf8(get_dive_location(d)) : QString()), + gps(d->dive_site ? printGPSCoords(&d->dive_site->location) : QString()), + gps_decimal(format_gps_decimal(d)), + dive_site(QVariant::fromValue(d->dive_site)), + duration(get_dive_duration_string(d->duration.seconds, gettextFromC::tr("h"), gettextFromC::tr("min"))), + noDive(d->duration.seconds == 0 && d->dc.duration.seconds == 0), + depth(get_depth_string(d->dc.maxdepth.mm, true, true)), + divemaster(d->divemaster ? d->divemaster : QString()), + buddy(d->buddy ? d->buddy : QString()), + airTemp(get_temperature_string(d->airtemp, true)), + waterTemp(get_temperature_string(d->watertemp, true)), + notes(formatNotes(d)), + tags(get_taglist_string(d->tag_list)), + gas(formatGas(d)), + sac(formatSac(d)), + weightList(formatWeightList(d)), + weights(formatWeights(d)), + singleWeight(d->weightsystems.nr <= 1), + suit(d->suit ? d->suit : QString()), + cylinders(formatCylinders(d)), + maxcns(d->maxcns), + otu(d->otu), + sumWeight(get_weight_string(weight_t { total_weight(d) }, true)), + getCylinder(formatGetCylinder(d)), + startPressure(getStartPressure(d)), + endPressure(getEndPressure(d)), + firstGas(getFirstGas(d)) +{ +#if defined(DEBUG_DOH) + void *array[4]; + size_t size; + + // get void*'s for all entries on the stack + size = backtrace(array, 4); + + // print out all the frames to stderr + fprintf(stderr, "\n\nCalling DiveObjectHelper constructor for dive %d - call #%d\n", d->number, ++callCounter); + backtrace_symbols_fd(array, size, STDERR_FILENO); +#endif /* defined(DEBUG_DOH) */ +} + +DiveObjectHelperGrantlee::DiveObjectHelperGrantlee() +{ +} + +DiveObjectHelperGrantlee::DiveObjectHelperGrantlee(const struct dive *d) : + DiveObjectHelper(d), + cylinderObjects(makeCylinderObjects(d)) +{ +} + +QString DiveObjectHelper::date() const +{ + QDateTime localTime = QDateTime::fromMSecsSinceEpoch(1000 * timestamp, Qt::UTC); + localTime.setTimeSpec(Qt::UTC); + return localTime.date().toString(prefs.date_format_short); +} + +QString DiveObjectHelper::time() const +{ + QDateTime localTime = QDateTime::fromMSecsSinceEpoch(1000 * timestamp, Qt::UTC); + localTime.setTimeSpec(Qt::UTC); + return localTime.time().toString(prefs.time_format); +} + +QStringList DiveObjectHelper::cylinderList() const +{ + return getFullCylinderList(); +} diff --git a/core/subsurface-qt/diveobjecthelper.h b/core/subsurface-qt/diveobjecthelper.h new file mode 100644 index 000000000..7723df0f4 --- /dev/null +++ b/core/subsurface-qt/diveobjecthelper.h @@ -0,0 +1,105 @@ +// SPDX-License-Identifier: GPL-2.0 +#ifndef DIVE_QOBJECT_H +#define DIVE_QOBJECT_H + +#include "cylinderobjecthelper.h" +#include +#include +#include +#include +#include + +class DiveObjectHelper { + Q_GADGET + Q_PROPERTY(int number MEMBER number CONSTANT) + Q_PROPERTY(int id MEMBER id CONSTANT) + Q_PROPERTY(int rating MEMBER rating CONSTANT) + Q_PROPERTY(int visibility MEMBER visibility CONSTANT) + Q_PROPERTY(QString date READ date CONSTANT) + Q_PROPERTY(QString time READ time CONSTANT) + Q_PROPERTY(int timestamp MEMBER timestamp CONSTANT) + Q_PROPERTY(QString location MEMBER location CONSTANT) + Q_PROPERTY(QString gps MEMBER gps CONSTANT) + Q_PROPERTY(QString gps_decimal MEMBER gps_decimal CONSTANT) + Q_PROPERTY(QVariant dive_site MEMBER dive_site CONSTANT) + Q_PROPERTY(QString duration MEMBER duration CONSTANT) + Q_PROPERTY(bool noDive MEMBER noDive CONSTANT) + Q_PROPERTY(QString depth MEMBER depth CONSTANT) + Q_PROPERTY(QString divemaster MEMBER divemaster CONSTANT) + Q_PROPERTY(QString buddy MEMBER buddy CONSTANT) + Q_PROPERTY(QString airTemp MEMBER airTemp CONSTANT) + Q_PROPERTY(QString waterTemp MEMBER waterTemp CONSTANT) + Q_PROPERTY(QString notes MEMBER notes CONSTANT) + Q_PROPERTY(QString tags MEMBER tags CONSTANT) + Q_PROPERTY(QString gas MEMBER gas CONSTANT) + Q_PROPERTY(QString sac MEMBER sac CONSTANT) + Q_PROPERTY(QString weightList MEMBER weightList CONSTANT) + Q_PROPERTY(QStringList weights MEMBER weights CONSTANT) + Q_PROPERTY(bool singleWeight MEMBER singleWeight CONSTANT) + Q_PROPERTY(QString suit MEMBER suit CONSTANT) + Q_PROPERTY(QStringList cylinderList READ cylinderList CONSTANT) + Q_PROPERTY(QStringList cylinders MEMBER cylinders CONSTANT) + Q_PROPERTY(int maxcns MEMBER maxcns CONSTANT) + Q_PROPERTY(int otu MEMBER otu CONSTANT) + Q_PROPERTY(QString sumWeight MEMBER sumWeight CONSTANT) + Q_PROPERTY(QStringList getCylinder MEMBER getCylinder CONSTANT) + Q_PROPERTY(QStringList startPressure MEMBER startPressure CONSTANT) + Q_PROPERTY(QStringList endPressure MEMBER endPressure CONSTANT) + Q_PROPERTY(QStringList firstGas MEMBER firstGas CONSTANT) +public: + DiveObjectHelper(); // This is only to be used by Qt's metatype system! + DiveObjectHelper(const struct dive *dive); + int number; + int id; + int rating; + int visibility; + QString date() const; + timestamp_t timestamp; + QString time() const; + QString location; + QString gps; + QString gps_decimal; + QVariant dive_site; + QString duration; + bool noDive; + QString depth; + QString divemaster; + QString buddy; + QString airTemp; + QString waterTemp; + QString notes; + QString tags; + QString gas; + QString sac; + QString weightList; + QStringList weights; + bool singleWeight; + QString suit; + QStringList cylinderList() const; + QStringList cylinders; + int maxcns; + int otu; + QString sumWeight; + QStringList getCylinder; + QStringList startPressure; + QStringList endPressure; + QStringList firstGas; +}; + +// This is an extended version of DiveObjectHelper that also keeps track of cylinder data. +// It is used by grantlee to display structured cylinder data. +// Note: this grantlee feature is undocumented. If there turns out to be no users, we might +// want to remove this class. +class DiveObjectHelperGrantlee : public DiveObjectHelper { + Q_GADGET + Q_PROPERTY(QVector cylinderObjects MEMBER cylinderObjects CONSTANT) +public: + DiveObjectHelperGrantlee(); + DiveObjectHelperGrantlee(const struct dive *dive); + QVector cylinderObjects; +}; + +Q_DECLARE_METATYPE(DiveObjectHelper) +Q_DECLARE_METATYPE(DiveObjectHelperGrantlee) + +#endif -- cgit v1.2.3-70-g09d2