2020-12-14 22:42:07 +01:00
|
|
|
#include "string-format.h"
|
|
|
|
#include "dive.h"
|
2024-06-07 10:25:09 +02:00
|
|
|
#include "divelist.h"
|
|
|
|
#include "divelog.h"
|
2020-12-14 22:42:07 +01:00
|
|
|
#include "divesite.h"
|
2024-02-19 09:04:14 +01:00
|
|
|
#include "event.h"
|
2023-03-17 08:21:47 +01:00
|
|
|
#include "format.h"
|
2020-12-14 22:42:07 +01:00
|
|
|
#include "qthelper.h"
|
2024-05-28 21:31:11 +02:00
|
|
|
#include "range.h"
|
2020-12-14 22:42:07 +01:00
|
|
|
#include "subsurface-string.h"
|
2021-12-16 23:09:39 +01:00
|
|
|
#include "trip.h"
|
2020-12-15 15:33:39 +01:00
|
|
|
#include <QDateTime>
|
2021-12-16 23:09:39 +01:00
|
|
|
#include <QLocale>
|
2020-12-14 22:42:07 +01:00
|
|
|
#include <QTextDocument>
|
|
|
|
|
|
|
|
enum returnPressureSelector { START_PRESSURE, END_PRESSURE };
|
2021-12-16 23:09:39 +01:00
|
|
|
static QLocale loc;
|
2020-12-14 22:42:07 +01:00
|
|
|
|
2024-05-28 21:31:11 +02:00
|
|
|
static QString getPressures(const cylinder_t &cyl, enum returnPressureSelector ret)
|
2020-12-14 22:42:07 +01:00
|
|
|
{
|
|
|
|
if (ret == START_PRESSURE) {
|
2024-05-28 21:31:11 +02:00
|
|
|
if (cyl.start.mbar)
|
|
|
|
return get_pressure_string(cyl.start, true);
|
|
|
|
else if (cyl.sample_start.mbar)
|
|
|
|
return get_pressure_string(cyl.sample_start, true);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
if (ret == END_PRESSURE) {
|
2024-05-28 21:31:11 +02:00
|
|
|
if (cyl.end.mbar)
|
|
|
|
return get_pressure_string(cyl.end, true);
|
|
|
|
else if (cyl.sample_end.mbar)
|
|
|
|
return get_pressure_string(cyl.sample_end, true);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
2024-05-28 21:31:11 +02:00
|
|
|
return QString();
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString formatNotes(const dive *d)
|
|
|
|
{
|
2024-05-29 20:40:18 +02:00
|
|
|
QString tmp = QString::fromStdString(d->notes);
|
2024-05-27 17:09:48 +02:00
|
|
|
if (is_dc_planner(&d->dcs[0])) {
|
2020-12-14 22:42:07 +01:00
|
|
|
QTextDocument notes;
|
|
|
|
#define _NOTES_BR "\n"
|
|
|
|
tmp.replace("<thead>", "<thead>" _NOTES_BR)
|
|
|
|
.replace("<br>", "<br>" _NOTES_BR)
|
|
|
|
.replace("<br/>", "<br/>" _NOTES_BR)
|
|
|
|
.replace("<br />", "<br />" _NOTES_BR)
|
|
|
|
.replace("<tr>", "<tr>" _NOTES_BR)
|
|
|
|
.replace("</tr>", "</tr>" _NOTES_BR);
|
|
|
|
notes.setHtml(tmp);
|
|
|
|
tmp = notes.toPlainText();
|
|
|
|
tmp.replace(_NOTES_BR, "<br/>");
|
|
|
|
#undef _NOTES_BR
|
|
|
|
} else {
|
|
|
|
tmp.replace("\n", "<br/>");
|
|
|
|
}
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatGetCylinder(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList getCylinder;
|
2024-05-28 21:31:11 +02:00
|
|
|
for (auto [i, cyl]: enumerated_range(d->cylinders)) {
|
2024-06-25 07:43:32 +02:00
|
|
|
if (d->is_cylinder_used(i))
|
2024-05-28 21:31:11 +02:00
|
|
|
getCylinder << QString::fromStdString(cyl.type.description);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
return getCylinder;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatStartPressure(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList startPressure;
|
2024-05-28 21:31:11 +02:00
|
|
|
for (auto [i, cyl]: enumerated_range(d->cylinders)) {
|
2024-06-25 07:43:32 +02:00
|
|
|
if (d->is_cylinder_used(i))
|
2024-05-28 21:31:11 +02:00
|
|
|
startPressure << getPressures(cyl, START_PRESSURE);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
return startPressure;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatEndPressure(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList endPressure;
|
2024-05-28 21:31:11 +02:00
|
|
|
for (auto [i, cyl]: enumerated_range(d->cylinders)) {
|
2024-06-25 07:43:32 +02:00
|
|
|
if (d->is_cylinder_used(i))
|
2024-05-28 21:31:11 +02:00
|
|
|
endPressure << getPressures(cyl, END_PRESSURE);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
return endPressure;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatFirstGas(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList gas;
|
2024-05-28 21:31:11 +02:00
|
|
|
for (auto [i, cyl]: enumerated_range(d->cylinders)) {
|
2024-06-25 07:43:32 +02:00
|
|
|
if (d->is_cylinder_used(i))
|
2024-05-28 21:31:11 +02:00
|
|
|
gas << get_gas_string(cyl.gasmix);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
return gas;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add string to sorted QStringList, if it doesn't already exist and
|
|
|
|
// it isn't the empty string.
|
2024-05-03 23:18:45 +02:00
|
|
|
static void addStringToSortedList(QStringList &l, const std::string &s)
|
2020-12-14 22:42:07 +01:00
|
|
|
{
|
2024-05-03 23:18:45 +02:00
|
|
|
if (s.empty())
|
2020-12-14 22:42:07 +01:00
|
|
|
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.
|
2024-05-03 23:18:45 +02:00
|
|
|
QString qs = QString::fromStdString(s);
|
2020-12-14 22:42:07 +01:00
|
|
|
auto it = std::lower_bound(l.begin(), l.end(), qs); // TODO: use locale-aware sorting
|
2024-05-03 23:18:45 +02:00
|
|
|
if (it != l.end() && *it == qs)
|
2020-12-14 22:42:07 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Add new string at sorted position
|
2024-05-03 23:18:45 +02:00
|
|
|
l.insert(it, qs);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatFullCylinderList()
|
|
|
|
{
|
|
|
|
QStringList cylinders;
|
2024-06-07 10:25:09 +02:00
|
|
|
for (auto &d: divelog.dives) {
|
2024-05-28 21:31:11 +02:00
|
|
|
for (const cylinder_t &cyl: d->cylinders)
|
|
|
|
addStringToSortedList(cylinders, cyl.type.description);
|
2020-12-14 22:42:07 +01:00
|
|
|
}
|
|
|
|
|
2024-05-03 23:18:45 +02:00
|
|
|
for (const auto &ti: tank_info_table)
|
|
|
|
addStringToSortedList(cylinders, ti.name);
|
2020-12-14 22:42:07 +01:00
|
|
|
|
|
|
|
return cylinders;
|
|
|
|
}
|
|
|
|
|
2024-05-28 21:31:11 +02:00
|
|
|
static QString formattedCylinder(const cylinder_t &cyl)
|
2020-12-14 23:21:58 +01:00
|
|
|
{
|
2024-05-28 21:31:11 +02:00
|
|
|
const std::string &desc = cyl.type.description;
|
|
|
|
QString fmt = !desc.empty() ? QString::fromStdString(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);
|
2020-12-14 23:21:58 +01:00
|
|
|
return fmt;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatCylinders(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList cylinders;
|
2024-05-28 21:31:11 +02:00
|
|
|
for (const cylinder_t &cyl: d->cylinders)
|
|
|
|
cylinders << formattedCylinder(cyl);
|
2020-12-14 23:21:58 +01:00
|
|
|
return cylinders;
|
|
|
|
}
|
2020-12-15 15:33:39 +01:00
|
|
|
|
|
|
|
QString formatGas(const dive *d)
|
|
|
|
{
|
|
|
|
/*WARNING: here should be the gastlist, returned
|
|
|
|
* from the get_gas_string function or this is correct?
|
|
|
|
*/
|
2024-05-28 21:31:11 +02:00
|
|
|
QString gases;
|
|
|
|
for (auto [i, cyl]: enumerated_range(d->cylinders)) {
|
2024-06-25 07:43:32 +02:00
|
|
|
if (!d->is_cylinder_used(i))
|
2020-12-15 15:33:39 +01:00
|
|
|
continue;
|
2024-05-28 21:31:11 +02:00
|
|
|
QString gas = QString::fromStdString(cyl.type.description);
|
2020-12-15 15:33:39 +01:00
|
|
|
if (!gas.isEmpty())
|
|
|
|
gas += QChar(' ');
|
2024-07-02 12:38:36 +02:00
|
|
|
gas += QString::fromStdString(cyl.gasmix.name());
|
2020-12-15 15:33:39 +01:00
|
|
|
// 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 formatSumWeight(const dive *d)
|
|
|
|
{
|
2024-06-30 11:13:39 +02:00
|
|
|
return get_weight_string(d->total_weight(), true);
|
2020-12-15 15:33:39 +01:00
|
|
|
}
|
|
|
|
|
2024-05-29 07:03:03 +02:00
|
|
|
static QString getFormattedWeight(const weightsystem_t &weight)
|
2020-12-15 15:33:39 +01:00
|
|
|
{
|
2024-05-29 07:03:03 +02:00
|
|
|
if (weight.description.empty())
|
2020-12-15 15:33:39 +01:00
|
|
|
return QString();
|
2024-05-29 07:03:03 +02:00
|
|
|
return QString::fromStdString(weight.description) +
|
|
|
|
", " + get_weight_string(weight.weight, true);
|
2020-12-15 15:33:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString formatWeightList(const dive *d)
|
|
|
|
{
|
|
|
|
QString weights;
|
2024-05-29 07:03:03 +02:00
|
|
|
for (auto &ws: d->weightsystems) {
|
|
|
|
QString w = getFormattedWeight(ws);
|
2020-12-15 15:33:39 +01:00
|
|
|
if (w.isEmpty())
|
|
|
|
continue;
|
|
|
|
weights += w + "; ";
|
|
|
|
}
|
|
|
|
return weights;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList formatWeights(const dive *d)
|
|
|
|
{
|
|
|
|
QStringList weights;
|
2024-05-29 07:03:03 +02:00
|
|
|
for (auto &ws: d->weightsystems) {
|
|
|
|
QString w = getFormattedWeight(ws);
|
2020-12-15 15:33:39 +01:00
|
|
|
if (w.isEmpty())
|
|
|
|
continue;
|
|
|
|
weights << w;
|
|
|
|
}
|
|
|
|
return weights;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString formatDiveDuration(const dive *d)
|
|
|
|
{
|
|
|
|
return get_dive_duration_string(d->duration.seconds,
|
|
|
|
gettextFromC::tr("h"), gettextFromC::tr("min"));
|
|
|
|
}
|
|
|
|
|
|
|
|
QString formatDiveGPS(const dive *d)
|
|
|
|
{
|
|
|
|
return d->dive_site ? printGPSCoords(&d->dive_site->location) : QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString formatDiveDate(const dive *d)
|
|
|
|
{
|
|
|
|
QDateTime localTime = timestampToDateTime(d->when);
|
2024-06-13 22:59:32 +02:00
|
|
|
return localTime.date().toString(QString::fromStdString(prefs.date_format_short));
|
2020-12-15 15:33:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString formatDiveTime(const dive *d)
|
|
|
|
{
|
|
|
|
QDateTime localTime = timestampToDateTime(d->when);
|
2024-06-13 22:59:32 +02:00
|
|
|
return localTime.time().toString(QString::fromStdString(prefs.time_format));
|
2020-12-15 15:33:39 +01:00
|
|
|
}
|
2020-12-15 16:10:08 +01:00
|
|
|
|
|
|
|
QString formatDiveDateTime(const dive *d)
|
|
|
|
{
|
|
|
|
QDateTime localTime = timestampToDateTime(d->when);
|
2024-06-13 22:59:32 +02:00
|
|
|
return QStringLiteral("%1 %2").arg(localTime.date().toString(QString::fromStdString(prefs.date_format_short)),
|
|
|
|
localTime.time().toString(QString::fromStdString(prefs.time_format)));
|
2020-12-15 16:10:08 +01:00
|
|
|
}
|
2021-01-01 12:49:50 +01:00
|
|
|
|
2023-03-17 08:21:47 +01:00
|
|
|
QString formatDiveGasString(const dive *d)
|
|
|
|
{
|
2024-06-24 21:04:31 +02:00
|
|
|
auto [o2, he, o2max ] = d->get_maximal_gas();
|
2023-03-17 08:21:47 +01:00
|
|
|
o2 = (o2 + 5) / 10;
|
|
|
|
he = (he + 5) / 10;
|
|
|
|
o2max = (o2max + 5) / 10;
|
|
|
|
|
|
|
|
if (he) {
|
|
|
|
if (o2 == o2max)
|
|
|
|
return qasprintf_loc("%d/%d", o2, he);
|
|
|
|
else
|
|
|
|
return qasprintf_loc("%d/%d…%d%%", o2, he, o2max);
|
|
|
|
} else if (o2) {
|
|
|
|
if (o2 == o2max)
|
|
|
|
return qasprintf_loc("%d%%", o2);
|
|
|
|
else
|
|
|
|
return qasprintf_loc("%d…%d%%", o2, o2max);
|
|
|
|
} else {
|
|
|
|
return gettextFromC::tr("air");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-01 12:49:50 +01:00
|
|
|
QString formatDayOfWeek(int day)
|
|
|
|
{
|
|
|
|
switch (day) {
|
|
|
|
default:
|
|
|
|
case 0: return gettextFromC::tr("Sunday");
|
|
|
|
case 1: return gettextFromC::tr("Monday");
|
|
|
|
case 2: return gettextFromC::tr("Tuesday");
|
|
|
|
case 3: return gettextFromC::tr("Wednesday");
|
|
|
|
case 4: return gettextFromC::tr("Thursday");
|
|
|
|
case 5: return gettextFromC::tr("Friday");
|
|
|
|
case 6: return gettextFromC::tr("Saturday");
|
|
|
|
}
|
|
|
|
}
|
2021-12-16 23:09:39 +01:00
|
|
|
|
2022-09-03 10:37:58 +02:00
|
|
|
QString formatMinutes(int seconds)
|
|
|
|
{
|
2024-05-01 21:01:06 +12:00
|
|
|
return QString::asprintf("%d:%.2d", FRACTION_TUPLE(seconds, 60));
|
2022-09-03 10:37:58 +02:00
|
|
|
}
|
|
|
|
|
2024-06-01 22:05:57 +02:00
|
|
|
QString formatTripTitle(const dive_trip &trip)
|
2021-12-16 23:09:39 +01:00
|
|
|
{
|
2024-06-08 15:28:16 +02:00
|
|
|
timestamp_t when = trip.date();
|
|
|
|
bool getday = trip.is_single_day();
|
2021-12-16 23:09:39 +01:00
|
|
|
|
|
|
|
QDateTime localTime = timestampToDateTime(when);
|
|
|
|
|
2024-06-01 22:05:57 +02:00
|
|
|
QString prefix = !trip.location.empty() ? QString::fromStdString(trip.location) + ", " : QString();
|
2021-12-16 23:09:39 +01:00
|
|
|
if (getday)
|
2024-06-13 22:59:32 +02:00
|
|
|
return prefix + loc.toString(localTime, QString::fromStdString(prefs.date_format));
|
2021-12-16 23:09:39 +01:00
|
|
|
else
|
|
|
|
return prefix + loc.toString(localTime, "MMM yyyy");
|
|
|
|
}
|
|
|
|
|
2024-06-01 22:05:57 +02:00
|
|
|
QString formatTripTitleWithDives(const dive_trip &trip)
|
2021-12-16 23:09:39 +01:00
|
|
|
{
|
2024-06-02 17:06:18 +02:00
|
|
|
int nr = static_cast<int>(trip.dives.size());
|
2021-12-16 23:09:39 +01:00
|
|
|
return formatTripTitle(trip) + " " +
|
|
|
|
gettextFromC::tr("(%n dive(s))", "", nr);
|
|
|
|
}
|