summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--dive.h1
-rw-r--r--divelist.c209
2 files changed, 109 insertions, 101 deletions
diff --git a/dive.h b/dive.h
index 4d736b119..d05f1d1b4 100644
--- a/dive.h
+++ b/dive.h
@@ -246,6 +246,7 @@ typedef struct dive_trip {
timestamp_t when_from_file;
char *location;
char *notes;
+ int expanded:1, selected:1;
} dive_trip_t;
struct dive {
diff --git a/divelist.c b/divelist.c
index ce9796d82..4dcf605bc 100644
--- a/divelist.c
+++ b/divelist.c
@@ -142,28 +142,6 @@ static void first_leaf(GtkTreeModel *model, GtkTreeIter *iter, int *diveidx)
}
}
-static GtkTreePath *path_match;
-
-static gboolean match_dive(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
-{
- int idx;
- struct dive *dive = data;
-
- gtk_tree_model_get(model, iter, DIVE_INDEX, &idx, -1);
- if (idx >= 0)
- if (get_dive(idx)->when == dive->when) {
- path_match = gtk_tree_path_copy(path);
- return TRUE;
- }
- return FALSE;
-}
-
-static GtkTreePath *get_path_from(struct dive *dive)
-{
- gtk_tree_model_foreach(TREEMODEL(dive_list), match_dive, dive);
- return path_match;
-}
-
static struct dive *dive_from_path(GtkTreePath *path)
{
GtkTreeIter iter;
@@ -1842,85 +1820,120 @@ void merge_trips_cb(GtkWidget *menuitem, GtkTreePath *trippath)
mark_divelist_changed(TRUE);
}
-/* delete a dive by passing a tree iterator */
-static void delete_single_dive(GtkTreeIter *iter)
+/* this implements the mechanics of removing the dive from the table,
+ * but doesn't deal with updating dive trips, etc */
+static void delete_single_dive(int idx)
{
- int i, idx;
- struct dive *dive, *pdive, *ndive;
- GtkTreeIter parent;
-
- gtk_tree_model_get(MODEL(dive_list), iter, DIVE_INDEX, &idx, -1);
- dive = get_dive(idx);
-
- if (dive->divetrip) {
- /* we could be displaying the list model, in which case we can't find out
- * if this is part of a trip and the only dive in that trip from the model,
- * so let's do this the manual way */
- pdive = get_dive(idx - 1);
- ndive = get_dive(idx + 1);
- if (! (pdive && pdive->divetrip == dive->divetrip) &&
- ! (ndive && ndive->divetrip == dive->divetrip)) {
- /* if this is the only dive in the trip, remove the trip - the
- * dive list update below will deal with making sure the treemodel
- * is correct */
- GList *trip = find_matching_trip(dive->when);
- delete_trip(trip);
- free(dive->divetrip);
- dive->divetrip = NULL;
- }
- }
-
- /* simply remove the dive and recreate the divelist
- * (we can't just manipulate the tree_view as the indices for dives change) */
+ int i;
+ struct dive *dive = get_dive(idx);
+ if (!dive)
+ return; /* this should never happen */
for (i = idx; i < dive_table.nr - 1; i++)
dive_table.dives[i] = dive_table.dives[i+1];
dive_table.nr--;
if (dive->selected)
amount_selected--;
-
- /* once the dive is deleted, update the 'when' flag of the trip it was part of
- * to the 'when' flag of the earliest dive left in the same trip */
- if (dive->divetrip) {
- gtk_tree_model_iter_parent(MODEL(dive_list), &parent, iter);
- gtk_tree_store_remove(STORE(dive_list), iter);
- update_trip_timestamp(&parent, dive->divetrip);
- }
-
free(dive);
}
-/* workaround for not using gtk_tree_selection_get_selected_rows() or modifying the tree
- * directly from the gtk_tree_selection_selected_foreach() callback function */
-struct tree_selected_st {
- GtkTreeIter *list;
- int total;
-};
+/* remember expanded state */
+static void remember_tree_state()
+{
+ GtkTreeIter iter;
+ if (!gtk_tree_model_get_iter_first(TREEMODEL(dive_list), &iter))
+ return;
+ do {
+ int idx;
+ timestamp_t when;
+ GtkTreePath *path;
+
+ gtk_tree_model_get(TREEMODEL(dive_list), &iter,
+ DIVE_INDEX, &idx, DIVE_DATE, &when, -1);
+ if (idx >= 0)
+ continue;
+ path = gtk_tree_model_get_path(TREEMODEL(dive_list), &iter);
+ if (gtk_tree_view_row_expanded(GTK_TREE_VIEW(dive_list.tree_view), path))
+ DIVE_TRIP(find_trip(when))->expanded = TRUE;
+ } while (gtk_tree_model_iter_next(TREEMODEL(dive_list), &iter));
+}
-static void tree_selected_foreach(GtkTreeModel *model, GtkTreePath *path,
- GtkTreeIter *iter, gpointer userdata)
+static gboolean restore_node_state(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
{
- struct tree_selected_st *st = (struct tree_selected_st *)userdata;
+ int idx;
+ timestamp_t when;
+ struct dive *dive;
+ GtkTreeView *tree_view = GTK_TREE_VIEW(dive_list.tree_view);
+ GtkTreeSelection *selection = gtk_tree_view_get_selection(tree_view);
+
+ gtk_tree_model_get(model, iter, DIVE_INDEX, &idx, DIVE_DATE, &when, -1);
+ if (idx < 0) {
+ if (DIVE_TRIP(find_trip(when))->expanded)
+ gtk_tree_view_expand_row(tree_view, path, FALSE);
+ if (DIVE_TRIP(find_trip(when))->selected)
+ gtk_tree_selection_select_iter(selection, iter);
+ } else {
+ dive = get_dive(idx);
+ if (dive->selected)
+ gtk_tree_selection_select_iter(selection, iter);
+ }
+ /* continue foreach */
+ return FALSE;
+}
- st->total++;
- st->list = (GtkTreeIter *)realloc(st->list, sizeof(GtkTreeIter) * st->total);
- memcpy(&st->list[st->total - 1], iter, sizeof(GtkTreeIter));
+/* restore expanded and selected state */
+static void restore_tree_state()
+{
+ gtk_tree_model_foreach(MODEL(dive_list), restore_node_state, NULL);
}
/* called when multiple dives are selected and one of these is right-clicked for delete */
static void delete_selected_dives_cb(GtkWidget *menuitem, GtkTreePath *path)
{
int i;
- GtkTreeView *tree_view = GTK_TREE_VIEW(dive_list.tree_view);
- GtkTreeSelection *selection = gtk_tree_view_get_selection(tree_view);
- struct tree_selected_st st = {NULL, 0};
-
- gtk_tree_selection_selected_foreach(selection, tree_selected_foreach, &st);
+ gboolean divetrip_needs_update = FALSE;
+ dive_trip_t *divetrip_to_update = NULL;
+ struct dive *dive;
- for (i = 0; i < st.total; i++)
- delete_single_dive(&st.list[i]);
- free(st.list);
+ remember_tree_state();
+ /* walk the dive list in chronological order */
+ for (i = 0; i < dive_table.nr; i++) {
+ dive = get_dive(i);
+ if (!dive)
+ continue;
+ if (! dive->selected) {
+ if (divetrip_needs_update) {
+ divetrip_needs_update = FALSE;
+ /* this is the first unselected dive since we found the
+ * trip that needed to be updated; if we are still on the
+ * same trip, update the time, else delete the trip */
+ if (dive->divetrip == divetrip_to_update)
+ divetrip_to_update->when = dive->when;
+ else
+ delete_trip(find_trip(divetrip_to_update->when));
+ }
+ continue;
+ }
+ /* this is a selected (i.e., to be deleted) dive; if we have a divetrip
+ * that needs to be updated, check if this dive is still in that trip;
+ * if not, delete the trip */
+ if (divetrip_needs_update && dive->divetrip != divetrip_to_update) {
+ delete_trip(find_trip(divetrip_to_update->when));
+ divetrip_needs_update = FALSE;
+ }
+ /* if this dive is part of a divetrip and is the first dive that
+ * determines the timestamp, then remember that divetrip for updating */
+ if (dive->divetrip && dive->when == dive->divetrip->when) {
+ divetrip_needs_update = TRUE;
+ divetrip_to_update = dive->divetrip;
+ }
+ /* now remove the dive from the table and free it. also move the iterator back,
+ * so that we don't skip a dive */
+ delete_single_dive(i);
+ i--;
+ }
dive_list_update_dives();
+ restore_tree_state();
mark_divelist_changed(TRUE);
}
@@ -1928,32 +1941,26 @@ static void delete_selected_dives_cb(GtkWidget *menuitem, GtkTreePath *path)
* or as part of a trip */
static void delete_dive_cb(GtkWidget *menuitem, GtkTreePath *path)
{
- GtkTreeIter iter;
- int i, idx;
+ int idx;
struct dive *dive;
- GtkTreeView *tree_view = GTK_TREE_VIEW(dive_list.tree_view);
- GtkTreeSelection *selection = gtk_tree_view_get_selection(tree_view);
-
- gtk_tree_model_get_iter(MODEL(dive_list), &iter, path);
- delete_single_dive(&iter);
+ GtkTreeIter iter;
+ remember_tree_state();
+ if (!gtk_tree_model_get_iter(MODEL(dive_list), &iter, path))
+ return;
gtk_tree_model_get(MODEL(dive_list), &iter, DIVE_INDEX, &idx, -1);
dive = get_dive(idx);
-
- /* now make sure the correct dive list is displayed, the same
- * dives stay selected and if necessary their trips are
- * expanded. If no selected dives are left then fill_dive_list()
- * (which gets called from dive_list_update_dives()) will once
- * again select the last dive in the dive list */
- dive_list_update_dives();
- for_each_dive(i, dive) {
- if (dive->selected) {
- GtkTreePath *path = get_path_from(dive);
- if (MODEL(dive_list) == TREEMODEL(dive_list))
- gtk_tree_view_expand_to_path(tree_view, path);
- gtk_tree_selection_select_path(selection, path);
- }
+ /* do we need to update the trip this dive is part of? */
+ if (dive->divetrip && dive->when == dive->divetrip->when) {
+ struct dive *next_dive = get_dive(idx + 1);
+ if (!next_dive || next_dive->divetrip != dive->divetrip)
+ delete_trip(find_trip(dive->when));
+ else
+ next_dive->divetrip->when = next_dive->when;
}
+ delete_single_dive(idx);
+ dive_list_update_dives();
+ restore_tree_state();
mark_divelist_changed(TRUE);
}