/* * viking -- GPS Data and Topo Analyzer, Explorer, and Manager * * Copyright (C) 2003-2005, Evan Battaglia * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include #include #include "viking.h" #include "babel.h" #include "gpx.h" #include "acquire.h" #if GTK_CHECK_VERSION(2,6,0) #define USE_NEW_COMBO_BOX #endif static gboolean gps_acquire_in_progress = FALSE; static gpointer datasource_gps_init_func ( ); static void datasource_gps_get_cmd_string ( gpointer add_widgets_data_not_used, gchar **babelargs, gchar **input_file ); static void datasource_gps_cleanup ( gpointer user_data ); static void datasource_gps_progress ( BabelProgressCode c, gpointer data, acq_dialog_widgets_t *w ); static void datasource_gps_add_setup_widgets ( GtkWidget *dialog, VikViewport *vvp, gpointer user_data ); static void datasource_gps_add_progress_widgets ( GtkWidget *dialog, gpointer user_data ); VikDataSourceInterface vik_datasource_gps_interface = { "Acquire from GPS", "Acquired from GPS", VIK_DATASOURCE_GPSBABEL_DIRECT, VIK_DATASOURCE_CREATENEWLAYER, (VikDataSourceInitFunc) datasource_gps_init_func, (VikDataSourceCheckExistenceFunc) NULL, (VikDataSourceAddSetupWidgetsFunc) datasource_gps_add_setup_widgets, (VikDataSourceGetCmdStringFunc) datasource_gps_get_cmd_string, (VikDataSourceProgressFunc) datasource_gps_progress, (VikDataSourceAddProgressWidgetsFunc) datasource_gps_add_progress_widgets, (VikDataSourceCleanupFunc) datasource_gps_cleanup }; /********************************************************* * Definitions and routines for acquiring data from GPS *********************************************************/ /* widgets in setup dialog specific to GPS */ /* widgets in progress dialog specific to GPS */ /* also counts needed for progress */ typedef struct { /* setup dialog */ GtkWidget *proto_l; GtkComboBox *proto_b; GtkWidget *ser_l; GtkComboBox *ser_b; /* progress dialog */ GtkWidget *gps_label; GtkWidget *ver_label; GtkWidget *id_label; GtkWidget *wp_label; GtkWidget *trk_label; GtkWidget *progress_label; /* state */ int total_count; int count; } gps_user_data_t; static gpointer datasource_gps_init_func () { return g_malloc (sizeof(gps_user_data_t)); } static void datasource_gps_get_cmd_string ( gpointer user_data, gchar **babelargs, gchar **input_file ) { char *proto = NULL; char *ser = NULL; char *device = NULL; #ifndef USE_NEW_COMBO_BOX GtkTreeIter iter; #endif gps_user_data_t *w = (gps_user_data_t *)user_data; if (gps_acquire_in_progress) { *babelargs = *input_file = NULL; } gps_acquire_in_progress = TRUE; #ifdef USE_NEW_COMBO_BOX proto = gtk_combo_box_get_active_text(GTK_COMBO_BOX(w->proto_b)); #else proto = gtk_combo_box_get_active_iter(GTK_COMBO_BOX(w->proto_b),&iter); #endif if (!strcmp(proto, "Garmin")) { device = "garmin"; } else { device = "magellan"; } *babelargs = g_strdup_printf("-D 9 -t -w -i %s", device); /* device points to static content => no free */ device = NULL; /* Old stuff */ #ifdef USE_NEW_COMBO_BOX ser = gtk_combo_box_get_active_text(GTK_COMBO_BOX(w->ser_b)); #else ser = gtk_combo_box_get_active_iter(GTK_COMBO_BOX(w->ser_b),&iter); #endif *input_file = g_strdup(ser); g_debug("using cmdline '%s' and file '%s'\n", *babelargs, *input_file); } static void datasource_gps_cleanup ( gpointer user_data ) { g_free ( user_data ); gps_acquire_in_progress = FALSE; } static void set_total_count(gint cnt, acq_dialog_widgets_t *w) { gchar *s = NULL; gdk_threads_enter(); if (w->ok) { gps_user_data_t *gps_data = (gps_user_data_t *)w->user_data; s = g_strdup_printf("Downloading %d %s...", cnt, (gps_data->progress_label == gps_data->wp_label) ? "waypoints" : "trackpoints"); gtk_label_set_text ( GTK_LABEL(gps_data->progress_label), s ); gtk_widget_show ( gps_data->progress_label ); gps_data->total_count = cnt; } g_free(s); s = NULL; gdk_threads_leave(); } static void set_current_count(gint cnt, acq_dialog_widgets_t *w) { gchar *s = NULL; gdk_threads_enter(); if (w->ok) { gps_user_data_t *gps_data = (gps_user_data_t *)w->user_data; if (cnt < gps_data->total_count) { s = g_strdup_printf("Downloaded %d out of %d %s...", cnt, gps_data->total_count, (gps_data->progress_label == gps_data->wp_label) ? "waypoints" : "trackpoints"); } else { s = g_strdup_printf("Downloaded %d %s.", cnt, (gps_data->progress_label == gps_data->wp_label) ? "waypoints" : "trackpoints"); } gtk_label_set_text ( GTK_LABEL(gps_data->progress_label), s ); } g_free(s); s = NULL; gdk_threads_leave(); } static void set_gps_info(const gchar *info, acq_dialog_widgets_t *w) { gchar *s = NULL; gdk_threads_enter(); if (w->ok) { s = g_strdup_printf("GPS Device: %s", info); gtk_label_set_text ( GTK_LABEL(((gps_user_data_t *)w->user_data)->gps_label), s ); } g_free(s); s = NULL; gdk_threads_leave(); } /* * This routine relies on gpsbabel's diagnostic output to display the progress information. * These outputs differ when different GPS devices are used, so we will need to test * them on several and add the corresponding support. */ static void datasource_gps_progress ( BabelProgressCode c, gpointer data, acq_dialog_widgets_t *w ) { gchar *line; gps_user_data_t *gps_data = (gps_user_data_t *)w->user_data; switch(c) { case BABEL_DIAG_OUTPUT: line = (gchar *)data; /* tells us how many items there will be */ if (strstr(line, "Xfer Wpt")) { gps_data->progress_label = gps_data->wp_label; } if (strstr(line, "Xfer Trk")) { gps_data->progress_label = gps_data->trk_label; } if (strstr(line, "PRDDAT")) { gchar **tokens = g_strsplit(line, " ", 0); gchar info[128]; int ilen = 0; int i; int n_tokens = 0; while (tokens[n_tokens]) n_tokens++; if (n_tokens > 8) { for (i=8; tokens[i] && ilen < sizeof(info)-2 && strcmp(tokens[i], "00"); i++) { guint ch; sscanf(tokens[i], "%x", &ch); info[ilen++] = ch; } info[ilen++] = 0; set_gps_info(info, w); } g_strfreev(tokens); } if (strstr(line, "RECORD")) { int lsb, msb, cnt; if (strlen(line) > 20) { sscanf(line+17, "%x", &lsb); sscanf(line+20, "%x", &msb); cnt = lsb + msb * 256; set_total_count(cnt, w); gps_data->count = 0; } } if ( strstr(line, "WPTDAT") || strstr(line, "TRKHDR") || strstr(line, "TRKDAT") ) { gps_data->count++; set_current_count(gps_data->count, w); } break; case BABEL_DONE: break; default: break; } } void datasource_gps_add_setup_widgets ( GtkWidget *dialog, VikViewport *vvp, gpointer user_data ) { gps_user_data_t *w = (gps_user_data_t *)user_data; GtkTable* box; w->proto_l = gtk_label_new ("GPS Protocol:"); w->proto_b = GTK_COMBO_BOX(gtk_combo_box_new_text ()); gtk_combo_box_append_text (w->proto_b, "Garmin"); gtk_combo_box_append_text (w->proto_b, "Magellan"); gtk_combo_box_set_active (w->proto_b, 0); g_object_ref(w->proto_b); w->ser_l = gtk_label_new ("Serial Port:"); w->ser_b = GTK_COMBO_BOX(gtk_combo_box_entry_new_text ()); gtk_combo_box_append_text (w->ser_b, "/dev/ttyS0"); gtk_combo_box_append_text (w->ser_b, "/dev/ttyS1"); gtk_combo_box_append_text (w->ser_b, "/dev/ttyUSB0"); gtk_combo_box_append_text (w->ser_b, "/dev/ttyUSB1"); gtk_combo_box_append_text (w->ser_b, "usb:"); gtk_combo_box_set_active (w->ser_b, 0); g_object_ref(w->ser_b); box = GTK_TABLE(gtk_table_new(2, 2, FALSE)); gtk_table_attach_defaults(box, GTK_WIDGET(w->proto_l), 0, 1, 0, 1); gtk_table_attach_defaults(box, GTK_WIDGET(w->proto_b), 1, 2, 0, 1); gtk_table_attach_defaults(box, GTK_WIDGET(w->ser_l), 0, 1, 1, 2); gtk_table_attach_defaults(box, GTK_WIDGET(w->ser_b), 1, 2, 1, 2); gtk_box_pack_start ( GTK_BOX(GTK_DIALOG(dialog)->vbox), GTK_WIDGET(box), FALSE, FALSE, 5 ); gtk_widget_show_all ( dialog ); } void datasource_gps_add_progress_widgets ( GtkWidget *dialog, gpointer user_data ) { GtkWidget *gpslabel, *verlabel, *idlabel, *wplabel, *trklabel; gps_user_data_t *w_gps = (gps_user_data_t *)user_data; gpslabel = gtk_label_new ("GPS device: N/A"); verlabel = gtk_label_new (""); idlabel = gtk_label_new (""); wplabel = gtk_label_new (""); trklabel = gtk_label_new (""); gtk_box_pack_start ( GTK_BOX(GTK_DIALOG(dialog)->vbox), gpslabel, FALSE, FALSE, 5 ); gtk_box_pack_start ( GTK_BOX(GTK_DIALOG(dialog)->vbox), wplabel, FALSE, FALSE, 5 ); gtk_box_pack_start ( GTK_BOX(GTK_DIALOG(dialog)->vbox), trklabel, FALSE, FALSE, 5 ); gtk_widget_show_all ( dialog ); w_gps->gps_label = gpslabel; w_gps->id_label = idlabel; w_gps->ver_label = verlabel; w_gps->progress_label = w_gps->wp_label = wplabel; w_gps->trk_label = trklabel; w_gps->total_count = -1; }