2 /* this creates the UI for the dive list -
3 * controlled through the following interfaces:
5 * void flush_divelist(struct dive *dive)
6 * GtkWidget dive_list_create(void)
7 * void dive_list_update_dives(void)
8 * void update_dive_list_units(void)
9 * void set_divelist_font(const char *font)
10 * void mark_divelist_changed(int changed)
11 * int unsaved_changes()
22 #include "display-gtk.h"
26 GtkWidget *container_widget;
28 GtkTreeViewColumn *date, *depth, *duration, *location;
29 GtkTreeViewColumn *temperature, *cylinder, *nitrox, *sac;
33 static struct DiveList dive_list;
36 * The dive list has the dive data in both string format (for showing)
37 * and in "raw" format (for sorting purposes)
41 DIVE_DATE, /* time_t: dive->when */
42 DIVE_DEPTH, /* int: dive->maxdepth in mm */
43 DIVE_DURATION, /* int: in seconds */
44 DIVE_TEMPERATURE, /* int: in mkelvin */
46 DIVE_NITROX, /* int: in permille */
47 DIVE_SAC, /* int: in ml/min */
48 DIVE_LOCATION, /* "2nd Cathedral, Lanai" */
52 static void selection_cb(GtkTreeSelection *selection, GtkTreeModel *model)
57 if (!gtk_tree_selection_get_selected(selection, NULL, &iter))
60 gtk_tree_model_get_value(model, &iter, DIVE_INDEX, &value);
61 selected_dive = g_value_get_int(&value);
65 static void date_data_func(GtkTreeViewColumn *col,
66 GtkCellRenderer *renderer,
76 gtk_tree_model_get(model, iter, DIVE_DATE, &val, -1);
82 snprintf(buffer, sizeof(buffer),
83 "%s, %s %d, %d %02d:%02d",
85 monthname(tm->tm_mon),
86 tm->tm_mday, tm->tm_year + 1900,
87 tm->tm_hour, tm->tm_min);
88 g_object_set(renderer, "text", buffer, NULL);
91 static void depth_data_func(GtkTreeViewColumn *col,
92 GtkCellRenderer *renderer,
97 int depth, integer, frac, len;
100 gtk_tree_model_get(model, iter, DIVE_DEPTH, &depth, -1);
102 switch (output_units.length) {
104 /* To tenths of meters */
105 depth = (depth + 49) / 100;
106 integer = depth / 10;
114 integer = mm_to_feet(depth) + 0.5;
120 len = snprintf(buffer, sizeof(buffer), "%d", integer);
122 len += snprintf(buffer+len, sizeof(buffer)-len, ".%d", frac);
124 g_object_set(renderer, "text", buffer, NULL);
127 static void duration_data_func(GtkTreeViewColumn *col,
128 GtkCellRenderer *renderer,
136 gtk_tree_model_get(model, iter, DIVE_DURATION, &sec, -1);
137 snprintf(buffer, sizeof(buffer), "%d:%02d", sec / 60, sec % 60);
139 g_object_set(renderer, "text", buffer, NULL);
142 static void temperature_data_func(GtkTreeViewColumn *col,
143 GtkCellRenderer *renderer,
151 gtk_tree_model_get(model, iter, DIVE_TEMPERATURE, &value, -1);
156 switch (output_units.temperature) {
158 deg = mkelvin_to_C(value);
161 deg = mkelvin_to_F(value);
166 snprintf(buffer, sizeof(buffer), "%.1f", deg);
169 g_object_set(renderer, "text", buffer, NULL);
172 static void nitrox_data_func(GtkTreeViewColumn *col,
173 GtkCellRenderer *renderer,
181 gtk_tree_model_get(model, iter, DIVE_NITROX, &value, -1);
184 snprintf(buffer, sizeof(buffer), "%.1f", value/10.0);
186 strcpy(buffer, "air");
188 g_object_set(renderer, "text", buffer, NULL);
191 /* Render the SAC data (integer value of "ml / min") */
192 static void sac_data_func(GtkTreeViewColumn *col,
193 GtkCellRenderer *renderer,
199 const double liters_per_cuft = 28.317;
204 gtk_tree_model_get(model, iter, DIVE_SAC, &value, -1);
207 g_object_set(renderer, "text", "", NULL);
211 sac = value / 1000.0;
212 switch (output_units.volume) {
218 sac /= liters_per_cuft;
221 snprintf(buffer, sizeof(buffer), fmt, sac);
223 g_object_set(renderer, "text", buffer, NULL);
226 /* calculate OTU for a dive */
227 static double calculate_otu(struct dive *dive)
232 for (i = 1; i < dive->samples; i++) {
235 struct sample *sample = dive->sample + i;
236 struct sample *psample = sample - 1;
237 t = sample->time.seconds - psample->time.seconds;
238 po2 = dive->cylinder[sample->cylinderindex].gasmix.o2.permille / 1000.0 *
239 (sample->depth.mm + 10000) / 10000.0;
241 otu += pow(po2 - 0.5, 0.83) * t / 30.0;
246 * Return air usage (in liters).
248 static double calculate_airuse(struct dive *dive)
253 for (i = 0; i < MAX_CYLINDERS; i++) {
254 cylinder_t *cyl = dive->cylinder + i;
255 int size = cyl->type.size.mliter;
261 kilo_atm = (cyl->start.mbar - cyl->end.mbar) / 1013250.0;
263 /* Liters of air at 1 atm == milliliters at 1k atm*/
264 airuse += kilo_atm * size;
269 static void get_sac(struct dive *dive, int *val)
271 double airuse, pressure, sac;
274 airuse = calculate_airuse(dive);
277 if (!dive->duration.seconds)
280 /* Mean pressure in atm: 1 atm per 10m */
281 pressure = 1 + (dive->meandepth.mm / 10000.0);
282 sac = airuse / pressure * 60 / dive->duration.seconds;
284 /* milliliters per minute.. */
288 static void get_string(char **str, const char *s)
304 static void get_location(struct dive *dive, char **str)
306 get_string(str, dive->location);
309 static void get_cylinder(struct dive *dive, char **str)
311 get_string(str, dive->cylinder[0].type.description);
314 static void fill_one_dive(struct dive *dive,
319 char *location, *cylinder;
321 get_cylinder(dive, &cylinder);
322 get_location(dive, &location);
326 * We only set the fields that changed: the strings.
327 * The core data itself is unaffected by units
329 gtk_list_store_set(GTK_LIST_STORE(model), iter,
330 DIVE_LOCATION, location,
331 DIVE_CYLINDER, cylinder,
336 static gboolean set_one_dive(GtkTreeModel *model,
341 GValue value = {0, };
344 /* Get the dive number */
345 gtk_tree_model_get_value(model, iter, DIVE_INDEX, &value);
346 dive = get_dive(g_value_get_int(&value));
349 if (data && dive != data)
352 fill_one_dive(dive, model, iter);
356 void flush_divelist(struct dive *dive)
358 GtkTreeModel *model = GTK_TREE_MODEL(dive_list.model);
360 gtk_tree_model_foreach(model, set_one_dive, dive);
363 void set_divelist_font(const char *font)
365 PangoFontDescription *font_desc = pango_font_description_from_string(font);
366 gtk_widget_modify_font(dive_list.tree_view, font_desc);
367 pango_font_description_free(font_desc);
370 void update_dive_list_units(void)
373 GtkTreeModel *model = GTK_TREE_MODEL(dive_list.model);
375 switch (output_units.length) {
383 gtk_tree_view_column_set_title(dive_list.depth, unit);
385 switch (output_units.temperature) {
387 unit = UTF8_DEGREE "C";
390 unit = UTF8_DEGREE "F";
396 gtk_tree_view_column_set_title(dive_list.temperature, unit);
398 gtk_tree_model_foreach(model, set_one_dive, NULL);
401 static void fill_dive_list(void)
407 store = GTK_LIST_STORE(dive_list.model);
409 for (i = 0; i < dive_table.nr; i++) {
410 struct dive *dive = dive_table.dives[i];
412 dive->otu = calculate_otu(dive);
413 gtk_list_store_append(store, &iter);
414 gtk_list_store_set(store, &iter,
416 DIVE_DATE, dive->when,
417 DIVE_DEPTH, dive->maxdepth,
418 DIVE_DURATION, dive->duration.seconds,
419 DIVE_LOCATION, "location",
420 DIVE_TEMPERATURE, dive->watertemp.mkelvin,
421 DIVE_NITROX, dive->cylinder[0].gasmix.o2,
426 update_dive_list_units();
427 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(dive_list.model), &iter)) {
428 GtkTreeSelection *selection;
429 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(dive_list.tree_view));
430 gtk_tree_selection_select_iter(selection, &iter);
434 void dive_list_update_dives(void)
436 gtk_list_store_clear(GTK_LIST_STORE(dive_list.model));
441 typedef void (*data_func_t)(GtkTreeViewColumn *col,
442 GtkCellRenderer *renderer,
447 static GtkTreeViewColumn *divelist_column(struct DiveList *dl, int index, const char *title,
448 data_func_t data_func, PangoAlignment align)
450 GtkCellRenderer *renderer;
451 GtkTreeViewColumn *col;
452 double xalign = 0.0; /* left as default */
454 renderer = gtk_cell_renderer_text_new();
455 col = gtk_tree_view_column_new();
457 gtk_tree_view_column_set_title(col, title);
458 gtk_tree_view_column_set_sort_column_id(col, index);
459 gtk_tree_view_column_set_resizable(col, TRUE);
460 gtk_tree_view_column_pack_start(col, renderer, TRUE);
462 gtk_tree_view_column_set_cell_data_func(col, renderer, data_func, NULL, NULL);
464 gtk_tree_view_column_add_attribute(col, renderer, "text", index);
465 gtk_object_set(GTK_OBJECT(renderer), "alignment", align, NULL);
467 case PANGO_ALIGN_LEFT:
470 case PANGO_ALIGN_CENTER:
473 case PANGO_ALIGN_RIGHT:
477 gtk_cell_renderer_set_alignment(GTK_CELL_RENDERER(renderer), xalign, 0.5);
478 gtk_tree_view_append_column(GTK_TREE_VIEW(dl->tree_view), col);
483 * This is some crazy crap. The only way to get default focus seems
484 * to be to grab focus as the widget is being shown the first time.
486 static void realize_cb(GtkWidget *tree_view, gpointer userdata)
488 gtk_widget_grab_focus(tree_view);
491 GtkWidget *dive_list_create(void)
493 GtkTreeSelection *selection;
495 dive_list.model = gtk_list_store_new(DIVELIST_COLUMNS,
496 G_TYPE_INT, /* index */
497 G_TYPE_INT, /* Date */
498 G_TYPE_INT, /* Depth */
499 G_TYPE_INT, /* Duration */
500 G_TYPE_INT, /* Temperature */
501 G_TYPE_STRING, /* Cylinder */
502 G_TYPE_INT, /* Nitrox */
503 G_TYPE_INT, /* SAC */
504 G_TYPE_STRING /* Location */
506 dive_list.tree_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(dive_list.model));
507 set_divelist_font(divelist_font);
509 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(dive_list.tree_view));
511 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_BROWSE);
512 gtk_widget_set_size_request(dive_list.tree_view, 200, 200);
514 dive_list.date = divelist_column(&dive_list, DIVE_DATE, "Date", date_data_func, PANGO_ALIGN_LEFT);
515 dive_list.depth = divelist_column(&dive_list, DIVE_DEPTH, "ft", depth_data_func, PANGO_ALIGN_RIGHT);
516 dive_list.duration = divelist_column(&dive_list, DIVE_DURATION, "min", duration_data_func, PANGO_ALIGN_RIGHT);
517 dive_list.temperature = divelist_column(&dive_list, DIVE_TEMPERATURE, UTF8_DEGREE "F", temperature_data_func, PANGO_ALIGN_RIGHT);
518 dive_list.cylinder = divelist_column(&dive_list, DIVE_CYLINDER, "Cyl", NULL, PANGO_ALIGN_CENTER);
519 dive_list.nitrox = divelist_column(&dive_list, DIVE_NITROX, "O" UTF8_SUBSCRIPT_2 "%", nitrox_data_func, PANGO_ALIGN_CENTER);
520 dive_list.sac = divelist_column(&dive_list, DIVE_SAC, "SAC", sac_data_func, PANGO_ALIGN_CENTER);
521 dive_list.location = divelist_column(&dive_list, DIVE_LOCATION, "Location", NULL, PANGO_ALIGN_LEFT);
525 g_object_set(G_OBJECT(dive_list.tree_view), "headers-visible", TRUE,
526 "search-column", DIVE_LOCATION,
530 g_signal_connect_after(dive_list.tree_view, "realize", G_CALLBACK(realize_cb), NULL);
531 g_signal_connect(selection, "changed", G_CALLBACK(selection_cb), dive_list.model);
533 dive_list.container_widget = gtk_scrolled_window_new(NULL, NULL);
534 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(dive_list.container_widget),
535 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
536 gtk_container_add(GTK_CONTAINER(dive_list.container_widget), dive_list.tree_view);
538 dive_list.changed = 0;
540 return dive_list.container_widget;
543 void mark_divelist_changed(int changed)
545 dive_list.changed = changed;
548 int unsaved_changes()
550 return dive_list.changed;