mirror of
https://github.com/subsurface/subsurface.git
synced 2024-11-28 05:00:20 +00:00
Dive sites: don't delete unused dive sites on save
Unused dive sites were deleted on save. This clashed with the undo system in the following scenario: 1) Delete single-use dive site. 2) Save (dive site deleted) 3) Undo (reference to freed dive site) Therefore, as a quick-fix, keep the referenced dive site around. Note that this also means that empty dive sites must not be deleted, as it might refer to a dive in the undo system. Instead only clear references to empty dive sites in the global dive table. Factor this functionality out, as it was common to the XML and git savers. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
This commit is contained in:
parent
da52440963
commit
4d06ddd723
4 changed files with 24 additions and 16 deletions
|
@ -346,6 +346,23 @@ struct dive_site *find_or_create_dive_site_with_name(const char *name, timestamp
|
|||
return create_dive_site(name, divetime);
|
||||
}
|
||||
|
||||
void purge_empty_dive_sites()
|
||||
{
|
||||
int i, j;
|
||||
struct dive *d;
|
||||
struct dive_site *ds;
|
||||
|
||||
for (i = 0; i < dive_site_table.nr; i++) {
|
||||
ds = get_dive_site(i);
|
||||
if (!dive_site_is_empty(ds))
|
||||
continue;
|
||||
for_each_dive(j, d) {
|
||||
if (d->dive_site == ds)
|
||||
d->dive_site = NULL;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int compare_sites(const void *_a, const void *_b)
|
||||
{
|
||||
const struct dive_site *a = (const struct dive_site *)*(void **)_a;
|
||||
|
|
|
@ -73,6 +73,7 @@ void clear_dive_site(struct dive_site *ds);
|
|||
unsigned int get_distance(const location_t *loc1, const location_t *loc2);
|
||||
struct dive_site *find_or_create_dive_site_with_name(const char *name, timestamp_t divetime);
|
||||
void merge_dive_sites(struct dive_site *ref, struct dive_site *dive_sites[], int count);
|
||||
void purge_empty_dive_sites();
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
|
|
|
@ -879,18 +879,12 @@ static void save_divesites(git_repository *repo, struct dir *tree)
|
|||
subdir = new_directory(repo, tree, &dirname);
|
||||
free_buffer(&dirname);
|
||||
|
||||
purge_empty_dive_sites();
|
||||
for (int i = 0; i < dive_site_table.nr; i++) {
|
||||
struct membuffer b = { 0 };
|
||||
struct dive_site *ds = get_dive_site(i);
|
||||
if (dive_site_is_empty(ds) || !is_dive_site_used(ds, false)) {
|
||||
int j;
|
||||
struct dive *d;
|
||||
for_each_dive(j, d) {
|
||||
if (d->dive_site == ds)
|
||||
d->dive_site = NULL;
|
||||
}
|
||||
delete_dive_site(ds);
|
||||
i--; // since we just deleted that one
|
||||
if (!is_dive_site_used(ds, false)) {
|
||||
/* Only write used dive sites */
|
||||
continue;
|
||||
} else if (ds->name &&
|
||||
(strncmp(ds->name, "Auto-created dive", 17) == 0 ||
|
||||
|
|
|
@ -592,18 +592,14 @@ void save_dives_buffer(struct membuffer *b, const bool select_only, bool anonymi
|
|||
|
||||
/* save the dive sites - to make the output consistent let's sort the table, first */
|
||||
dive_site_table_sort();
|
||||
purge_empty_dive_sites();
|
||||
put_format(b, "<divesites>\n");
|
||||
for (i = 0; i < dive_site_table.nr; i++) {
|
||||
int j;
|
||||
struct dive *d;
|
||||
struct dive_site *ds = get_dive_site(i);
|
||||
if (dive_site_is_empty(ds) || !is_dive_site_used(ds, false)) {
|
||||
for_each_dive(j, d) {
|
||||
if (d->dive_site == ds)
|
||||
d->dive_site = NULL;
|
||||
}
|
||||
delete_dive_site(ds);
|
||||
i--; // since we just deleted that one
|
||||
if (!is_dive_site_used(ds, false)) {
|
||||
/* Only write used dive sites */
|
||||
continue;
|
||||
} else if (ds->name &&
|
||||
(strncmp(ds->name, "Auto-created dive", 17) == 0 ||
|
||||
|
|
Loading…
Reference in a new issue