From 4613321e3e4c8dac659df2ec537f2978cee56c64 Mon Sep 17 00:00:00 2001 From: Doug Junkins Date: Sat, 9 Mar 2019 18:21:46 -0800 Subject: Add backend support for statistics by depth and temperature Added stats_t structures to summarize dive statistics by depth and by temperature. Process each dive to add the dive stats to the proper depth and temperature bucket. Buckets are defined using constants STATS_MAX_DEPTH, STATS_DEPTH_BUCKET, STATS_MAX_TEMP, and STATS_TEMP_BUCKET which are defined in statistics.h Signed-off-by: Doug Junkins --- core/statistics.c | 65 +++++++++++++++++++++++++++++++++++++++++++++++++++++-- core/statistics.h | 7 ++++++ 2 files changed, 70 insertions(+), 2 deletions(-) diff --git a/core/statistics.c b/core/statistics.c index d6d9418d7..5eb2f3ac5 100644 --- a/core/statistics.c +++ b/core/statistics.c @@ -14,6 +14,7 @@ #include "display.h" #include "divelist.h" #include "statistics.h" +#include "units.h" static void process_temperatures(struct dive *dp, stats_t *stats) { @@ -95,6 +96,7 @@ char *get_minutes(int seconds) void calculate_stats_summary(struct stats_summary *out, bool selected_only) { int idx; + int t_idx, d_idx, r; struct dive *dp; struct tm tm; int current_year = 0; @@ -104,7 +106,7 @@ void calculate_stats_summary(struct stats_summary *out, bool selected_only) int prev_month = 0, prev_year = 0; int trip_iter = 0; dive_trip_t *trip_ptr = 0; - unsigned int size, tsize; + size_t size, tsize, dsize, tmsize; stats_t stats = { 0 }; if (dive_table.nr > 0) { @@ -119,17 +121,24 @@ void calculate_stats_summary(struct stats_summary *out, bool selected_only) size = sizeof(stats_t) * (dive_table.nr + 1); tsize = sizeof(stats_t) * (NUM_DIVEMODE + 1); + dsize = sizeof(stats_t) * ((STATS_MAX_DEPTH / STATS_DEPTH_BUCKET) + 1); + tmsize = sizeof(stats_t) * ((STATS_MAX_TEMP / STATS_TEMP_BUCKET) + 1); free_stats_summary(out); out->stats_yearly = malloc(size); out->stats_monthly = malloc(size); out->stats_by_trip = malloc(size); out->stats_by_type = malloc(tsize); - if (!out->stats_yearly || !out->stats_monthly || !out->stats_by_trip || !out->stats_by_type) + out->stats_by_depth = malloc(dsize); + out->stats_by_temp = malloc(tmsize); + if (!out->stats_yearly || !out->stats_monthly || !out->stats_by_trip || + !out->stats_by_type || !out->stats_by_depth || !out->stats_by_temp) return; memset(out->stats_yearly, 0, size); memset(out->stats_monthly, 0, size); memset(out->stats_by_trip, 0, size); memset(out->stats_by_type, 0, tsize); + memset(out->stats_by_depth, 0, dsize); + memset(out->stats_by_temp, 0, tmsize); out->stats_yearly[0].is_year = true; /* Setting the is_trip to true to show the location as first @@ -145,6 +154,12 @@ void calculate_stats_summary(struct stats_summary *out, bool selected_only) out->stats_by_type[4].location = strdup(translate("gettextFromC", divemode_text_ui[FREEDIVE])); out->stats_by_type[4].is_trip = true; + out->stats_by_depth[0].location = strdup(translate("gettextFromC", "All (by max depth stats)")); + out->stats_by_depth[0].is_trip = true; + + out->stats_by_temp[0].location = strdup(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 */ for_each_dive (idx, dp) { @@ -174,6 +189,30 @@ void calculate_stats_summary(struct stats_summary *out, bool selected_only) process_dive(dp, &(out->stats_by_type[dp->dc.divemode + 1])); out->stats_by_type[dp->dc.divemode + 1].selection_size++; + /* stats_by_depth[0] is all the dives combined */ + out->stats_by_depth[0].selection_size++; + process_dive(dp, &(out->stats_by_depth[0])); + + d_idx = dp->maxdepth.mm / (STATS_DEPTH_BUCKET * 1000); + if (d_idx < 0) + d_idx = 0; + if (d_idx >= STATS_MAX_DEPTH / STATS_DEPTH_BUCKET) + d_idx = STATS_MAX_DEPTH / STATS_DEPTH_BUCKET - 1; + process_dive(dp, &(out->stats_by_depth[d_idx + 1])); + 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++; + process_dive(dp, &(out->stats_by_temp[0])); + + t_idx = ((int)mkelvin_to_C(dp->mintemp.mkelvin)) / STATS_TEMP_BUCKET; + if (t_idx < 0) + t_idx = 0; + if (t_idx >= STATS_MAX_TEMP / STATS_TEMP_BUCKET) + t_idx = STATS_MAX_TEMP / STATS_TEMP_BUCKET - 1; + process_dive(dp, &(out->stats_by_temp[t_idx + 1])); + out->stats_by_temp[t_idx + 1].selection_size++; + if (dp->divetrip != NULL) { if (trip_ptr != dp->divetrip) { trip_ptr = dp->divetrip; @@ -207,6 +246,24 @@ void calculate_stats_summary(struct stats_summary *out, bool selected_only) 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) { + d_idx = out->stats_by_depth[0].max_depth.mm; + if (d_idx > STATS_MAX_DEPTH * 1000) + d_idx = STATS_MAX_DEPTH * 1000; + for (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) { + 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 (r = 0; r * STATS_TEMP_BUCKET < t_idx; ++r) + out->stats_by_temp[r+1].is_trip = true; + } } void free_stats_summary(struct stats_summary *stats) @@ -215,6 +272,8 @@ void free_stats_summary(struct stats_summary *stats) free(stats->stats_monthly); free(stats->stats_by_trip); free(stats->stats_by_type); + free(stats->stats_by_depth); + free(stats->stats_by_temp); } void init_stats_summary(struct stats_summary *stats) @@ -223,6 +282,8 @@ void init_stats_summary(struct stats_summary *stats) stats->stats_monthly = NULL; stats->stats_by_trip = NULL; stats->stats_by_type = NULL; + stats->stats_by_depth = NULL; + stats->stats_by_temp = NULL; } /* make sure we skip the selected summary entries */ diff --git a/core/statistics.h b/core/statistics.h index 6072f93b2..257303e19 100644 --- a/core/statistics.h +++ b/core/statistics.h @@ -12,6 +12,11 @@ #include "core/units.h" #include "core/dive.h" // For MAX_CYLINDERS +#define STATS_MAX_DEPTH 300 /* Max depth for stats bucket is 300m */ +#define STATS_DEPTH_BUCKET 10 /* Size of buckets for depth range */ +#define STATS_MAX_TEMP 40 /* Max temp for stats bucket is 40C */ +#define STATS_TEMP_BUCKET 5 /* Size of buckets for temp range */ + typedef struct { int period; @@ -44,6 +49,8 @@ struct stats_summary { stats_t *stats_monthly; stats_t *stats_by_trip; stats_t *stats_by_type; + stats_t *stats_by_depth; + stats_t *stats_by_temp; }; #ifdef __cplusplus -- cgit v1.2.3-70-g09d2