]> git.street.me.uk Git - andy/viking.git/blame - src/geonamessearch.c
[QA] Improve header usage
[andy/viking.git] / src / geonamessearch.c
CommitLineData
93c47137
HR
1/*
2 * viking -- GPS Data and Topo Analyzer, Explorer, and Manager
3 *
a889d671 4 * Copyright (C) 2009, Hein Ragas
93c47137
HR
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 *
93c47137
HR
20 */
21#ifdef HAVE_CONFIG_H
22#include "config.h"
23#endif
24#include <stdlib.h>
25#include <stdio.h>
26#include <string.h>
27#include <glib.h>
28#include <glib/gstdio.h>
29#include <glib/gprintf.h>
30#include <glib/gi18n.h>
31
32#include "viking.h"
ba4a5e11 33#include "util.h"
93c47137 34#include "curl_download.h"
317aa38d 35#include "geonamessearch.h"
93c47137 36
316945d8
GB
37/* Compatibility */
38#if ! GLIB_CHECK_VERSION(2,22,0)
39#define g_mapped_file_unref g_mapped_file_free
40#endif
41
7700a91f 42#define GEONAMES_WIKIPEDIA_URL_FMT "http://ws.geonames.org/wikipediaBoundingBoxJSON?formatted=true&north=%s&south=%s&east=%s&west=%s"
09a77f0d
HR
43#define GEONAMES_COUNTRY_PATTERN "\"countryName\": \""
44#define GEONAMES_LONGITUDE_PATTERN "\"lng\": "
45#define GEONAMES_NAME_PATTERN "\"name\": \""
46#define GEONAMES_LATITUDE_PATTERN "\"lat\": "
165a4fa9
HR
47#define GEONAMES_TITLE_PATTERN "\"title\": \""
48#define GEONAMES_WIKIPEDIAURL_PATTERN "\"wikipediaUrl\": \""
49#define GEONAMES_THUMBNAILIMG_PATTERN "\"thumbnailImg\": \""
93c47137
HR
50#define GEONAMES_SEARCH_NOT_FOUND "not understand the location"
51
165a4fa9
HR
52/* found_geoname: Type to contain data returned from GeoNames.org */
53
09a77f0d
HR
54typedef struct {
55 gchar *name;
56 gchar *country;
57 struct LatLon ll;
165a4fa9 58 gchar *desc;
09a77f0d
HR
59} found_geoname;
60
165a4fa9
HR
61found_geoname *new_found_geoname()
62{
63 found_geoname *ret;
64
65 ret = (found_geoname *)g_malloc(sizeof(found_geoname));
66 ret->name = NULL;
67 ret->country = NULL;
68 ret->desc = NULL;
69 ret->ll.lat = 0.0;
70 ret->ll.lon = 0.0;
71 return(ret);
72}
73
09a77f0d
HR
74found_geoname *copy_found_geoname(found_geoname *src)
75{
165a4fa9 76 found_geoname *dest = new_found_geoname();
09a77f0d
HR
77 dest->name = g_strdup(src->name);
78 dest->country = g_strdup(src->country);
79 dest->ll.lat = src->ll.lat;
80 dest->ll.lon = src->ll.lon;
165a4fa9 81 dest->desc = g_strdup(src->desc);
09a77f0d
HR
82 return(dest);
83}
84
165a4fa9
HR
85static void free_list_geonames(found_geoname *geoname, gpointer userdata)
86{
87 g_free(geoname->name);
88 g_free(geoname->country);
89 g_free(geoname->desc);
90}
91
92void free_geoname_list(GList *found_places)
93{
94 g_list_foreach(found_places, (GFunc)free_list_geonames, NULL);
95 g_list_free(found_places);
96}
97
165a4fa9
HR
98static void none_found(VikWindow *vw)
99{
100 GtkWidget *dialog = NULL;
101
102 dialog = gtk_dialog_new_with_buttons ( "", GTK_WINDOW(vw), 0, GTK_STOCK_OK, GTK_RESPONSE_ACCEPT, NULL );
103 gtk_window_set_title(GTK_WINDOW(dialog), _("Search"));
104
105 GtkWidget *search_label = gtk_label_new(_("No entries found!"));
106 gtk_box_pack_start ( GTK_BOX(GTK_DIALOG(dialog)->vbox), search_label, FALSE, FALSE, 5 );
3a256c3c 107 gtk_dialog_set_default_response ( GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT );
165a4fa9
HR
108 gtk_widget_show_all(dialog);
109
110 gtk_dialog_run ( GTK_DIALOG(dialog) );
111 gtk_widget_destroy(dialog);
112}
113
09a77f0d
HR
114void buttonToggled(GtkCellRendererToggle* renderer, gchar* pathStr, gpointer data)
115{
116 GtkTreeIter iter;
117 gboolean enabled;
118 GtkTreePath* path = gtk_tree_path_new_from_string(pathStr);
119 gtk_tree_model_get_iter(GTK_TREE_MODEL (data), &iter, path);
120 gtk_tree_model_get(GTK_TREE_MODEL (data), &iter, 0, &enabled, -1);
121 enabled = !enabled;
122 gtk_tree_store_set(GTK_TREE_STORE (data), &iter, 0, enabled, -1);
123}
124
125GList *a_select_geoname_from_list(GtkWindow *parent, GList *geonames, gboolean multiple_selection_allowed, const gchar *title, const gchar *msg)
126{
127 GtkTreeIter iter;
128 GtkCellRenderer *renderer;
129 GtkCellRenderer *toggle_render;
130 GtkWidget *view;
131 found_geoname *geoname;
132 gchar *latlon_string;
133 int column_runner;
165a4fa9
HR
134 gboolean checked;
135 gboolean to_copy;
09a77f0d
HR
136
137 GtkWidget *dialog = gtk_dialog_new_with_buttons (title,
138 parent,
139 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
140 GTK_STOCK_CANCEL,
141 GTK_RESPONSE_REJECT,
142 GTK_STOCK_OK,
143 GTK_RESPONSE_ACCEPT,
144 NULL);
3a256c3c
RN
145 /* When something is selected then OK */
146 gtk_dialog_set_default_response ( GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT );
147 GtkWidget *response_w = NULL;
148#if GTK_CHECK_VERSION (2, 20, 0)
149 /* Default to not apply - as initially nothing is selected! */
150 response_w = gtk_dialog_get_widget_for_response ( GTK_DIALOG(dialog), GTK_RESPONSE_REJECT );
151#endif
09a77f0d
HR
152 GtkWidget *label = gtk_label_new ( msg );
153 GtkTreeStore *store;
154 if (multiple_selection_allowed)
155 {
156 store = gtk_tree_store_new(4, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
157 }
158 else
159 {
160 store = gtk_tree_store_new(3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
161 }
162 GList *geoname_runner = geonames;
163 while (geoname_runner)
164 {
165 geoname = (found_geoname *)geoname_runner->data;
166 latlon_string = g_strdup_printf("(%f,%f)", geoname->ll.lat, geoname->ll.lon);
167 gtk_tree_store_append(store, &iter, NULL);
168 if (multiple_selection_allowed)
169 {
170 gtk_tree_store_set(store, &iter, 0, FALSE, 1, geoname->name, 2, geoname->country, 3, latlon_string, -1);
171 }
172 else
173 {
174 gtk_tree_store_set(store, &iter, 0, geoname->name, 1, geoname->country, 2, latlon_string, -1);
175 }
176 geoname_runner = g_list_next(geoname_runner);
177 g_free(latlon_string);
178 }
179 view = gtk_tree_view_new();
180 renderer = gtk_cell_renderer_text_new();
181 column_runner = 0;
182 if (multiple_selection_allowed)
183 {
184 toggle_render = gtk_cell_renderer_toggle_new();
185 g_object_set(toggle_render, "activatable", TRUE, NULL);
186 g_signal_connect(toggle_render, "toggled", (GCallback) buttonToggled, GTK_TREE_MODEL(store));
187 gtk_tree_view_insert_column_with_attributes( GTK_TREE_VIEW(view), -1, "Select", toggle_render, "active", column_runner, NULL);
188 column_runner++;
189 }
190 gtk_tree_view_insert_column_with_attributes( GTK_TREE_VIEW(view), -1, "Name", renderer, "text", column_runner, NULL);
191 column_runner++;
192 gtk_tree_view_insert_column_with_attributes( GTK_TREE_VIEW(view), -1, "Country", renderer, "text", column_runner, NULL);
193 column_runner++;
194 gtk_tree_view_insert_column_with_attributes( GTK_TREE_VIEW(view), -1, "Lat/Lon", renderer, "text", column_runner, NULL);
195 gtk_tree_view_set_headers_visible( GTK_TREE_VIEW(view), TRUE);
196 gtk_tree_view_set_model(GTK_TREE_VIEW(view), GTK_TREE_MODEL(store));
197 gtk_tree_selection_set_mode( gtk_tree_view_get_selection(GTK_TREE_VIEW(view)),
198 multiple_selection_allowed ? GTK_SELECTION_MULTIPLE : GTK_SELECTION_BROWSE );
199 g_object_unref(store);
200
201 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), label, FALSE, FALSE, 0);
202 gtk_widget_show ( label );
203 gtk_box_pack_start (GTK_BOX(GTK_DIALOG(dialog)->vbox), view, FALSE, FALSE, 0);
204 gtk_widget_show ( view );
3a256c3c
RN
205 if ( response_w )
206 gtk_widget_grab_focus ( response_w );
09a77f0d
HR
207 while ( gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT )
208 {
209 GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(view));
210 GList *selected_geonames = NULL;
211
212 gtk_tree_model_get_iter_first( GTK_TREE_MODEL(store), &iter);
213 geoname_runner = geonames;
214 while (geoname_runner)
215 {
165a4fa9 216 to_copy = FALSE;
09a77f0d
HR
217 if (multiple_selection_allowed)
218 {
165a4fa9
HR
219 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, 0, &checked, -1);
220 if (checked) {
221 to_copy = TRUE;
222 }
09a77f0d
HR
223 }
224 else
225 {
226 if (gtk_tree_selection_iter_is_selected(selection, &iter))
227 {
165a4fa9 228 to_copy = TRUE;
09a77f0d
HR
229 }
230 }
165a4fa9
HR
231 if (to_copy) {
232 found_geoname *copied = copy_found_geoname(geoname_runner->data);
233 selected_geonames = g_list_prepend(selected_geonames, copied);
234 }
09a77f0d
HR
235 geoname_runner = g_list_next(geoname_runner);
236 gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter);
237 }
238 if (selected_geonames)
239 {
240 gtk_widget_destroy ( dialog );
241 return (selected_geonames);
242 }
243 a_dialog_error_msg(parent, _("Nothing was selected"));
244 }
245 gtk_widget_destroy ( dialog );
246 return NULL;
247}
248
d84ade77 249static GList *get_entries_from_file(gchar *file_name)
93c47137
HR
250{
251 gchar *text, *pat;
252 GMappedFile *mf;
253 gsize len;
09a77f0d 254 gboolean more = TRUE;
93c47137
HR
255 gchar lat_buf[32], lon_buf[32];
256 gchar *s;
09a77f0d
HR
257 gint fragment_len;
258 GList *found_places = NULL;
259 found_geoname *geoname = NULL;
260 gchar **found_entries;
261 gchar *entry;
262 int entry_runner;
165a4fa9
HR
263 gchar *wikipedia_url = NULL;
264 gchar *thumbnail_url = NULL;
93c47137
HR
265
266 lat_buf[0] = lon_buf[0] = '\0';
267
268 if ((mf = g_mapped_file_new(file_name, FALSE, NULL)) == NULL) {
269 g_critical(_("couldn't map temp file"));
93c47137
HR
270 }
271 len = g_mapped_file_get_length(mf);
272 text = g_mapped_file_get_contents(mf);
273
274 if (g_strstr_len(text, len, GEONAMES_SEARCH_NOT_FOUND) != NULL) {
09a77f0d 275 more = FALSE;
93c47137 276 }
09a77f0d
HR
277 found_entries = g_strsplit(text, "},", 0);
278 entry_runner = 0;
279 entry = found_entries[entry_runner];
280 while (entry)
281 {
282 more = TRUE;
165a4fa9
HR
283 geoname = new_found_geoname();
284 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_COUNTRY_PATTERN))) {
09a77f0d
HR
285 pat += strlen(GEONAMES_COUNTRY_PATTERN);
286 fragment_len = 0;
287 s = pat;
288 while (*pat != '"') {
289 fragment_len++;
290 pat++;
291 }
292 geoname -> country = g_strndup(s, fragment_len);
293 }
294 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_LONGITUDE_PATTERN)) == NULL) {
295 more = FALSE;
296 }
297 else {
298 pat += strlen(GEONAMES_LONGITUDE_PATTERN);
299 s = lon_buf;
300 if (*pat == '-')
301 *s++ = *pat++;
302 while ((s < (lon_buf + sizeof(lon_buf))) && (pat < (text + len)) &&
303 (g_ascii_isdigit(*pat) || (*pat == '.')))
304 *s++ = *pat++;
305 *s = '\0';
306 if ((pat >= (text + len)) || (lon_buf[0] == '\0')) {
307 more = FALSE;
308 }
309 geoname->ll.lon = g_ascii_strtod(lon_buf, NULL);
310 }
165a4fa9 311 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_NAME_PATTERN))) {
09a77f0d
HR
312 pat += strlen(GEONAMES_NAME_PATTERN);
313 fragment_len = 0;
314 s = pat;
315 while (*pat != '"') {
316 fragment_len++;
317 pat++;
318 }
319 geoname -> name = g_strndup(s, fragment_len);
320 }
165a4fa9
HR
321 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_TITLE_PATTERN))) {
322 pat += strlen(GEONAMES_TITLE_PATTERN);
323 fragment_len = 0;
324 s = pat;
325 while (*pat != '"') {
326 fragment_len++;
327 pat++;
328 }
329 geoname -> name = g_strndup(s, fragment_len);
330 }
331 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_WIKIPEDIAURL_PATTERN))) {
332 pat += strlen(GEONAMES_WIKIPEDIAURL_PATTERN);
333 fragment_len = 0;
334 s = pat;
335 while (*pat != '"') {
336 fragment_len++;
337 pat++;
338 }
339 wikipedia_url = g_strndup(s, fragment_len);
340 }
341 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_THUMBNAILIMG_PATTERN))) {
342 pat += strlen(GEONAMES_THUMBNAILIMG_PATTERN);
343 fragment_len = 0;
344 s = pat;
345 while (*pat != '"') {
346 fragment_len++;
347 pat++;
348 }
349 thumbnail_url = g_strndup(s, fragment_len);
350 }
09a77f0d
HR
351 if ((pat = g_strstr_len(entry, strlen(entry), GEONAMES_LATITUDE_PATTERN)) == NULL) {
352 more = FALSE;
353 }
354 else {
355 pat += strlen(GEONAMES_LATITUDE_PATTERN);
356 s = lat_buf;
357 if (*pat == '-')
358 *s++ = *pat++;
359 while ((s < (lat_buf + sizeof(lat_buf))) && (pat < (text + len)) &&
360 (g_ascii_isdigit(*pat) || (*pat == '.')))
361 *s++ = *pat++;
362 *s = '\0';
363 if ((pat >= (text + len)) || (lat_buf[0] == '\0')) {
364 more = FALSE;
365 }
366 geoname->ll.lat = g_ascii_strtod(lat_buf, NULL);
367 }
368 if (!more) {
369 if (geoname) {
370 g_free(geoname);
371 }
372 }
373 else {
165a4fa9
HR
374 if (wikipedia_url) {
375 if (thumbnail_url) {
376 geoname -> desc = g_strdup_printf("<a href=\"http://%s\" target=\"_blank\"><img src=\"%s\" border=\"0\"/></a>", wikipedia_url, thumbnail_url);
377 }
378 else {
379 geoname -> desc = g_strdup_printf("<a href=\"http://%s\" target=\"_blank\">%s</a>", wikipedia_url, geoname->name);
380 }
381 }
382 if (wikipedia_url) {
383 g_free(wikipedia_url);
384 wikipedia_url = NULL;
385 }
386 if (thumbnail_url) {
387 g_free(thumbnail_url);
388 thumbnail_url = NULL;
389 }
09a77f0d
HR
390 found_places = g_list_prepend(found_places, geoname);
391 }
392 entry_runner++;
393 entry = found_entries[entry_runner];
93c47137 394 }
09a77f0d 395 g_strfreev(found_entries);
165a4fa9 396 found_places = g_list_reverse(found_places);
316945d8 397 g_mapped_file_unref(mf);
165a4fa9
HR
398 return(found_places);
399}
400
401
d84ade77 402static gchar *download_url(gchar *uri)
93c47137
HR
403{
404 FILE *tmp_file;
405 int tmp_fd;
406 gchar *tmpname;
93c47137
HR
407
408 if ((tmp_fd = g_file_open_tmp ("vikgsearch.XXXXXX", &tmpname, NULL)) == -1) {
409 g_critical(_("couldn't open temp file"));
d84ade77 410 return NULL;
93c47137 411 }
93c47137 412 tmp_file = fdopen(tmp_fd, "r+");
93c47137 413
d84ade77
RN
414 if (curl_download_uri(uri, tmp_file, NULL, 0, NULL)) {
415 // error
93c47137
HR
416 fclose(tmp_file);
417 tmp_file = NULL;
165a4fa9
HR
418 g_remove(tmpname);
419 g_free(tmpname);
420 return(NULL);
93c47137 421 }
93c47137
HR
422 fclose(tmp_file);
423 tmp_file = NULL;
165a4fa9
HR
424 return(tmpname);
425}
426
165a4fa9
HR
427void a_geonames_wikipedia_box(VikWindow *vw, VikTrwLayer *vtl, VikLayersPanel *vlp, struct LatLon maxmin[2])
428{
429 gchar *uri;
430 gchar *tmpname;
431 GList *wiki_places;
432 GList *selected;
433 GList *wp_runner;
434 VikWaypoint *wiki_wp;
435 found_geoname *wiki_geoname;
436
7700a91f
GB
437 /* encode doubles in a C locale */
438 gchar *north = a_coords_dtostr(maxmin[0].lat);
439 gchar *south = a_coords_dtostr(maxmin[1].lat);
440 gchar *east = a_coords_dtostr(maxmin[0].lon);
441 gchar *west = a_coords_dtostr(maxmin[1].lon);
442 uri = g_strdup_printf(GEONAMES_WIKIPEDIA_URL_FMT, north, south, east, west);
443 g_free(north); north = NULL;
444 g_free(south); south = NULL;
445 g_free(east); east = NULL;
446 g_free(west); west = NULL;
165a4fa9
HR
447 tmpname = download_url(uri);
448 if (!tmpname) {
449 none_found(vw);
450 return;
451 }
452 wiki_places = get_entries_from_file(tmpname);
453 if (g_list_length(wiki_places) == 0) {
454 none_found(vw);
455 return;
456 }
457 selected = a_select_geoname_from_list(VIK_GTK_WINDOW_FROM_WIDGET(vw), wiki_places, TRUE, "Select articles", "Select the articles you want to add.");
458 wp_runner = selected;
459 while (wp_runner) {
460 wiki_geoname = (found_geoname *)wp_runner->data;
461 wiki_wp = vik_waypoint_new();
165a4fa9
HR
462 wiki_wp->visible = TRUE;
463 vik_coord_load_from_latlon(&(wiki_wp->coord), vik_trw_layer_get_coord_mode ( vtl ), &(wiki_geoname->ll));
464 vik_waypoint_set_comment(wiki_wp, wiki_geoname->desc);
465 vik_trw_layer_filein_add_waypoint ( vtl, wiki_geoname->name, wiki_wp );
466 wp_runner = g_list_next(wp_runner);
467 }
468 free_geoname_list(wiki_places);
469 free_geoname_list(selected);
470 g_free(uri);
471 if (tmpname) {
472 g_free(tmpname);
473 }
474 vik_layers_panel_emit_update(vlp);
475}