// 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 = &dive->cylinders.cylinders[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 = &dive->cylinders.cylinders[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 = d->cylinders.cylinders[i].type.description; if (!gas.isEmpty()) gas += QChar(' '); gas += gasname(d->cylinders.cylinders[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 (d->cylinders.cylinders[i].type.description) res.append(CylinderObjectHelper(&d->cylinders.cylinders[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 << d->cylinders.cylinders[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(d->cylinders.cylinders[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, d->cylinders.cylinders[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(); }