2019-05-31 14:09:14 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
#include "trip.h"
|
2020-05-01 11:43:52 +00:00
|
|
|
#include "dive.h"
|
core: introduce divelog structure
The parser API was very annoying, as a number of tables
to-be-filled were passed in as pointers. The goal of this
commit is to collect all these tables in a single struct.
This should make it (more or less) clear what is actually
written into the divelog files.
Moreover, it should now be rather easy to search for
instances, where the global logfile is accessed (and it
turns out that there are many!).
The divelog struct does not contain the tables as substructs,
but only collects pointers. The idea is that the "divelog.h"
file can be included without all the other files describing
the numerous tables.
To make it easier to use from C++ parts of the code, the
struct implements a constructor and a destructor. Sadly,
we can't use smart pointers, since the pointers are accessed
from C code. Therfore the constructor and destructor are
quite complex.
The whole commit is large, but was mostly an automatic
conversion.
One oddity of note: the divelog structure also contains
the "autogroup" flag, since that is saved in the divelog.
This actually fixes a bug: Before, when importing dives
from a different log, the autogroup flag was overwritten.
This was probably not intended and does not happen anymore.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2022-11-08 20:31:08 +00:00
|
|
|
#include "divelog.h"
|
2020-05-01 12:07:59 +00:00
|
|
|
#include "subsurface-time.h"
|
2019-05-31 14:09:14 +00:00
|
|
|
#include "subsurface-string.h"
|
2019-11-24 14:02:34 +00:00
|
|
|
#include "selection.h"
|
2019-05-31 14:09:14 +00:00
|
|
|
#include "table.h"
|
2020-04-17 20:02:25 +00:00
|
|
|
#include "core/qthelper.h"
|
2019-05-31 14:09:14 +00:00
|
|
|
|
|
|
|
#ifdef DEBUG_TRIP
|
|
|
|
void dump_trip_list(void)
|
|
|
|
{
|
|
|
|
dive_trip_t *trip;
|
|
|
|
int i = 0;
|
|
|
|
timestamp_t last_time = 0;
|
|
|
|
|
core: introduce divelog structure
The parser API was very annoying, as a number of tables
to-be-filled were passed in as pointers. The goal of this
commit is to collect all these tables in a single struct.
This should make it (more or less) clear what is actually
written into the divelog files.
Moreover, it should now be rather easy to search for
instances, where the global logfile is accessed (and it
turns out that there are many!).
The divelog struct does not contain the tables as substructs,
but only collects pointers. The idea is that the "divelog.h"
file can be included without all the other files describing
the numerous tables.
To make it easier to use from C++ parts of the code, the
struct implements a constructor and a destructor. Sadly,
we can't use smart pointers, since the pointers are accessed
from C code. Therfore the constructor and destructor are
quite complex.
The whole commit is large, but was mostly an automatic
conversion.
One oddity of note: the divelog structure also contains
the "autogroup" flag, since that is saved in the divelog.
This actually fixes a bug: Before, when importing dives
from a different log, the autogroup flag was overwritten.
This was probably not intended and does not happen anymore.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2022-11-08 20:31:08 +00:00
|
|
|
for (i = 0; i < divelog.trips->nr; ++i) {
|
2019-05-31 14:09:14 +00:00
|
|
|
struct tm tm;
|
core: introduce divelog structure
The parser API was very annoying, as a number of tables
to-be-filled were passed in as pointers. The goal of this
commit is to collect all these tables in a single struct.
This should make it (more or less) clear what is actually
written into the divelog files.
Moreover, it should now be rather easy to search for
instances, where the global logfile is accessed (and it
turns out that there are many!).
The divelog struct does not contain the tables as substructs,
but only collects pointers. The idea is that the "divelog.h"
file can be included without all the other files describing
the numerous tables.
To make it easier to use from C++ parts of the code, the
struct implements a constructor and a destructor. Sadly,
we can't use smart pointers, since the pointers are accessed
from C code. Therfore the constructor and destructor are
quite complex.
The whole commit is large, but was mostly an automatic
conversion.
One oddity of note: the divelog structure also contains
the "autogroup" flag, since that is saved in the divelog.
This actually fixes a bug: Before, when importing dives
from a different log, the autogroup flag was overwritten.
This was probably not intended and does not happen anymore.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2022-11-08 20:31:08 +00:00
|
|
|
trip = divelog.trips->trips[i];
|
2019-05-31 14:09:14 +00:00
|
|
|
utc_mkdate(trip_date(trip), &tm);
|
|
|
|
if (trip_date(trip) < last_time)
|
|
|
|
printf("\n\ntrip_table OUT OF ORDER!!!\n\n\n");
|
|
|
|
printf("%s trip %d to \"%s\" on %04u-%02u-%02u %02u:%02u:%02u (%d dives - %p)\n",
|
|
|
|
trip->autogen ? "autogen " : "",
|
|
|
|
i + 1, trip->location,
|
|
|
|
tm.tm_year, tm.tm_mon + 1, tm.tm_mday, tm.tm_hour, tm.tm_min, tm.tm_sec,
|
|
|
|
trip->dives.nr, trip);
|
|
|
|
last_time = trip_date(trip);
|
|
|
|
}
|
|
|
|
printf("-----\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* free resources associated with a trip structure */
|
|
|
|
void free_trip(dive_trip_t *trip)
|
|
|
|
{
|
|
|
|
if (trip) {
|
|
|
|
free(trip->location);
|
|
|
|
free(trip->notes);
|
|
|
|
free(trip->dives.dives);
|
|
|
|
free(trip);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Trip table functions */
|
|
|
|
static MAKE_GET_IDX(trip_table, struct dive_trip *, trips)
|
|
|
|
static MAKE_GROW_TABLE(trip_table, struct dive_trip *, trips)
|
|
|
|
static MAKE_GET_INSERTION_INDEX(trip_table, struct dive_trip *, trips, trip_less_than)
|
|
|
|
static MAKE_ADD_TO(trip_table, struct dive_trip *, trips)
|
|
|
|
static MAKE_REMOVE_FROM(trip_table, trips)
|
|
|
|
MAKE_SORT(trip_table, struct dive_trip *, trips, comp_trips)
|
|
|
|
MAKE_REMOVE(trip_table, struct dive_trip *, trip)
|
2019-06-04 18:59:08 +00:00
|
|
|
MAKE_CLEAR_TABLE(trip_table, trips, trip)
|
2022-11-12 07:40:04 +00:00
|
|
|
MAKE_MOVE_TABLE(trip_table, trips)
|
2019-05-31 14:09:14 +00:00
|
|
|
|
|
|
|
timestamp_t trip_date(const struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
if (!trip || trip->dives.nr == 0)
|
|
|
|
return 0;
|
|
|
|
return trip->dives.dives[0]->when;
|
|
|
|
}
|
|
|
|
|
|
|
|
timestamp_t trip_enddate(const struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
if (!trip || trip->dives.nr == 0)
|
|
|
|
return 0;
|
|
|
|
return dive_endtime(trip->dives.dives[trip->dives.nr - 1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if we have a trip right before / after this dive */
|
|
|
|
bool is_trip_before_after(const struct dive *dive, bool before)
|
|
|
|
{
|
|
|
|
int idx = get_idx_by_uniq_id(dive->id);
|
|
|
|
if (before) {
|
2020-10-24 21:50:59 +00:00
|
|
|
const struct dive *d = get_dive(idx - 1);
|
|
|
|
if (d && d->divetrip)
|
2019-05-31 14:09:14 +00:00
|
|
|
return true;
|
|
|
|
} else {
|
2020-10-24 21:50:59 +00:00
|
|
|
const struct dive *d = get_dive(idx + 1);
|
|
|
|
if (d && d->divetrip)
|
2019-05-31 14:09:14 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add dive to a trip. Caller is responsible for removing dive
|
|
|
|
* from trip beforehand. */
|
|
|
|
void add_dive_to_trip(struct dive *dive, dive_trip_t *trip)
|
|
|
|
{
|
|
|
|
if (dive->divetrip == trip)
|
|
|
|
return;
|
|
|
|
if (dive->divetrip)
|
2020-04-17 20:02:25 +00:00
|
|
|
SSRF_INFO("Warning: adding dive to trip that has trip set\n");
|
2019-05-31 14:09:14 +00:00
|
|
|
insert_dive(&trip->dives, dive);
|
|
|
|
dive->divetrip = trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove a dive from the trip it's associated to, but don't delete the
|
|
|
|
* trip if this was the last dive in the trip. the caller is responsible
|
|
|
|
* for removing the trip, if the trip->dives.nr went to 0.
|
|
|
|
*/
|
|
|
|
struct dive_trip *unregister_dive_from_trip(struct dive *dive)
|
|
|
|
{
|
|
|
|
dive_trip_t *trip = dive->divetrip;
|
|
|
|
|
|
|
|
if (!trip)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
remove_dive(dive, &trip->dives);
|
|
|
|
dive->divetrip = NULL;
|
|
|
|
return trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void delete_trip(dive_trip_t *trip, struct trip_table *trip_table_arg)
|
|
|
|
{
|
|
|
|
remove_trip(trip, trip_table_arg);
|
|
|
|
free_trip(trip);
|
|
|
|
}
|
|
|
|
|
|
|
|
void remove_dive_from_trip(struct dive *dive, struct trip_table *trip_table_arg)
|
|
|
|
{
|
|
|
|
struct dive_trip *trip = unregister_dive_from_trip(dive);
|
|
|
|
if (trip && trip->dives.nr == 0)
|
|
|
|
delete_trip(trip, trip_table_arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
dive_trip_t *alloc_trip(void)
|
|
|
|
{
|
2019-09-14 17:26:34 +00:00
|
|
|
dive_trip_t *res = calloc(1, sizeof(dive_trip_t));
|
|
|
|
res->id = dive_getUniqID();
|
|
|
|
return res;
|
2019-05-31 14:09:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* insert the trip into the trip table */
|
|
|
|
void insert_trip(dive_trip_t *dive_trip, struct trip_table *trip_table_arg)
|
|
|
|
{
|
|
|
|
int idx = trip_table_get_insertion_index(trip_table_arg, dive_trip);
|
|
|
|
add_to_trip_table(trip_table_arg, idx, dive_trip);
|
|
|
|
#ifdef DEBUG_TRIP
|
|
|
|
dump_trip_list();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
dive_trip_t *create_trip_from_dive(struct dive *dive)
|
|
|
|
{
|
|
|
|
dive_trip_t *trip;
|
|
|
|
|
|
|
|
trip = alloc_trip();
|
|
|
|
trip->location = copy_string(get_dive_location(dive));
|
|
|
|
|
|
|
|
return trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
dive_trip_t *create_and_hookup_trip_from_dive(struct dive *dive, struct trip_table *trip_table_arg)
|
|
|
|
{
|
2019-10-27 01:20:03 +00:00
|
|
|
dive_trip_t *dive_trip;
|
2019-05-31 14:09:14 +00:00
|
|
|
|
|
|
|
dive_trip = create_trip_from_dive(dive);
|
|
|
|
|
|
|
|
add_dive_to_trip(dive, dive_trip);
|
|
|
|
insert_trip(dive_trip, trip_table_arg);
|
|
|
|
return dive_trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* random threshold: three days without diving -> new trip
|
|
|
|
* this works very well for people who usually dive as part of a trip and don't
|
|
|
|
* regularly dive at a local facility; this is why trips are an optional feature */
|
|
|
|
#define TRIP_THRESHOLD 3600 * 24 * 3
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find a trip a new dive should be autogrouped with. If no such trips
|
|
|
|
* exist, allocate a new trip. The bool "*allocated" is set to true
|
|
|
|
* if a new trip was allocated.
|
|
|
|
*/
|
|
|
|
dive_trip_t *get_trip_for_new_dive(struct dive *new_dive, bool *allocated)
|
|
|
|
{
|
|
|
|
struct dive *d;
|
|
|
|
dive_trip_t *trip;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Find dive that is within TRIP_THRESHOLD of current dive */
|
|
|
|
for_each_dive(i, d) {
|
|
|
|
/* Check if we're past the range of possible dives */
|
|
|
|
if (d->when >= new_dive->when + TRIP_THRESHOLD)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (d->when + TRIP_THRESHOLD >= new_dive->when && d->divetrip) {
|
|
|
|
/* Found a dive with trip in the range */
|
|
|
|
*allocated = false;
|
|
|
|
return d->divetrip;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Didn't find a trip -> allocate a new one */
|
|
|
|
trip = create_trip_from_dive(new_dive);
|
|
|
|
trip->autogen = true;
|
|
|
|
*allocated = true;
|
|
|
|
return trip;
|
|
|
|
}
|
|
|
|
|
2020-02-20 21:20:03 +00:00
|
|
|
/* lookup of trip in main trip_table based on its id */
|
|
|
|
dive_trip_t *get_trip_by_uniq_id(int tripId)
|
|
|
|
{
|
core: introduce divelog structure
The parser API was very annoying, as a number of tables
to-be-filled were passed in as pointers. The goal of this
commit is to collect all these tables in a single struct.
This should make it (more or less) clear what is actually
written into the divelog files.
Moreover, it should now be rather easy to search for
instances, where the global logfile is accessed (and it
turns out that there are many!).
The divelog struct does not contain the tables as substructs,
but only collects pointers. The idea is that the "divelog.h"
file can be included without all the other files describing
the numerous tables.
To make it easier to use from C++ parts of the code, the
struct implements a constructor and a destructor. Sadly,
we can't use smart pointers, since the pointers are accessed
from C code. Therfore the constructor and destructor are
quite complex.
The whole commit is large, but was mostly an automatic
conversion.
One oddity of note: the divelog structure also contains
the "autogroup" flag, since that is saved in the divelog.
This actually fixes a bug: Before, when importing dives
from a different log, the autogroup flag was overwritten.
This was probably not intended and does not happen anymore.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
2022-11-08 20:31:08 +00:00
|
|
|
for (int i = 0; i < divelog.trips->nr; i++) {
|
|
|
|
if (divelog.trips->trips[i]->id == tripId)
|
|
|
|
return divelog.trips->trips[i];
|
2020-02-20 21:20:03 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-06-28 05:33:51 +00:00
|
|
|
/* Check if two trips overlap time-wise up to trip threshold. */
|
|
|
|
bool trips_overlap(const struct dive_trip *t1, const struct dive_trip *t2)
|
|
|
|
{
|
|
|
|
/* First, handle the empty-trip cases. */
|
|
|
|
if (t1->dives.nr == 0 || t2->dives.nr == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (trip_date(t1) < trip_date(t2))
|
|
|
|
return trip_enddate(t1) + TRIP_THRESHOLD >= trip_date(t2);
|
|
|
|
else
|
|
|
|
return trip_enddate(t2) + TRIP_THRESHOLD >= trip_date(t1);
|
|
|
|
}
|
|
|
|
|
2019-05-31 14:09:14 +00:00
|
|
|
/*
|
|
|
|
* Collect dives for auto-grouping. Pass in first dive which should be checked.
|
|
|
|
* Returns range of dives that should be autogrouped and trip it should be
|
|
|
|
* associated to. If the returned trip was newly allocated, the last bool
|
|
|
|
* is set to true. Caller still has to register it in the system. Note
|
|
|
|
* whereas this looks complicated - it is needed by the undo-system, which
|
|
|
|
* manually injects the new trips. If there are no dives to be autogrouped,
|
|
|
|
* return NULL.
|
|
|
|
*/
|
|
|
|
dive_trip_t *get_dives_to_autogroup(struct dive_table *table, int start, int *from, int *to, bool *allocated)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct dive *lastdive = NULL;
|
|
|
|
|
|
|
|
/* Find first dive that should be merged and remember any previous
|
|
|
|
* dive that could be merged into.
|
|
|
|
*/
|
|
|
|
for (i = start; i < table->nr; i++) {
|
|
|
|
struct dive *dive = table->dives[i];
|
|
|
|
dive_trip_t *trip;
|
|
|
|
|
|
|
|
if (dive->divetrip) {
|
|
|
|
lastdive = dive;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Only consider dives that have not been explicitly removed from
|
|
|
|
* a dive trip by the user. */
|
|
|
|
if (dive->notrip) {
|
|
|
|
lastdive = NULL;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We found a dive, let's see if we have to allocate a new trip */
|
|
|
|
if (!lastdive || dive->when >= lastdive->when + TRIP_THRESHOLD) {
|
|
|
|
/* allocate new trip */
|
|
|
|
trip = create_trip_from_dive(dive);
|
|
|
|
trip->autogen = true;
|
|
|
|
*allocated = true;
|
|
|
|
} else {
|
|
|
|
/* use trip of previous dive */
|
|
|
|
trip = lastdive->divetrip;
|
|
|
|
*allocated = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now, find all dives that will be added to this trip
|
|
|
|
lastdive = dive;
|
|
|
|
*from = i;
|
|
|
|
for (*to = *from + 1; *to < table->nr; (*to)++) {
|
|
|
|
dive = table->dives[*to];
|
|
|
|
if (dive->divetrip || dive->notrip ||
|
|
|
|
dive->when >= lastdive->when + TRIP_THRESHOLD)
|
|
|
|
break;
|
|
|
|
if (get_dive_location(dive) && !trip->location)
|
|
|
|
trip->location = copy_string(get_dive_location(dive));
|
|
|
|
lastdive = dive;
|
|
|
|
}
|
|
|
|
return trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Did not find anyhting - mark as end */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void deselect_dives_in_trip(struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
if (!trip)
|
|
|
|
return;
|
|
|
|
for (int i = 0; i < trip->dives.nr; ++i)
|
|
|
|
deselect_dive(trip->dives.dives[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void select_dives_in_trip(struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
struct dive *dive;
|
|
|
|
if (!trip)
|
|
|
|
return;
|
|
|
|
for (int i = 0; i < trip->dives.nr; ++i) {
|
|
|
|
dive = trip->dives.dives[i];
|
|
|
|
if (!dive->hidden_by_filter)
|
|
|
|
select_dive(dive);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Out of two strings, copy the string that is not empty (if any). */
|
|
|
|
static char *copy_non_empty_string(const char *a, const char *b)
|
|
|
|
{
|
|
|
|
return copy_string(empty_string(b) ? a : b);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This combines the information of two trips, generating a
|
|
|
|
* new trip. To support undo, we have to preserve the old trips. */
|
|
|
|
dive_trip_t *combine_trips(struct dive_trip *trip_a, struct dive_trip *trip_b)
|
|
|
|
{
|
|
|
|
dive_trip_t *trip;
|
|
|
|
|
|
|
|
trip = alloc_trip();
|
|
|
|
trip->location = copy_non_empty_string(trip_a->location, trip_b->location);
|
|
|
|
trip->notes = copy_non_empty_string(trip_a->notes, trip_b->notes);
|
|
|
|
|
|
|
|
return trip;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Trips are compared according to the first dive in the trip. */
|
|
|
|
int comp_trips(const struct dive_trip *a, const struct dive_trip *b)
|
|
|
|
{
|
|
|
|
/* This should never happen, nevertheless don't crash on trips
|
|
|
|
* with no (or worse a negative number of) dives. */
|
|
|
|
if (a->dives.nr <= 0)
|
|
|
|
return b->dives.nr <= 0 ? 0 : -1;
|
|
|
|
if (b->dives.nr <= 0)
|
|
|
|
return 1;
|
|
|
|
return comp_dives(a->dives.dives[0], b->dives.dives[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trip_less_than(const struct dive_trip *a, const struct dive_trip *b)
|
|
|
|
{
|
|
|
|
return comp_trips(a, b) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool is_same_day(timestamp_t trip_when, timestamp_t dive_when)
|
|
|
|
{
|
|
|
|
static timestamp_t twhen = (timestamp_t) 0;
|
|
|
|
static struct tm tmt;
|
|
|
|
struct tm tmd;
|
|
|
|
|
|
|
|
utc_mkdate(dive_when, &tmd);
|
|
|
|
|
|
|
|
if (twhen != trip_when) {
|
|
|
|
twhen = trip_when;
|
|
|
|
utc_mkdate(twhen, &tmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (tmd.tm_mday == tmt.tm_mday) && (tmd.tm_mon == tmt.tm_mon) && (tmd.tm_year == tmt.tm_year);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trip_is_single_day(const struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
if (trip->dives.nr <= 1)
|
|
|
|
return true;
|
|
|
|
return is_same_day(trip->dives.dives[0]->when,
|
|
|
|
trip->dives.dives[trip->dives.nr - 1]->when);
|
|
|
|
}
|
|
|
|
|
|
|
|
int trip_shown_dives(const struct dive_trip *trip)
|
|
|
|
{
|
|
|
|
int res = 0;
|
|
|
|
for (int i = 0; i < trip->dives.nr; ++i) {
|
|
|
|
if (!trip->dives.dives[i]->hidden_by_filter)
|
|
|
|
res++;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|