mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-06 00:01:29 +00:00
e65c7cedc8
Make it more obvious that the FRACTION and SIGNED_FRAC defines return a tuple / triplet of values. Fixes https://github.com/subsurface/subsurface/pull/4171#discussion_r1585941133 Complained-about-by: @bstoeger Signed-off-by: Michael Keller <github@ike.ch>
330 lines
8.2 KiB
C++
330 lines
8.2 KiB
C++
#include "string-format.h"
|
|
#include "dive.h"
|
|
#include "divesite.h"
|
|
#include "event.h"
|
|
#include "format.h"
|
|
#include "qthelper.h"
|
|
#include "subsurface-string.h"
|
|
#include "trip.h"
|
|
#include <QDateTime>
|
|
#include <QLocale>
|
|
#include <QTextDocument>
|
|
|
|
enum returnPressureSelector { START_PRESSURE, END_PRESSURE };
|
|
static QLocale loc;
|
|
|
|
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 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)
|
|
{
|
|
QString tmp = d->notes ? QString::fromUtf8(d->notes) : QString();
|
|
if (is_dc_planner(&d->dc)) {
|
|
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;
|
|
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 formatStartPressure(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 formatEndPressure(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 formatFirstGas(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 formatFullCylinderList()
|
|
{
|
|
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 < tank_info_table.nr; ti++)
|
|
addStringToSortedList(cylinders, tank_info_table.infos[ti].name);
|
|
|
|
return cylinders;
|
|
}
|
|
|
|
static QString formattedCylinder(const struct dive *dive, int idx)
|
|
{
|
|
const cylinder_t *cyl = get_cylinder(dive, idx);
|
|
const char *desc = cyl->type.description;
|
|
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;
|
|
}
|
|
|
|
QStringList formatCylinders(const dive *d)
|
|
{
|
|
QStringList cylinders;
|
|
for (int i = 0; i < d->cylinders.nr; i++) {
|
|
QString cyl = formattedCylinder(d, i);
|
|
cylinders << cyl;
|
|
}
|
|
return cylinders;
|
|
}
|
|
|
|
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 formatSumWeight(const dive *d)
|
|
{
|
|
return get_weight_string(weight_t { total_weight(d) }, true);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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);
|
|
return localTime.date().toString(prefs.date_format_short);
|
|
}
|
|
|
|
QString formatDiveTime(const dive *d)
|
|
{
|
|
QDateTime localTime = timestampToDateTime(d->when);
|
|
return localTime.time().toString(prefs.time_format);
|
|
}
|
|
|
|
QString formatDiveDateTime(const dive *d)
|
|
{
|
|
QDateTime localTime = timestampToDateTime(d->when);
|
|
return QStringLiteral("%1 %2").arg(localTime.date().toString(prefs.date_format_short),
|
|
localTime.time().toString(prefs.time_format));
|
|
}
|
|
|
|
QString formatDiveGasString(const dive *d)
|
|
{
|
|
int o2, he, o2max;
|
|
get_dive_gas(d, &o2, &he, &o2max);
|
|
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");
|
|
}
|
|
}
|
|
|
|
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");
|
|
}
|
|
}
|
|
|
|
QString formatMinutes(int seconds)
|
|
{
|
|
return QString::asprintf("%d:%.2d", FRACTION_TUPLE(seconds, 60));
|
|
}
|
|
|
|
QString formatTripTitle(const dive_trip *trip)
|
|
{
|
|
if (!trip)
|
|
return QString();
|
|
|
|
timestamp_t when = trip_date(trip);
|
|
bool getday = trip_is_single_day(trip);
|
|
|
|
QDateTime localTime = timestampToDateTime(when);
|
|
|
|
QString prefix = !empty_string(trip->location) ? QString(trip->location) + ", " : QString();
|
|
if (getday)
|
|
return prefix + loc.toString(localTime, prefs.date_format);
|
|
else
|
|
return prefix + loc.toString(localTime, "MMM yyyy");
|
|
}
|
|
|
|
QString formatTripTitleWithDives(const dive_trip *trip)
|
|
{
|
|
int nr = trip->dives.nr;
|
|
return formatTripTitle(trip) + " " +
|
|
gettextFromC::tr("(%n dive(s))", "", nr);
|
|
}
|