2024-05-01 23:16:00 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/* statistics.cpp
|
|
|
|
*
|
|
|
|
* core logic for the Info & Stats page
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "statistics.h"
|
|
|
|
#include "dive.h"
|
|
|
|
#include "divelog.h"
|
|
|
|
#include "event.h"
|
|
|
|
#include "gettext.h"
|
2024-05-28 21:31:11 +02:00
|
|
|
#include "range.h"
|
2024-05-01 23:16:00 +02:00
|
|
|
#include "sample.h"
|
|
|
|
#include "subsurface-time.h"
|
|
|
|
#include "trip.h"
|
|
|
|
#include "units.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
static void process_temperatures(const struct dive &dp, stats_t &stats)
|
2024-05-01 23:16:00 +02:00
|
|
|
{
|
|
|
|
temperature_t min_temp, mean_temp, max_temp = {.mkelvin = 0};
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
max_temp.mkelvin = dp.maxtemp.mkelvin;
|
2024-05-01 23:16:00 +02:00
|
|
|
if (max_temp.mkelvin && (!stats.max_temp.mkelvin || max_temp.mkelvin > stats.max_temp.mkelvin))
|
|
|
|
stats.max_temp.mkelvin = max_temp.mkelvin;
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
min_temp.mkelvin = dp.mintemp.mkelvin;
|
2024-05-01 23:16:00 +02:00
|
|
|
if (min_temp.mkelvin && (!stats.min_temp.mkelvin || min_temp.mkelvin < stats.min_temp.mkelvin))
|
|
|
|
stats.min_temp.mkelvin = min_temp.mkelvin;
|
|
|
|
|
|
|
|
if (min_temp.mkelvin || max_temp.mkelvin) {
|
|
|
|
mean_temp.mkelvin = min_temp.mkelvin;
|
|
|
|
if (mean_temp.mkelvin)
|
|
|
|
mean_temp.mkelvin = (mean_temp.mkelvin + max_temp.mkelvin) / 2;
|
|
|
|
else
|
|
|
|
mean_temp.mkelvin = max_temp.mkelvin;
|
|
|
|
stats.combined_temp.mkelvin += mean_temp.mkelvin;
|
|
|
|
stats.combined_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
static void process_dive(const struct dive &dive, stats_t &stats)
|
2024-05-01 23:16:00 +02:00
|
|
|
{
|
|
|
|
int old_tadt, sac_time = 0;
|
2024-06-07 10:25:09 +02:00
|
|
|
int32_t duration = dive.duration.seconds;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
old_tadt = stats.total_average_depth_time.seconds;
|
|
|
|
stats.total_time.seconds += duration;
|
|
|
|
if (duration > stats.longest_time.seconds)
|
|
|
|
stats.longest_time.seconds = duration;
|
|
|
|
if (stats.shortest_time.seconds == 0 || duration < stats.shortest_time.seconds)
|
|
|
|
stats.shortest_time.seconds = duration;
|
2024-06-07 10:25:09 +02:00
|
|
|
if (dive.maxdepth.mm > stats.max_depth.mm)
|
|
|
|
stats.max_depth.mm = dive.maxdepth.mm;
|
|
|
|
if (stats.min_depth.mm == 0 || dive.maxdepth.mm < stats.min_depth.mm)
|
|
|
|
stats.min_depth.mm = dive.maxdepth.mm;
|
|
|
|
stats.combined_max_depth.mm += dive.maxdepth.mm;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
process_temperatures(dive, stats);
|
|
|
|
|
|
|
|
/* Maybe we should drop zero-duration dives */
|
|
|
|
if (!duration)
|
|
|
|
return;
|
2024-06-07 10:25:09 +02:00
|
|
|
if (dive.meandepth.mm) {
|
2024-05-01 23:16:00 +02:00
|
|
|
stats.total_average_depth_time.seconds += duration;
|
|
|
|
stats.avg_depth.mm = lrint((1.0 * old_tadt * stats.avg_depth.mm +
|
2024-06-07 10:25:09 +02:00
|
|
|
duration * dive.meandepth.mm) /
|
2024-05-01 23:16:00 +02:00
|
|
|
stats.total_average_depth_time.seconds);
|
|
|
|
}
|
2024-06-07 10:25:09 +02:00
|
|
|
if (dive.sac > 100) { /* less than .1 l/min is bogus, even with a pSCR */
|
2024-05-01 23:16:00 +02:00
|
|
|
sac_time = stats.total_sac_time.seconds + duration;
|
|
|
|
stats.avg_sac.mliter = lrint((1.0 * stats.total_sac_time.seconds * stats.avg_sac.mliter +
|
2024-06-07 10:25:09 +02:00
|
|
|
duration * dive.sac) /
|
2024-05-01 23:16:00 +02:00
|
|
|
sac_time);
|
2024-06-07 10:25:09 +02:00
|
|
|
if (dive.sac > stats.max_sac.mliter)
|
|
|
|
stats.max_sac.mliter = dive.sac;
|
|
|
|
if (stats.min_sac.mliter == 0 || dive.sac < stats.min_sac.mliter)
|
|
|
|
stats.min_sac.mliter = dive.sac;
|
2024-05-01 23:16:00 +02:00
|
|
|
stats.total_sac_time.seconds = sac_time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Calculate a summary of the statistics and put in the stats_summary
|
|
|
|
* structure provided in the first parameter.
|
|
|
|
* Before first use, it should be initialized with init_stats_summary().
|
|
|
|
* After use, memory must be released with free_stats_summary().
|
|
|
|
*/
|
|
|
|
stats_summary calculate_stats_summary(bool selected_only)
|
|
|
|
{
|
|
|
|
struct tm tm;
|
|
|
|
int current_year = -1;
|
|
|
|
int current_month = 0;
|
|
|
|
int prev_month = 0, prev_year = 0;
|
2024-05-31 17:15:47 +02:00
|
|
|
dive_trip *trip_ptr = nullptr;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
stats_summary out;
|
|
|
|
|
|
|
|
/* stats_by_trip[0] is all the dives combined */
|
|
|
|
out.stats_by_trip.emplace_back();
|
|
|
|
|
|
|
|
/* Setting the is_trip to true to show the location as first
|
|
|
|
* field in the statistics window */
|
|
|
|
out.stats_by_type.resize(NUM_DIVEMODE + 1);
|
|
|
|
out.stats_by_type[0].location = translate("gettextFromC", "All (by type stats)");
|
|
|
|
out.stats_by_type[0].is_trip = true;
|
|
|
|
out.stats_by_type[1].location = translate("gettextFromC", divemode_text_ui[OC]);
|
|
|
|
out.stats_by_type[1].is_trip = true;
|
|
|
|
out.stats_by_type[2].location = translate("gettextFromC", divemode_text_ui[CCR]);
|
|
|
|
out.stats_by_type[2].is_trip = true;
|
|
|
|
out.stats_by_type[3].location = translate("gettextFromC", divemode_text_ui[PSCR]);
|
|
|
|
out.stats_by_type[3].is_trip = true;
|
|
|
|
out.stats_by_type[4].location = translate("gettextFromC", divemode_text_ui[FREEDIVE]);
|
|
|
|
out.stats_by_type[4].is_trip = true;
|
|
|
|
|
|
|
|
out.stats_by_depth.resize((STATS_MAX_DEPTH / STATS_DEPTH_BUCKET) + 1);
|
|
|
|
out.stats_by_depth[0].location = translate("gettextFromC", "All (by max depth stats)");
|
|
|
|
out.stats_by_depth[0].is_trip = true;
|
|
|
|
|
|
|
|
out.stats_by_temp.resize((STATS_MAX_TEMP / STATS_TEMP_BUCKET) + 1);
|
|
|
|
out.stats_by_temp[0].location = translate("gettextFromC", "All (by min. temp stats)");
|
|
|
|
out.stats_by_temp[0].is_trip = true;
|
|
|
|
|
|
|
|
/* this relies on the fact that the dives in the dive_table
|
|
|
|
* are in chronological order */
|
2024-06-07 10:25:09 +02:00
|
|
|
for (auto &dp: divelog.dives) {
|
2024-05-01 23:16:00 +02:00
|
|
|
if (selected_only && !dp->selected)
|
|
|
|
continue;
|
|
|
|
if (dp->invalid)
|
|
|
|
continue;
|
|
|
|
//process_dive(dp, &stats);
|
|
|
|
|
|
|
|
/* yearly statistics */
|
|
|
|
utc_mkdate(dp->when, &tm);
|
|
|
|
|
|
|
|
if (current_year != tm.tm_year || out.stats_yearly.empty()) {
|
|
|
|
current_year = tm.tm_year;
|
|
|
|
out.stats_yearly.emplace_back();
|
|
|
|
out.stats_yearly.back().is_year = true;
|
|
|
|
}
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_yearly.back());
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
out.stats_yearly.back().selection_size++;
|
|
|
|
out.stats_yearly.back().period = current_year;
|
|
|
|
|
|
|
|
/* stats_by_type[0] is all the dives combined */
|
|
|
|
out.stats_by_type[0].selection_size++;
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_type[0]);
|
2024-05-01 23:16:00 +02:00
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_type[dp->dcs[0].divemode + 1]);
|
2024-05-27 17:09:48 +02:00
|
|
|
out.stats_by_type[dp->dcs[0].divemode + 1].selection_size++;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
/* stats_by_depth[0] is all the dives combined */
|
|
|
|
out.stats_by_depth[0].selection_size++;
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_depth[0]);
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
int d_idx = dp->maxdepth.mm / (STATS_DEPTH_BUCKET * 1000);
|
|
|
|
d_idx = std::clamp(d_idx, 0, STATS_MAX_DEPTH / STATS_DEPTH_BUCKET);
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_depth[d_idx + 1]);
|
2024-05-01 23:16:00 +02:00
|
|
|
out.stats_by_depth[d_idx + 1].selection_size++;
|
|
|
|
|
|
|
|
/* stats_by_temp[0] is all the dives combined */
|
|
|
|
out.stats_by_temp[0].selection_size++;
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_temp[0]);
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
int t_idx = ((int)mkelvin_to_C(dp->mintemp.mkelvin)) / STATS_TEMP_BUCKET;
|
|
|
|
t_idx = std::clamp(t_idx, 0, STATS_MAX_TEMP / STATS_TEMP_BUCKET);
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_temp[t_idx + 1]);
|
2024-05-01 23:16:00 +02:00
|
|
|
out.stats_by_temp[t_idx + 1].selection_size++;
|
|
|
|
|
|
|
|
if (dp->divetrip != NULL) {
|
|
|
|
if (trip_ptr != dp->divetrip) {
|
|
|
|
trip_ptr = dp->divetrip;
|
|
|
|
out.stats_by_trip.emplace_back();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stats_by_trip[0] is all the dives combined */
|
|
|
|
/* TODO: yet, this doesn't seem to consider dives outside of trips !? */
|
|
|
|
out.stats_by_trip[0].selection_size++;
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_trip[0]);
|
2024-05-01 23:16:00 +02:00
|
|
|
out.stats_by_trip[0].is_trip = true;
|
|
|
|
out.stats_by_trip[0].location = translate("gettextFromC", "All (by trip stats)");
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_by_trip.back());
|
2024-05-01 23:16:00 +02:00
|
|
|
out.stats_by_trip.back().selection_size++;
|
|
|
|
out.stats_by_trip.back().is_trip = true;
|
|
|
|
out.stats_by_trip.back().location = dp->divetrip->location;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* monthly statistics */
|
|
|
|
if (current_month == 0 || out.stats_monthly.empty()) {
|
|
|
|
current_month = tm.tm_mon + 1;
|
|
|
|
out.stats_monthly.emplace_back();
|
|
|
|
} else {
|
|
|
|
if (current_month != tm.tm_mon + 1)
|
|
|
|
current_month = tm.tm_mon + 1;
|
|
|
|
if (prev_month != current_month || prev_year != current_year)
|
|
|
|
out.stats_monthly.emplace_back();
|
|
|
|
}
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dp, out.stats_monthly.back());
|
2024-05-01 23:16:00 +02:00
|
|
|
out.stats_monthly.back().selection_size++;
|
|
|
|
out.stats_monthly.back().period = current_month;
|
|
|
|
prev_month = current_month;
|
|
|
|
prev_year = current_year;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add labels for depth ranges up to maximum depth seen */
|
|
|
|
if (out.stats_by_depth[0].selection_size) {
|
|
|
|
int d_idx = out.stats_by_depth[0].max_depth.mm;
|
|
|
|
if (d_idx > STATS_MAX_DEPTH * 1000)
|
|
|
|
d_idx = STATS_MAX_DEPTH * 1000;
|
|
|
|
for (int r = 0; r * (STATS_DEPTH_BUCKET * 1000) < d_idx; ++r)
|
|
|
|
out.stats_by_depth[r+1].is_trip = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add labels for depth ranges up to maximum temperature seen */
|
|
|
|
if (out.stats_by_temp[0].selection_size) {
|
|
|
|
int t_idx = (int)mkelvin_to_C(out.stats_by_temp[0].max_temp.mkelvin);
|
|
|
|
if (t_idx > STATS_MAX_TEMP)
|
|
|
|
t_idx = STATS_MAX_TEMP;
|
|
|
|
for (int r = 0; r * STATS_TEMP_BUCKET < t_idx; ++r)
|
|
|
|
out.stats_by_temp[r+1].is_trip = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
stats_summary::stats_summary() = default;
|
|
|
|
stats_summary::~stats_summary() = default;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
/* make sure we skip the selected summary entries */
|
|
|
|
stats_t calculate_stats_selected()
|
|
|
|
{
|
|
|
|
stats_t stats_selection;
|
2024-06-07 10:25:09 +02:00
|
|
|
unsigned int nr = 0;
|
2024-05-01 23:16:00 +02:00
|
|
|
|
2024-06-07 10:25:09 +02:00
|
|
|
for (auto &dive: divelog.dives) {
|
2024-05-01 23:16:00 +02:00
|
|
|
if (dive->selected && !dive->invalid) {
|
2024-06-07 10:25:09 +02:00
|
|
|
process_dive(*dive, stats_selection);
|
2024-05-01 23:16:00 +02:00
|
|
|
nr++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
stats_selection.selection_size = nr;
|
|
|
|
return stats_selection;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SOME_GAS 5000 // 5bar drop in cylinder pressure makes cylinder used
|
|
|
|
|
2024-05-28 21:31:11 +02:00
|
|
|
/* Returns a vector with dive->cylinders.size() entries */
|
2024-05-01 23:16:00 +02:00
|
|
|
std::vector<volume_t> get_gas_used(struct dive *dive)
|
|
|
|
{
|
2024-05-28 21:31:11 +02:00
|
|
|
std::vector<volume_t> gases(dive->cylinders.size());
|
|
|
|
for (auto [idx, cyl]: enumerated_range(dive->cylinders)) {
|
2024-05-01 23:16:00 +02:00
|
|
|
pressure_t start, end;
|
|
|
|
|
2024-05-28 21:31:11 +02:00
|
|
|
start = cyl.start.mbar ? cyl.start : cyl.sample_start;
|
|
|
|
end = cyl.end.mbar ? cyl.end : cyl.sample_end;
|
2024-05-01 23:16:00 +02:00
|
|
|
if (end.mbar && start.mbar > end.mbar)
|
2024-05-28 21:31:11 +02:00
|
|
|
gases[idx].mliter = gas_volume(&cyl, start) - gas_volume(&cyl, end);
|
2024-05-01 23:16:00 +02:00
|
|
|
else
|
|
|
|
gases[idx].mliter = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gases;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Quite crude reverse-blender-function, but it produces an approx result.
|
|
|
|
* Returns an (O2, He) pair. */
|
|
|
|
static std::pair<volume_t, volume_t> get_gas_parts(struct gasmix mix, volume_t vol, int o2_in_topup)
|
|
|
|
{
|
|
|
|
if (gasmix_is_air(mix))
|
2024-05-28 21:31:11 +02:00
|
|
|
return { volume_t() , volume_t() };
|
2024-05-01 23:16:00 +02:00
|
|
|
|
|
|
|
volume_t air = { (int)lrint(((double)vol.mliter * get_n2(mix)) / (1000 - o2_in_topup)) };
|
|
|
|
volume_t he = { (int)lrint(((double)vol.mliter * get_he(mix)) / 1000.0) };
|
|
|
|
volume_t o2 = { vol.mliter - he.mliter - air.mliter };
|
|
|
|
return std::make_pair(o2, he);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<volume_t, volume_t> selected_dives_gas_parts()
|
|
|
|
{
|
2024-05-04 19:15:47 +02:00
|
|
|
volume_t o2_tot, he_tot;
|
2024-06-07 10:25:09 +02:00
|
|
|
for (auto &d: divelog.dives) {
|
2024-05-01 23:16:00 +02:00
|
|
|
if (!d->selected || d->invalid)
|
|
|
|
continue;
|
|
|
|
int j = 0;
|
2024-06-07 10:25:09 +02:00
|
|
|
for (auto &gas: get_gas_used(d.get())) {
|
2024-05-01 23:16:00 +02:00
|
|
|
if (gas.mliter) {
|
2024-06-25 07:43:32 +02:00
|
|
|
auto [o2, he] = get_gas_parts(d->get_cylinder(j)->gasmix, gas, O2_IN_AIR);
|
2024-05-01 23:16:00 +02:00
|
|
|
o2_tot.mliter += o2.mliter;
|
|
|
|
he_tot.mliter += he.mliter;
|
|
|
|
}
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return std::make_pair(o2_tot, he_tot);
|
|
|
|
}
|