summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--dive.h2
-rw-r--r--divelist.c74
2 files changed, 74 insertions, 2 deletions
diff --git a/dive.h b/dive.h
index afccfe023..cf3a735fc 100644
--- a/dive.h
+++ b/dive.h
@@ -299,7 +299,7 @@ typedef struct dive_trip {
struct dive *dives;
int nrdives;
int index;
- unsigned expanded:1, selected:1, autogen:1;
+ unsigned expanded:1, selected:1, autogen:1, fixup:1;
struct dive_trip *next;
} dive_trip_t;
diff --git a/divelist.c b/divelist.c
index b5fe48e5b..76cce83f5 100644
--- a/divelist.c
+++ b/divelist.c
@@ -2744,24 +2744,96 @@ static void entry_selected(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *
if (!trip)
return;
trip->selected = 1;
+
/* If this is expanded, let the gtk selection happen for each dive under it */
- if (gtk_tree_view_row_expanded(GTK_TREE_VIEW(dive_list.tree_view), path))
+ if (gtk_tree_view_row_expanded(GTK_TREE_VIEW(dive_list.tree_view), path)) {
+ trip->fixup = 1;
return;
+ }
+
/* Otherwise, consider each dive under it selected */
for_each_dive(i, dive) {
if (dive->divetrip == trip)
select_dive(i);
}
+ trip->fixup = 0;
} else {
select_dive(idx);
}
}
+static void update_gtk_selection(GtkTreeSelection *selection, GtkTreeModel *model)
+{
+ GtkTreeIter iter;
+
+ if (!gtk_tree_model_get_iter_first(model, &iter))
+ return;
+ do {
+ GtkTreeIter child;
+
+ if (!gtk_tree_model_iter_children(model, &child, &iter))
+ continue;
+
+ do {
+ int idx;
+ struct dive *dive;
+ dive_trip_t *trip;
+
+ gtk_tree_model_get(model, &child, DIVE_INDEX, &idx, -1);
+ dive = get_dive(idx);
+ if (!dive || !dive->selected)
+ break;
+ trip = dive->divetrip;
+ if (!trip)
+ break;
+ gtk_tree_selection_select_iter(selection, &child);
+ } while (gtk_tree_model_iter_next(model, &child));
+ } while (gtk_tree_model_iter_next(model, &iter));
+}
+
/* this is called when gtk thinks that the selection has changed */
static void selection_cb(GtkTreeSelection *selection, GtkTreeModel *model)
{
+ int i, fixup;
+ struct dive *dive;
+
gtk_tree_selection_selected_foreach(selection, entry_selected, model);
+ /*
+ * Go through all the dives, if there is a trip that is selected but no
+ * dives under it are selected, force-select all the dives
+ */
+
+ /* First, clear "fixup" for any trip that has selected dives */
+ for_each_dive(i, dive) {
+ dive_trip_t *trip = dive->divetrip;
+ if (!trip || !trip->fixup)
+ continue;
+ if (dive->selected || !trip->selected)
+ trip->fixup = 0;
+ }
+
+ /*
+ * Ok, not fixup is only set for trips that are selected
+ * but have no selected dives in them. Select all dives
+ * for such trips.
+ */
+ fixup = 0;
+ for_each_dive(i, dive) {
+ dive_trip_t *trip = dive->divetrip;
+ if (!trip || !trip->fixup)
+ continue;
+ fixup = 1;
+ select_dive(i);
+ }
+
+ /*
+ * Ok, we did a forced selection of dives, now we need to update the gtk
+ * view of what is selected too..
+ */
+ if (fixup)
+ update_gtk_selection(selection, model);
+
#if DEBUG_SELECTION_TRACKING
dump_selection();
#endif