Contents of /trunk/src/wms.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 339 - (hide annotations)
Sat Dec 26 21:54:13 2009 UTC (14 years, 5 months ago) by harbaum
File MIME type: text/plain
File size: 42644 byte(s)
New maep and wms fix
1 harbaum 1 /*
2     * Copyright (C) 2008 Till Harbaum <till@harbaum.org>.
3     *
4     * This file is part of OSM2Go.
5     *
6     * OSM2Go 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 3 of the License, or
9     * (at your option) any later version.
10     *
11     * OSM2Go 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 OSM2Go. If not, see <http://www.gnu.org/licenses/>.
18     */
19    
20     #include "appdata.h"
21    
22     #include <libxml/parser.h>
23     #include <libxml/tree.h>
24    
25     #ifndef LIBXML_TREE_ENABLED
26     #error "Tree not enabled in libxml"
27     #endif
28    
29     #define WMS_FORMAT_JPG (1<<0)
30     #define WMS_FORMAT_JPEG (1<<1)
31     #define WMS_FORMAT_PNG (1<<2)
32     #define WMS_FORMAT_GIF (1<<3)
33    
34     typedef struct {
35     pos_t min, max;
36     gboolean valid;
37     } wms_llbbox_t;
38    
39     typedef struct wms_layer_s {
40     char *title;
41     char *name;
42 harbaum 297 char *srs;
43 harbaum 1 gboolean epsg4326, selected;
44     wms_llbbox_t llbbox;
45    
46     struct wms_layer_s *children;
47     struct wms_layer_s *next;
48     } wms_layer_t;
49    
50     typedef struct {
51     gulong format;
52     } wms_getmap_t;
53    
54     typedef struct {
55     wms_getmap_t *getmap;
56     } wms_request_t;
57    
58     typedef struct {
59     char *title;
60     } wms_service_t;
61    
62     typedef struct {
63     wms_layer_t *layer;
64     wms_request_t *request;
65     } wms_cap_t;
66    
67     typedef struct {
68     char *server;
69     char *path;
70     gint width, height;
71    
72     wms_service_t *service;
73     wms_cap_t *cap;
74     } wms_t;
75    
76     gboolean xmlTextIs(xmlDocPtr doc, xmlNodePtr list, char *str) {
77     char *nstr = (char*)xmlNodeListGetString(doc, list, 1);
78     if(!nstr) return FALSE;
79    
80     gboolean match = (strcmp(str, nstr) == 0);
81     xmlFree(nstr);
82     return match;
83     }
84    
85     gboolean xmlPropIs(xmlNode *node, char *prop, char *str) {
86     char *prop_str = (char*)xmlGetProp(node, BAD_CAST prop);
87     if(!prop_str) return FALSE;
88    
89     gboolean match = (strcmp(prop_str, str) == 0);
90     xmlFree(prop_str);
91     return match;
92     }
93    
94     float xmlGetPropFloat(xmlNode *node, char *prop) {
95     char *prop_str = (char*)xmlGetProp(node, BAD_CAST prop);
96     if(!prop_str) return NAN;
97    
98     float retval = g_ascii_strtod(prop_str, NULL);
99     xmlFree(prop_str);
100     return retval;
101     }
102    
103     static gboolean wms_bbox_is_valid(pos_t *min, pos_t *max) {
104     /* all four coordinates are valid? */
105     if(isnan(min->lat)||isnan(min->lon)||isnan(max->lat)||isnan(max->lon))
106     return FALSE;
107    
108     /* min/max span a useful range? */
109 harbaum 339 if(max->lat - min->lat < 0.1) return FALSE;
110     if(max->lon - min->lon < 0.1) return FALSE;
111 harbaum 1
112     /* useful angles? */
113     if(min->lat > 90.0 || min->lat < -90.0) return FALSE;
114     if(max->lat > 90.0 || max->lat < -90.0) return FALSE;
115     if(min->lon > 180.0 || min->lon < -180.0) return FALSE;
116     if(max->lon > 180.0 || max->lon < -180.0) return FALSE;
117    
118     return TRUE;
119     }
120    
121     static wms_layer_t *wms_cap_parse_layer(xmlDocPtr doc, xmlNode *a_node) {
122     wms_layer_t *wms_layer = NULL;
123     xmlNode *cur_node = NULL;
124     char *str = NULL;
125    
126     wms_layer = g_new0(wms_layer_t, 1);
127     wms_layer->llbbox.min.lon = wms_layer->llbbox.min.lat = NAN;
128     wms_layer->llbbox.max.lon = wms_layer->llbbox.max.lat = NAN;
129    
130     wms_layer_t **children = &(wms_layer->children);
131    
132     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
133     if (cur_node->type == XML_ELEMENT_NODE) {
134     if(strcasecmp((char*)cur_node->name, "Layer") == 0) {
135     *children = wms_cap_parse_layer(doc, cur_node);
136     if(*children) children = &((*children)->next);
137     } else if(strcasecmp((char*)cur_node->name, "Name") == 0) {
138     str = (char*)xmlNodeListGetString(doc, cur_node->children, 1);
139     wms_layer->name = g_strdup(str);
140     xmlFree(str);
141     } else if(strcasecmp((char*)cur_node->name, "Title") == 0) {
142     str = (char*)xmlNodeListGetString(doc, cur_node->children, 1);
143     wms_layer->title = g_strdup(str);
144     xmlFree(str);
145     } else if(strcasecmp((char*)cur_node->name, "SRS") == 0) {
146 harbaum 297 str = (char*)xmlNodeListGetString(doc, cur_node->children, 1);
147     wms_layer->srs = g_strdup(str);
148     xmlFree(str);
149    
150     printf("SRS = %s\n", wms_layer->srs);
151    
152     if(strcmp(wms_layer->srs, "EPSG:4326") == 0)
153 harbaum 1 wms_layer->epsg4326 = TRUE;
154 harbaum 297
155 harbaum 1 } else if(strcasecmp((char*)cur_node->name, "LatLonBoundingBox") == 0) {
156     wms_layer->llbbox.min.lat = xmlGetPropFloat(cur_node, "miny");
157     wms_layer->llbbox.min.lon = xmlGetPropFloat(cur_node, "minx");
158     wms_layer->llbbox.max.lat = xmlGetPropFloat(cur_node, "maxy");
159     wms_layer->llbbox.max.lon = xmlGetPropFloat(cur_node, "maxx");
160     } else
161     printf("found unhandled WMT_MS_Capabilities/Capability/Layer/%s\n",
162     cur_node->name);
163     }
164     }
165    
166     wms_layer->llbbox.valid = wms_bbox_is_valid(&wms_layer->llbbox.min,
167     &wms_layer->llbbox.max);
168    
169     printf("------------------- Layer: %s ---------------------------\n",
170     wms_layer->title);
171     printf("Name: %s\n", wms_layer->name);
172     printf("EPSG-4326: %s\n", wms_layer->epsg4326?"yes":"no");
173     if(wms_layer->llbbox.valid)
174     printf("LatLonBBox: %f/%f %f/%f\n",
175     wms_layer->llbbox.min.lat, wms_layer->llbbox.min.lon,
176     wms_layer->llbbox.max.lat, wms_layer->llbbox.max.lon);
177     else
178     printf("No/invalid LatLonBBox\n");
179    
180     return wms_layer;
181     }
182    
183     static wms_getmap_t *wms_cap_parse_getmap(xmlDocPtr doc, xmlNode *a_node) {
184     wms_getmap_t *wms_getmap = NULL;
185     xmlNode *cur_node = NULL;
186    
187     wms_getmap = g_new0(wms_getmap_t, 1);
188    
189     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
190     if (cur_node->type == XML_ELEMENT_NODE) {
191     if(strcasecmp((char*)cur_node->name, "Format") == 0) {
192     if(xmlTextIs(doc, cur_node->children, "image/png"))
193     wms_getmap->format |= WMS_FORMAT_PNG;
194     if(xmlTextIs(doc, cur_node->children, "image/gif"))
195     wms_getmap->format |= WMS_FORMAT_GIF;
196     if(xmlTextIs(doc, cur_node->children, "image/jpg"))
197     wms_getmap->format |= WMS_FORMAT_JPG;
198     if(xmlTextIs(doc, cur_node->children, "image/jpeg"))
199     wms_getmap->format |= WMS_FORMAT_JPEG;
200     } else
201     printf("found unhandled "
202     "WMT_MS_Capabilities/Capability/Request/GetMap/%s\n",
203     cur_node->name);
204     }
205     }
206    
207     printf("Supported formats: %s%s%s\n",
208     (wms_getmap->format & WMS_FORMAT_PNG)?"png ":"",
209     (wms_getmap->format & WMS_FORMAT_GIF)?"gif ":"",
210     (wms_getmap->format & (WMS_FORMAT_JPG | WMS_FORMAT_JPEG))?"jpg ":"");
211     return wms_getmap;
212     }
213    
214     static wms_request_t *wms_cap_parse_request(xmlDocPtr doc, xmlNode *a_node) {
215     wms_request_t *wms_request = NULL;
216     xmlNode *cur_node = NULL;
217    
218     wms_request = g_new0(wms_request_t, 1);
219    
220     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
221     if (cur_node->type == XML_ELEMENT_NODE) {
222     if(strcasecmp((char*)cur_node->name, "GetMap") == 0) {
223     wms_request->getmap = wms_cap_parse_getmap(doc, cur_node);
224     } else
225     printf("found unhandled WMT_MS_Capabilities/Capability/Request/%s\n",
226     cur_node->name);
227     }
228     }
229    
230     return wms_request;
231     }
232    
233     static wms_cap_t *wms_cap_parse_cap(xmlDocPtr doc, xmlNode *a_node) {
234     wms_cap_t *wms_cap = NULL;
235     xmlNode *cur_node = NULL;
236    
237     wms_cap = g_new0(wms_cap_t, 1);
238     wms_layer_t **layer = &(wms_cap->layer);
239    
240     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
241     if (cur_node->type == XML_ELEMENT_NODE) {
242     if(strcasecmp((char*)cur_node->name, "Request") == 0) {
243     wms_cap->request = wms_cap_parse_request(doc, cur_node);
244     } else if(strcasecmp((char*)cur_node->name, "Layer") == 0) {
245     *layer = wms_cap_parse_layer(doc, cur_node);
246     if(*layer) layer = &((*layer)->next);
247     } else
248     printf("found unhandled WMT_MS_Capabilities/Capability/%s\n",
249     cur_node->name);
250     }
251     }
252    
253     return wms_cap;
254     }
255    
256     static wms_service_t *wms_cap_parse_service(xmlDocPtr doc, xmlNode *a_node) {
257     wms_service_t *wms_service = NULL;
258     xmlNode *cur_node = NULL;
259     char *str = NULL;
260    
261     wms_service = g_new0(wms_service_t, 1);
262    
263     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
264     if (cur_node->type == XML_ELEMENT_NODE) {
265     if(strcasecmp((char*)cur_node->name, "title") == 0) {
266     str = (char*)xmlNodeListGetString(doc, cur_node->children, 1);
267     wms_service->title = g_strdup(str);
268     xmlFree(str);
269     } else
270     printf("found unhandled WMT_MS_Capabilities/Service/%s\n",
271     cur_node->name);
272     }
273     }
274    
275     printf("-- Service --\n");
276     printf("Title: %s\n", wms_service->title);
277    
278     return wms_service;
279     }
280    
281     static void wms_cap_parse(wms_t *wms, xmlDocPtr doc, xmlNode *a_node) {
282     xmlNode *cur_node = NULL;
283    
284     for (cur_node = a_node->children; cur_node; cur_node = cur_node->next) {
285     if (cur_node->type == XML_ELEMENT_NODE) {
286    
287     if(strcasecmp((char*)cur_node->name, "Service") == 0) {
288     if(!wms->service)
289     wms->service = wms_cap_parse_service(doc, cur_node);
290     } else if(strcasecmp((char*)cur_node->name, "Capability") == 0) {
291     if(!wms->cap)
292     wms->cap = wms_cap_parse_cap(doc, cur_node);
293     } else
294     printf("found unhandled WMT_MS_Capabilities/%s\n", cur_node->name);
295     }
296     }
297     }
298    
299     /* parse root element */
300     static void wms_cap_parse_root(wms_t *wms, xmlDocPtr doc, xmlNode *a_node) {
301     xmlNode *cur_node = NULL;
302    
303     for (cur_node = a_node; cur_node; cur_node = cur_node->next) {
304     if (cur_node->type == XML_ELEMENT_NODE) {
305    
306     if(strcasecmp((char*)cur_node->name, "WMT_MS_Capabilities") == 0) {
307     wms_cap_parse(wms, doc, cur_node);
308     } else
309     printf("found unhandled %s\n", cur_node->name);
310     }
311     }
312     }
313    
314     static void wms_cap_parse_doc(wms_t *wms, xmlDocPtr doc) {
315     /* Get the root element node */
316     xmlNode *root_element = xmlDocGetRootElement(doc);
317    
318     wms_cap_parse_root(wms, doc, root_element);
319    
320     /*free the document */
321     xmlFreeDoc(doc);
322    
323     /*
324     * Free the global variables that may
325     * have been allocated by the parser.
326     */
327     xmlCleanupParser();
328     }
329    
330     /* get pixel extent of image display */
331     void wms_setup_extent(project_t *project, wms_t *wms) {
332     pos_t center;
333     lpos_t lcenter, lmin, lmax;
334     float scale;
335    
336     center.lat = (project->min.lat + project->max.lat)/2;
337     center.lon = (project->min.lon + project->max.lon)/2;
338    
339     pos2lpos_center(&center, &lcenter);
340    
341     /* the scale is needed to accomodate for "streching" */
342     /* by the mercartor projection */
343     scale = cos(DEG2RAD(center.lat));
344    
345     pos2lpos_center(&project->min, &lmin);
346     lmin.x -= lcenter.x;
347     lmin.y -= lcenter.y;
348     lmin.x *= scale;
349     lmin.y *= scale;
350    
351     pos2lpos_center(&project->max, &lmax);
352     lmax.x -= lcenter.x;
353     lmax.y -= lcenter.y;
354     lmax.x *= scale;
355     lmax.y *= scale;
356    
357     wms->width = lmax.x - lmin.x;
358     wms->height = lmax.y - lmin.y;
359    
360     if(wms->width > 2048) wms->width = 2048;
361     if(wms->height > 2048) wms->height = 2048;
362    
363     printf("WMS: required image size = %dx%d\n",
364     wms->width, wms->height);
365     }
366    
367     /* --------------- freeing stuff ------------------- */
368    
369     static void wms_layer_free(wms_layer_t *layer) {
370     while(layer) {
371    
372     if(layer->title) g_free(layer->title);
373     if(layer->name) g_free(layer->name);
374 harbaum 297 if(layer->srs) g_free(layer->srs);
375 harbaum 1
376     if(layer->children) wms_layer_free(layer->children);
377    
378     wms_layer_t *next = layer->next;
379     g_free(layer);
380     layer = next;
381     }
382     }
383    
384     static void wms_getmap_free(wms_getmap_t *getmap) {
385     g_free(getmap);
386     }
387    
388     static void wms_request_free(wms_request_t *request) {
389     if(request->getmap) wms_getmap_free(request->getmap);
390     g_free(request);
391     }
392    
393     static void wms_cap_free(wms_cap_t *cap) {
394     if(cap->layer) wms_layer_free(cap->layer);
395     if(cap->request) wms_request_free(cap->request);
396     g_free(cap);
397     }
398    
399     static void wms_service_free(wms_service_t *service) {
400     if(service->title) g_free(service->title);
401     g_free(service);
402     }
403    
404     static void wms_free(wms_t *wms) {
405     if(wms->server) g_free(wms->server);
406     if(wms->path) g_free(wms->path);
407     if(wms->cap) wms_cap_free(wms->cap);
408     if(wms->service) wms_service_free(wms->service);
409     g_free(wms);
410     }
411    
412     /* ---------------------- use ------------------- */
413    
414     static gboolean wms_llbbox_fits(project_t *project, wms_llbbox_t *llbbox) {
415     if(!llbbox || !llbbox->valid ||
416     (project->min.lat < llbbox->min.lat) ||
417     (project->min.lon < llbbox->min.lon) ||
418     (project->max.lat > llbbox->max.lat) ||
419     (project->max.lon > llbbox->max.lon))
420     return FALSE;
421    
422     return TRUE;
423     }
424    
425     static void wms_get_child_layers(wms_layer_t *layer,
426 harbaum 297 gint depth, gboolean epsg4326, wms_llbbox_t *llbbox, char *srs,
427     wms_layer_t **c_layer) {
428 harbaum 1 while(layer) {
429    
430     /* get a copy of the parents values for the current one ... */
431     wms_llbbox_t *local_llbbox = llbbox;
432     gboolean local_epsg4326 = epsg4326;
433 harbaum 297 char *local_srs = srs?g_strdup(srs):NULL;
434 harbaum 1
435     /* ... and overwrite the inherited stuff with better local stuff */
436     if(layer->llbbox.valid) local_llbbox = &layer->llbbox;
437     if(layer->epsg4326) local_epsg4326 = TRUE;
438    
439     /* only named layers with useful bounding box are added to the list */
440     if(local_llbbox && layer->name) {
441     *c_layer = g_new0(wms_layer_t, 1);
442     (*c_layer)->name = g_strdup(layer->name);
443     (*c_layer)->title = g_strdup(layer->title);
444 harbaum 297 (*c_layer)->srs = g_strdup(local_srs);
445 harbaum 1 (*c_layer)->epsg4326 = local_epsg4326;
446     (*c_layer)->llbbox = *local_llbbox;
447     c_layer = &((*c_layer)->next);
448     }
449    
450     wms_get_child_layers(layer->children, depth+1,
451     local_epsg4326, local_llbbox,
452 harbaum 297 local_srs, c_layer);
453 harbaum 1
454 harbaum 297 if(local_srs) g_free(local_srs);
455 harbaum 1 layer = layer->next;
456     }
457     }
458    
459     static wms_layer_t *wms_get_requestable_layers(wms_t *wms) {
460     printf("\nSearching for usable layers\n");
461    
462     wms_layer_t *r_layer = NULL, **c_layer = &r_layer;
463    
464     wms_layer_t *layer = wms->cap->layer;
465     while(layer) {
466     wms_llbbox_t *llbbox = &layer->llbbox;
467     if(llbbox && !llbbox->valid) llbbox = NULL;
468    
469     wms_get_child_layers(layer->children, 1,
470 harbaum 297 layer->epsg4326, llbbox, layer->srs, c_layer);
471 harbaum 1
472     layer = layer->next;
473     }
474    
475     return r_layer;
476     }
477    
478 harbaum 14 enum {
479     WMS_SERVER_COL_NAME = 0,
480     WMS_SERVER_COL_DATA,
481     WMS_SERVER_NUM_COLS
482     };
483    
484     typedef struct {
485     appdata_t *appdata;
486     wms_t *wms;
487 harbaum 146 GtkWidget *dialog, *list;
488 harbaum 14 GtkListStore *store;
489     GtkWidget *server_label, *path_label;
490     } wms_server_context_t;
491    
492     static wms_server_t *get_selection(wms_server_context_t *context) {
493     GtkTreeSelection *selection;
494     GtkTreeModel *model;
495     GtkTreeIter iter;
496    
497 harbaum 146 selection = list_get_selection(context->list);
498 harbaum 14 if(gtk_tree_selection_get_selected(selection, &model, &iter)) {
499     wms_server_t *wms_server;
500     gtk_tree_model_get(model, &iter, WMS_SERVER_COL_DATA, &wms_server, -1);
501     return(wms_server);
502     }
503    
504     return NULL;
505     }
506    
507     static void wms_server_selected(wms_server_context_t *context,
508     wms_server_t *selected) {
509    
510     if(!selected && context->wms->server && context->wms->path) {
511     /* if the projects settings match a list entry, then select this */
512    
513 harbaum 146 GtkTreeSelection *selection = list_get_selection(context->list);
514 harbaum 14
515     /* walk the entire store to get all values */
516     wms_server_t *server = NULL;
517     GtkTreeIter iter;
518    
519     gboolean valid =
520     gtk_tree_model_get_iter_first(GTK_TREE_MODEL(context->store), &iter);
521    
522     while(valid && !selected) {
523     gtk_tree_model_get(GTK_TREE_MODEL(context->store), &iter,
524     WMS_SERVER_COL_DATA, &server, -1);
525     g_assert(server);
526    
527     if((strcmp(server->server, context->wms->server) == 0) &&
528     (strcmp(server->path, context->wms->path) == 0)) {
529     gtk_tree_selection_select_iter(selection, &iter);
530     selected = server;
531     }
532    
533     valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(context->store), &iter);
534     }
535     }
536    
537 harbaum 146 list_button_enable(context->list, LIST_BUTTON_REMOVE, selected != NULL);
538     list_button_enable(context->list, LIST_BUTTON_EDIT, selected != NULL);
539 harbaum 14
540     /* user can click ok if a entry is selected or if both fields are */
541     /* otherwise valid */
542     if(selected) {
543     gtk_dialog_set_response_sensitive(GTK_DIALOG(context->dialog),
544     GTK_RESPONSE_ACCEPT, TRUE);
545    
546     gtk_label_set_text(GTK_LABEL(context->server_label), selected->server);
547     gtk_label_set_text(GTK_LABEL(context->path_label), selected->path);
548     } else {
549     gtk_dialog_set_response_sensitive(GTK_DIALOG(context->dialog),
550     GTK_RESPONSE_ACCEPT, context->wms->server && context->wms->path);
551    
552     if(context->wms->server)
553     gtk_label_set_text(GTK_LABEL(context->server_label),
554     context->wms->server);
555     if(context->wms->path)
556     gtk_label_set_text(GTK_LABEL(context->path_label),
557     context->wms->path);
558     }
559     }
560    
561 harbaum 328 static void
562     wms_server_changed(GtkTreeSelection *selection, gpointer userdata) {
563 harbaum 14 wms_server_context_t *context = (wms_server_context_t*)userdata;
564 harbaum 328
565     GtkTreeModel *model = NULL;
566 harbaum 14 GtkTreeIter iter;
567 harbaum 328
568     if(gtk_tree_selection_get_selected(selection, &model, &iter)) {
569 harbaum 14 wms_server_t *wms_server = NULL;
570    
571     gtk_tree_model_get(model, &iter, WMS_SERVER_COL_DATA, &wms_server, -1);
572     wms_server_selected(context, wms_server);
573     }
574     }
575    
576     static void on_server_remove(GtkWidget *but, wms_server_context_t *context) {
577     GtkTreeSelection *selection;
578     GtkTreeModel *model;
579     GtkTreeIter iter;
580    
581 harbaum 146 selection = list_get_selection(context->list);
582 harbaum 14 if(gtk_tree_selection_get_selected(selection, &model, &iter)) {
583     wms_server_t *server = NULL;
584     gtk_tree_model_get(model, &iter, WMS_SERVER_COL_DATA, &server, -1);
585    
586     g_assert(server);
587    
588     /* de-chain */
589     printf("de-chaining server %s\n", server->name);
590     wms_server_t **prev = &context->appdata->settings->wms_server;
591     while(*prev != server) prev = &((*prev)->next);
592     *prev = server->next;
593    
594     /* free tag itself */
595     wms_server_free(server);
596    
597     /* and remove from store */
598     gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
599     }
600    
601     wms_server_selected(context, NULL);
602     }
603    
604     static void callback_modified_name(GtkWidget *widget, gpointer data) {
605     wms_server_context_t *context = (wms_server_context_t*)data;
606    
607     char *name = (char*)gtk_entry_get_text(GTK_ENTRY(widget));
608    
609     /* name must not contain some special chars */
610     gboolean ok = TRUE;
611    
612     /* search all entries except the last (which is the one we are editing) */
613     wms_server_t *server = context->appdata->settings->wms_server;
614     while(server && server->next) {
615     if(strcasecmp(server->name, name) == 0) ok = FALSE;
616     server = server->next;
617     }
618    
619     GtkWidget *toplevel = gtk_widget_get_toplevel(widget);
620     /* toplevel is a dialog only of dialog has been realized */
621     if(GTK_IS_DIALOG(toplevel))
622     gtk_dialog_set_response_sensitive(
623     GTK_DIALOG(gtk_widget_get_toplevel(widget)),
624     GTK_RESPONSE_ACCEPT, ok);
625     }
626    
627     /* edit url and path of a given wms server entry */
628     gboolean wms_server_edit(wms_server_context_t *context, gboolean edit_name,
629     wms_server_t *wms_server) {
630 harbaum 167 GtkWidget *dialog =
631     misc_dialog_new(MISC_DIALOG_WIDE, _("Edit WMS Server"),
632     GTK_WINDOW(context->dialog),
633     GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
634     GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
635     NULL);
636 harbaum 14
637     gtk_dialog_set_default_response(GTK_DIALOG(dialog),
638     GTK_RESPONSE_ACCEPT);
639    
640     GtkWidget *label, *name, *server, *path;
641     GtkWidget *table = gtk_table_new(2, 3, FALSE);
642    
643     gtk_table_attach(GTK_TABLE(table),
644     label = gtk_label_new(_("Name:")), 0, 1, 0, 1,
645     GTK_FILL, 0, 0, 0);
646     gtk_misc_set_alignment(GTK_MISC(label), 0.f, 0.5f);
647     gtk_table_attach_defaults(GTK_TABLE(table),
648 harbaum 315 name = entry_new(), 1, 2, 0, 1);
649 harbaum 14 gtk_entry_set_activates_default(GTK_ENTRY(name), TRUE);
650     HILDON_ENTRY_NO_AUTOCAP(name);
651     gtk_widget_set_sensitive(GTK_WIDGET(name), edit_name);
652     g_signal_connect(G_OBJECT(name), "changed",
653     G_CALLBACK(callback_modified_name), context);
654    
655     gtk_table_attach(GTK_TABLE(table),
656     label = gtk_label_new(_("Server:")), 0, 1, 1, 2,
657     GTK_FILL, 0, 0, 0);
658     gtk_misc_set_alignment(GTK_MISC(label), 0.f, 0.5f);
659     gtk_table_attach_defaults(GTK_TABLE(table),
660 harbaum 315 server = entry_new(), 1, 2, 1, 2);
661 harbaum 14 gtk_entry_set_activates_default(GTK_ENTRY(server), TRUE);
662     HILDON_ENTRY_NO_AUTOCAP(server);
663    
664     gtk_table_attach(GTK_TABLE(table),
665     label = gtk_label_new(_("Path:")), 0, 1, 2, 3,
666     GTK_FILL, 0, 0, 0);
667     gtk_misc_set_alignment(GTK_MISC(label), 0.f, 0.5f);
668     gtk_table_attach_defaults(GTK_TABLE(table),
669 harbaum 315 path = entry_new(), 1, 2, 2, 3);
670 harbaum 14 gtk_entry_set_activates_default(GTK_ENTRY(path), TRUE);
671     HILDON_ENTRY_NO_AUTOCAP(path);
672    
673     gtk_entry_set_text(GTK_ENTRY(name), wms_server->name);
674     gtk_entry_set_text(GTK_ENTRY(server), wms_server->server);
675     gtk_entry_set_text(GTK_ENTRY(path), wms_server->path);
676    
677     gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox), table);
678    
679     gtk_widget_show_all(dialog);
680    
681     if(GTK_RESPONSE_ACCEPT == gtk_dialog_run(GTK_DIALOG(dialog))) {
682     if(edit_name) {
683     GtkTreeSelection *selection;
684     GtkTreeModel *model;
685     GtkTreeIter iter;
686 harbaum 146 selection = list_get_selection(context->list);
687 harbaum 14 gtk_tree_selection_get_selected(selection, &model, &iter);
688     gtk_list_store_set(context->store, &iter,
689     WMS_SERVER_COL_NAME, wms_server->name,
690     -1);
691    
692     g_free(wms_server->name);
693     wms_server->name = strdup((char*)gtk_entry_get_text(GTK_ENTRY(name)));
694     }
695    
696     g_free(wms_server->server);
697     wms_server->server = strdup((char*)gtk_entry_get_text(GTK_ENTRY(server)));
698     g_free(wms_server->path);
699     wms_server->path = strdup((char*)gtk_entry_get_text(GTK_ENTRY(path)));
700     printf("setting %s/%s\n", wms_server->server, wms_server->path);
701    
702     /* set texts below */
703     gtk_label_set_text(GTK_LABEL(context->server_label), wms_server->server);
704     gtk_label_set_text(GTK_LABEL(context->path_label), wms_server->path);
705    
706     gtk_widget_destroy(dialog);
707    
708     return TRUE;
709     }
710    
711     gtk_widget_destroy(dialog);
712     return FALSE;
713     }
714    
715     /* user clicked "edit..." button in the wms server list */
716     static void on_server_edit(GtkWidget *but, wms_server_context_t *context) {
717     GtkTreeSelection *selection;
718     GtkTreeModel *model;
719     GtkTreeIter iter;
720    
721 harbaum 146 selection = list_get_selection(context->list);
722 harbaum 14 if(gtk_tree_selection_get_selected(selection, &model, &iter)) {
723     wms_server_t *server = NULL;
724     gtk_tree_model_get(model, &iter, WMS_SERVER_COL_DATA, &server, -1);
725     g_assert(server);
726    
727     wms_server_edit(context, FALSE, server);
728     }
729     }
730    
731     /* user clicked "add..." button in the wms server list */
732     static void on_server_add(GtkWidget *but, wms_server_context_t *context) {
733    
734     /* attach a new server item to the chain */
735     wms_server_t **prev = &context->appdata->settings->wms_server;
736     while(*prev) prev = &(*prev)->next;
737    
738     *prev = g_new0(wms_server_t, 1);
739     (*prev)->name = g_strdup("<service name>");
740     (*prev)->server = g_strdup("<server url>");
741     (*prev)->path = g_strdup("<path in server>");
742    
743 harbaum 146 GtkTreeModel *model = list_get_model(context->list);
744 harbaum 14
745     GtkTreeIter iter;
746     gtk_list_store_append(GTK_LIST_STORE(model), &iter);
747     gtk_list_store_set(GTK_LIST_STORE(model), &iter,
748     WMS_SERVER_COL_NAME, (*prev)->name,
749     WMS_SERVER_COL_DATA, *prev,
750     -1);
751    
752 harbaum 146 GtkTreeSelection *selection = list_get_selection(context->list);
753 harbaum 14 gtk_tree_selection_select_iter(selection, &iter);
754    
755     if(!wms_server_edit(context, TRUE, *prev)) {
756     /* user has cancelled request. remove newly added item */
757     printf("user clicked cancel\n");
758    
759     wms_server_free(*prev);
760     *prev = NULL;
761    
762     gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
763     } else
764     /* update name from edit result */
765     gtk_list_store_set(GTK_LIST_STORE(model), &iter,
766     WMS_SERVER_COL_NAME, (*prev)->name,
767     -1);
768    
769     wms_server_selected(context, *prev);
770     }
771    
772     /* widget to select a wms server from a list */
773     static GtkWidget *wms_server_widget(wms_server_context_t *context) {
774    
775 harbaum 148 context->list = list_new(LIST_HILDON_WITHOUT_HEADERS);
776 harbaum 14
777 harbaum 328 list_override_changed_event(context->list, wms_server_changed, context);
778 harbaum 14
779 harbaum 146 list_set_columns(context->list,
780 harbaum 148 _("Name"), WMS_SERVER_COL_NAME, LIST_FLAG_ELLIPSIZE,
781 harbaum 146 NULL);
782 harbaum 14
783     /* build and fill the store */
784     context->store = gtk_list_store_new(WMS_SERVER_NUM_COLS,
785     G_TYPE_STRING, G_TYPE_POINTER);
786    
787 harbaum 146 list_set_store(context->list, context->store);
788 harbaum 14
789     GtkTreeIter iter;
790     wms_server_t *wms_server = context->appdata->settings->wms_server;
791     while(wms_server) {
792     /* Append a row and fill in some data */
793     gtk_list_store_append(context->store, &iter);
794     gtk_list_store_set(context->store, &iter,
795     WMS_SERVER_COL_NAME, wms_server->name,
796     WMS_SERVER_COL_DATA, wms_server,
797     -1);
798    
799     wms_server = wms_server->next;
800     }
801    
802     g_object_unref(context->store);
803    
804 harbaum 315 list_set_static_buttons(context->list, 0,
805 harbaum 266 G_CALLBACK(on_server_add), G_CALLBACK(on_server_edit),
806     G_CALLBACK(on_server_remove), context);
807 harbaum 14
808 harbaum 146 return context->list;
809 harbaum 14 }
810    
811 harbaum 1 static gboolean wms_server_dialog(appdata_t *appdata, wms_t *wms) {
812     gboolean ok = FALSE;
813    
814 harbaum 14 wms_server_context_t *context = g_new0(wms_server_context_t, 1);
815     context->appdata = appdata;
816     context->wms = wms;
817    
818 harbaum 167 context->dialog =
819     misc_dialog_new(MISC_DIALOG_MEDIUM, _("WMS Server Selection"),
820     GTK_WINDOW(appdata->window),
821     GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
822     GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
823     NULL);
824 harbaum 1
825 harbaum 14 /* server selection box */
826     gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(context->dialog)->vbox),
827     wms_server_widget(context));
828    
829     gtk_box_pack_start(GTK_BOX(GTK_DIALOG(context->dialog)->vbox),
830     gtk_hseparator_new(), FALSE, FALSE, 0);
831    
832 harbaum 1 GtkWidget *label;
833     GtkWidget *table = gtk_table_new(2, 2, FALSE); // x, y
834 harbaum 14 gtk_table_set_col_spacing(GTK_TABLE(table), 0, 10);
835 harbaum 1
836     label = gtk_label_new(_("Server:"));
837 harbaum 14 gtk_misc_set_alignment(GTK_MISC(label), 0.f, 0.5f);
838     gtk_table_attach(GTK_TABLE(table), label, 0, 1, 0, 1, GTK_FILL, 0,0,0);
839     context->server_label = gtk_label_new("");
840     gtk_label_set_ellipsize(GTK_LABEL(context->server_label),
841     PANGO_ELLIPSIZE_MIDDLE);
842     gtk_misc_set_alignment(GTK_MISC(context->server_label), 0.f, 0.5f);
843     gtk_table_attach_defaults(GTK_TABLE(table), context->server_label,
844     1, 2, 0, 1);
845 harbaum 1
846     label = gtk_label_new(_("Path:"));
847 harbaum 14 gtk_misc_set_alignment(GTK_MISC(label), 0.f, 0.5f);
848     gtk_table_attach(GTK_TABLE(table), label, 0, 1, 1, 2, GTK_FILL, 0,0,0);
849     context->path_label = gtk_label_new("");
850     gtk_label_set_ellipsize(GTK_LABEL(context->path_label),
851     PANGO_ELLIPSIZE_MIDDLE);
852     gtk_misc_set_alignment(GTK_MISC(context->path_label), 0.f, 0.5f);
853     gtk_table_attach_defaults(GTK_TABLE(table), context->path_label,
854     1, 2, 1, 2);
855 harbaum 1
856 harbaum 14 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(context->dialog)->vbox),
857     table, FALSE, FALSE, 0);
858    
859    
860     wms_server_selected(context, NULL);
861    
862     gtk_widget_show_all(context->dialog);
863 harbaum 1
864 harbaum 14 if(GTK_RESPONSE_ACCEPT == gtk_dialog_run(GTK_DIALOG(context->dialog))) {
865     wms_server_t *server = get_selection(context);
866     if(server) {
867     /* fetch parameters from selected entry */
868     printf("WMS: using %s\n", server->name);
869     if(wms->server) g_free(wms->server);
870     wms->server = g_strdup(server->server);
871     if(wms->path) g_free(wms->path);
872     wms->path = g_strdup(server->path);
873     ok = TRUE;
874     } else {
875     if(wms->server && wms->path)
876     ok = TRUE;
877    
878     }
879 harbaum 1 }
880    
881 harbaum 14 gtk_widget_destroy(context->dialog);
882 harbaum 1
883 harbaum 14 g_free(context);
884 harbaum 1 return ok;
885     }
886    
887     enum {
888     LAYER_COL_TITLE = 0,
889     LAYER_COL_FITS,
890     LAYER_COL_DATA,
891     LAYER_NUM_COLS
892     };
893    
894 harbaum 321 #ifndef FREMANTLE
895     /* we handle these events on our own in order to implement */
896     /* a very direct selection mechanism (multiple selections usually */
897     /* require the control key to be pressed). This interferes with */
898     /* fremantle finger scrolling, but fortunately the fremantle */
899     /* default behaviour already is what we want. */
900     static gboolean on_view_clicked(GtkWidget *widget, GdkEventButton *event,
901     gpointer user_data) {
902     if(event->window == gtk_tree_view_get_bin_window(GTK_TREE_VIEW(widget))) {
903     GtkTreePath *path;
904    
905     if(gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget),
906     event->x, event->y, &path, NULL, NULL, NULL)) {
907     GtkTreeSelection *sel =
908     gtk_tree_view_get_selection(GTK_TREE_VIEW(widget));
909 harbaum 1
910 harbaum 321 if(!gtk_tree_selection_path_is_selected(sel, path))
911     gtk_tree_selection_select_path(sel, path);
912     else
913     gtk_tree_selection_unselect_path(sel, path);
914     }
915     return TRUE;
916     }
917     return FALSE;
918     }
919     #endif
920 harbaum 1
921 harbaum 321 static void changed(GtkTreeSelection *sel, gpointer user_data) {
922     /* we need to know what changed in order to let the user acknowlege it! */
923 harbaum 1
924 harbaum 321 /* get view from selection ... */
925     GtkTreeView *view = gtk_tree_selection_get_tree_view(sel);
926     g_assert(view);
927 harbaum 1
928 harbaum 321 /* ... and get model from view */
929     GtkTreeModel *model = gtk_tree_view_get_model(view);
930     g_assert(model);
931 harbaum 1
932 harbaum 321 /* walk the entire store */
933     GtkTreeIter iter;
934     gboolean one = FALSE, ok = gtk_tree_model_get_iter_first(model, &iter);
935     while(ok) {
936     wms_layer_t *layer = NULL;
937 harbaum 1
938 harbaum 321 gtk_tree_model_get(model, &iter, LAYER_COL_DATA, &layer, -1);
939     g_assert(layer);
940    
941     layer->selected = gtk_tree_selection_iter_is_selected(sel, &iter);
942     if(layer->selected) one = TRUE;
943    
944     ok = gtk_tree_model_iter_next(model, &iter);
945 harbaum 1 }
946    
947 harbaum 321 GtkWidget *dialog = gtk_widget_get_toplevel(GTK_WIDGET(view));
948 harbaum 1 gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog),
949 harbaum 321 GTK_RESPONSE_ACCEPT, one);
950 harbaum 1 }
951    
952     static GtkWidget *wms_layer_widget(appdata_t *appdata, wms_layer_t *layer,
953     GtkWidget *dialog) {
954 harbaum 314
955     #ifndef FREMANTLE_PANNABLE_AREA
956 harbaum 1 GtkWidget *view = gtk_tree_view_new();
957 harbaum 314 #else
958     GtkWidget *view = hildon_gtk_tree_view_new(HILDON_UI_MODE_EDIT);
959     #endif
960 harbaum 1
961 harbaum 321 /* change list mode to "multiple" */
962     GtkTreeSelection *selection =
963     gtk_tree_view_get_selection(GTK_TREE_VIEW(view));
964     gtk_tree_selection_set_mode(selection, GTK_SELECTION_MULTIPLE);
965    
966     #ifndef FREMANTLE
967     /* catch views button-press event for our custom handling */
968     g_signal_connect(view, "button-press-event",
969     G_CALLBACK(on_view_clicked), NULL);
970     #endif
971    
972 harbaum 1 /* build the store */
973     GtkListStore *store = gtk_list_store_new(LAYER_NUM_COLS,
974 harbaum 321 G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_POINTER);
975 harbaum 1
976     /* --- "Title" column --- */
977 harbaum 321 GtkCellRenderer *renderer = gtk_cell_renderer_text_new();
978 harbaum 1 g_object_set(renderer, "ellipsize", PANGO_ELLIPSIZE_END, NULL );
979     GtkTreeViewColumn *column = gtk_tree_view_column_new_with_attributes(
980     _("Title"), renderer,
981     "text", LAYER_COL_TITLE,
982 harbaum 180 "sensitive", LAYER_COL_FITS,
983 harbaum 1 NULL);
984    
985     gtk_tree_view_column_set_expand(column, TRUE);
986     gtk_tree_view_insert_column(GTK_TREE_VIEW(view), column, -1);
987    
988     gtk_tree_view_set_model(GTK_TREE_VIEW(view), GTK_TREE_MODEL(store));
989    
990     GtkTreeIter iter;
991     while(layer) {
992     gboolean fits = wms_llbbox_fits(appdata->project, &layer->llbbox);
993    
994     /* Append a row and fill in some data */
995     gtk_list_store_append(store, &iter);
996     gtk_list_store_set(store, &iter,
997     LAYER_COL_TITLE, layer->title,
998     LAYER_COL_FITS, fits,
999     LAYER_COL_DATA, layer,
1000     -1);
1001     layer = layer->next;
1002     }
1003    
1004     g_object_unref(store);
1005    
1006 harbaum 321 g_signal_connect(G_OBJECT(selection), "changed", G_CALLBACK(changed), layer);
1007    
1008 harbaum 314 #ifndef FREMANTLE_PANNABLE_AREA
1009 harbaum 1 /* put it into a scrolled window */
1010     GtkWidget *scrolled_window = gtk_scrolled_window_new(NULL, NULL);
1011     gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_window),
1012     GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
1013     gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled_window),
1014     GTK_SHADOW_ETCHED_IN);
1015     gtk_container_add(GTK_CONTAINER(scrolled_window), view);
1016     return scrolled_window;
1017 harbaum 314 #else
1018     /* put view into a pannable area */
1019     GtkWidget *pannable_area = hildon_pannable_area_new();
1020     gtk_container_add(GTK_CONTAINER(pannable_area), view);
1021     return pannable_area;
1022     #endif
1023 harbaum 1 }
1024    
1025    
1026     static gboolean wms_layer_dialog(appdata_t *appdata, wms_layer_t *layer) {
1027     gboolean ok = FALSE;
1028    
1029 harbaum 167 GtkWidget *dialog =
1030     misc_dialog_new(MISC_DIALOG_LARGE, _("WMS layer selection"),
1031     GTK_WINDOW(appdata->window),
1032     GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
1033     GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
1034     NULL);
1035 harbaum 1
1036     gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog),
1037     GTK_RESPONSE_ACCEPT, FALSE);
1038    
1039     /* layer list */
1040     gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox),
1041     wms_layer_widget(appdata, layer, dialog));
1042    
1043    
1044     gtk_widget_show_all(dialog);
1045    
1046     if(GTK_RESPONSE_ACCEPT == gtk_dialog_run(GTK_DIALOG(dialog)))
1047     ok = TRUE;
1048    
1049     gtk_widget_destroy(dialog);
1050    
1051     return ok;
1052     }
1053    
1054     static gboolean wms_one_layer_is_usable(project_t *project,
1055     wms_layer_t *layer) {
1056     gboolean ok = FALSE;
1057    
1058     while(layer) {
1059     if(layer->name && layer->epsg4326 && layer->llbbox.valid)
1060     ok = TRUE;
1061    
1062     printf("----- Layer \"%s\" -----\n", layer->title);
1063     printf("Name: %s\n", layer->name);
1064     printf("epsg4326: %s\n", layer->epsg4326?"yes":"no");
1065     if(layer->llbbox.valid) {
1066     printf("llbbox: %f/%f %f/%f\n",
1067     layer->llbbox.min.lat, layer->llbbox.min.lon,
1068     layer->llbbox.max.lat, layer->llbbox.max.lon);
1069    
1070     printf("llbbox fits project: %s\n",
1071     wms_llbbox_fits(project, &layer->llbbox)?"yes":"no");
1072     } else
1073     printf("llbbox: none/invalid\n");
1074    
1075     layer = layer->next;
1076     }
1077    
1078     return ok;
1079     }
1080    
1081     void wms_import(appdata_t *appdata) {
1082     if(!appdata->project) {
1083     errorf(GTK_WIDGET(appdata->window),
1084     _("Need an open project to derive WMS coordinates"));
1085     return;
1086     }
1087    
1088     /* this cancels any wms adjustment in progress */
1089     if(appdata->map->action.type == MAP_ACTION_BG_ADJUST)
1090     map_action_cancel(appdata);
1091    
1092     wms_t *wms = g_new0(wms_t,1);
1093     wms->server = g_strdup(appdata->project->wms_server);
1094     wms->path = g_strdup(appdata->project->wms_path);
1095    
1096     /* reset any background adjustments in the project ... */
1097     if((appdata->project->wms_offset.x != 0)||
1098     (appdata->project->wms_offset.y != 0)) {
1099    
1100     appdata->project->wms_offset.x = 0;
1101     appdata->project->wms_offset.y = 0;
1102     }
1103    
1104     /* ... as well as in the map */
1105     appdata->map->bg.offset.x = 0;
1106     appdata->map->bg.offset.y = 0;
1107    
1108     /* get server from dialog */
1109     if(!wms_server_dialog(appdata, wms)) {
1110     wms_free(wms);
1111     return;
1112     }
1113    
1114     /* ------------- copy values back into project ---------------- */
1115 harbaum 75 if(!appdata->project->wms_server ||
1116     strcmp(appdata->project->wms_server, wms->server) != 0) {
1117     if(appdata->project->wms_server)
1118     g_free(appdata->project->wms_server);
1119 harbaum 1 appdata->project->wms_server = g_strdup(wms->server);
1120     }
1121    
1122 harbaum 75 if(!appdata->project->wms_path ||
1123     strcmp(appdata->project->wms_path, wms->path) != 0) {
1124     if(appdata->project->wms_path)
1125     g_free(appdata->project->wms_path);
1126 harbaum 1 appdata->project->wms_path = g_strdup(wms->path);
1127     }
1128    
1129     /* ----------- request capabilities -------------- */
1130     gboolean path_contains_qm = (strchr(wms->path, '?') != NULL);
1131     gboolean path_ends_with_special =
1132     (wms->path[strlen(wms->path)-1] == '?') ||
1133     (wms->path[strlen(wms->path)-1] == '&');
1134    
1135     /* if there's already a question mark, then add further */
1136     /* parameters using the &, else use the ? */
1137     char *append_char = path_ends_with_special?"":(path_contains_qm?"&":"?");
1138    
1139     char *url = g_strdup_printf("%s%s"
1140     "%sSERVICE=wms"
1141 harbaum 14 // "&WMTVER=1.1.1"
1142     "&VERSION=1.1.1"
1143 harbaum 1 "&REQUEST=GetCapabilities",
1144     wms->server, wms->path, append_char);
1145    
1146 harbaum 7 char *cap = NULL;
1147 harbaum 169 net_io_download_mem(GTK_WIDGET(appdata->window), appdata->settings,
1148     url, &cap);
1149 harbaum 1 g_free(url);
1150    
1151     /* ----------- parse capabilities -------------- */
1152    
1153     if(!cap) {
1154     errorf(GTK_WIDGET(appdata->window),
1155     _("WMS download failed:\n\n"
1156     "GetCapabilities failed"));
1157     } else {
1158     xmlDoc *doc = NULL;
1159    
1160     LIBXML_TEST_VERSION;
1161    
1162     /* parse the file and get the DOM */
1163     if((doc = xmlReadMemory(cap, strlen(cap), NULL, NULL, 0)) == NULL) {
1164     xmlErrorPtr errP = xmlGetLastError();
1165     errorf(GTK_WIDGET(appdata->window),
1166     _("WMS download failed:\n\n"
1167     "XML error while parsing capabilities:\n"
1168     "%s"), errP->message);
1169     } else {
1170     printf("ok, parse doc tree\n");
1171    
1172     wms_cap_parse_doc(wms, doc);
1173     }
1174    
1175     g_free(cap);
1176     }
1177    
1178     /* ------------ basic checks ------------- */
1179    
1180     if(!wms->cap || !wms->service || !wms->cap->layer ||
1181     !wms->cap->request || !wms->cap->request->getmap) {
1182     errorf(GTK_WIDGET(appdata->window), _("Incomplete/unexpected reply!"));
1183     wms_free(wms);
1184     return;
1185     }
1186    
1187     if(!wms->cap->request->getmap->format) {
1188     errorf(GTK_WIDGET(appdata->window), _("No supported image format found."));
1189     wms_free(wms);
1190     return;
1191     }
1192    
1193     /* ---------- evaluate layers ----------- */
1194    
1195     wms_layer_t *layer = wms_get_requestable_layers(wms);
1196     gboolean at_least_one_ok = wms_one_layer_is_usable(appdata->project, layer);
1197    
1198     if(!at_least_one_ok) {
1199     errorf(GTK_WIDGET(appdata->window),
1200 harbaum 296 _("Server provides no data in the required format!\n\n"
1201 harbaum 1 "(epsg4326 and LatLonBoundingBox are mandatory for osm2go)"));
1202 harbaum 297 #if 0
1203 harbaum 1 wms_layer_free(layer);
1204     wms_free(wms);
1205     return;
1206 harbaum 297 #endif
1207 harbaum 1 }
1208    
1209     if(!wms_layer_dialog(appdata, layer)) {
1210     wms_layer_free(layer);
1211     wms_free(wms);
1212     return;
1213     }
1214    
1215     /* --------- build getmap request ----------- */
1216    
1217     /* get required image size */
1218     wms_setup_extent(appdata->project, wms);
1219    
1220     /* start building url */
1221     url = g_strdup_printf("%s%s"
1222     "%sSERVICE=wms"
1223 harbaum 14 // "&WMTVER=1.1.1"
1224     "&VERSION=1.1.1"
1225 harbaum 1 "&REQUEST=GetMap"
1226     "&LAYERS=",
1227     wms->server, wms->path, append_char);
1228    
1229     /* append layers */
1230     char *old;
1231     wms_layer_t *t = layer;
1232     gint cnt = 0;
1233     while(t) {
1234     if(t->selected) {
1235     old = url;
1236     url = g_strconcat(url, (!cnt)?"":",", t->name, NULL);
1237     g_free(old);
1238     cnt++;
1239     }
1240     t = t->next;
1241     }
1242 harbaum 297
1243     /* uses epsg4326 if possible */
1244     char *srs = NULL;
1245     if(layer->epsg4326)
1246     srs = g_strdup("EPSG:4326");
1247     else if(layer->srs)
1248     srs = g_strdup(layer->srs);
1249    
1250 harbaum 1 wms_layer_free(layer);
1251    
1252     /* append styles entry */
1253     old = url;
1254     url = g_strconcat(url, "&STYLES=", NULL);
1255     g_free(old);
1256    
1257     while(--cnt) {
1258     old = url;
1259     url = g_strconcat(url, ",", NULL);
1260     g_free(old);
1261     }
1262    
1263     /* and append rest */
1264     char minlon[G_ASCII_DTOSTR_BUF_SIZE], minlat[G_ASCII_DTOSTR_BUF_SIZE];
1265     char maxlon[G_ASCII_DTOSTR_BUF_SIZE], maxlat[G_ASCII_DTOSTR_BUF_SIZE];
1266    
1267 harbaum 14 /* build strings of min and max lat and lon to be used in url */
1268 harbaum 156 g_ascii_formatd(minlon, sizeof(minlon), LL_FORMAT,
1269     appdata->project->min.lon);
1270     g_ascii_formatd(minlat, sizeof(minlat), LL_FORMAT,
1271     appdata->project->min.lat);
1272     g_ascii_formatd(maxlon, sizeof(maxlon), LL_FORMAT,
1273     appdata->project->max.lon);
1274     g_ascii_formatd(maxlat, sizeof(maxlat), LL_FORMAT,
1275     appdata->project->max.lat);
1276 harbaum 1
1277 harbaum 14 /* find preferred supported video format */
1278 harbaum 1 gint format = 0;
1279     while(!(wms->cap->request->getmap->format & (1<<format)))
1280     format++;
1281    
1282     const char *formats[] = { "image/jpg", "image/jpeg",
1283     "image/png", "image/gif" };
1284    
1285 harbaum 14 /* build complete url */
1286 harbaum 1 old = url;
1287 harbaum 297
1288     url = g_strdup_printf("%s&SRS=%s&BBOX=%s,%s,%s,%s"
1289     "&WIDTH=%d&HEIGHT=%d&FORMAT=%s"
1290     "&reaspect=false", url, srs,
1291     minlon, minlat, maxlon, maxlat, wms->width,
1292     wms->height, formats[format]);
1293     if(srs) g_free(srs);
1294 harbaum 1 g_free(old);
1295    
1296     const char *exts[] = { "jpg", "jpg", "png", "gif" };
1297     char *filename = g_strdup_printf("%s/wms.%s", appdata->project->path,
1298     exts[format]);
1299    
1300    
1301     /* remove any existing image before */
1302     wms_remove(appdata);
1303    
1304 harbaum 169 if(!net_io_download_file(GTK_WIDGET(appdata->window), appdata->settings,
1305     url, filename)) {
1306 harbaum 1 g_free(filename);
1307     g_free(url);
1308     wms_free(wms);
1309     return;
1310     }
1311    
1312     /* there should be a matching file on disk now */
1313     map_set_bg_image(appdata->map, filename);
1314    
1315 harbaum 321 gint x = appdata->osm->bounds->min.x + appdata->map->bg.offset.x;
1316     gint y = appdata->osm->bounds->min.y + appdata->map->bg.offset.y;
1317     canvas_image_move(appdata->map->bg.item, x, y,
1318     appdata->map->bg.scale.x, appdata->map->bg.scale.y);
1319    
1320    
1321 harbaum 1 g_free(filename);
1322     g_free(url);
1323    
1324     /* --------- free wms structure -----------------*/
1325     wms_free(wms);
1326    
1327     gtk_widget_set_sensitive(appdata->menu_item_wms_clear, TRUE);
1328     gtk_widget_set_sensitive(appdata->menu_item_wms_adjust, TRUE);
1329     }
1330    
1331     /* try to load an existing image into map */
1332     void wms_load(appdata_t *appdata) {
1333     const char *exts[] = { "png", "gif", "jpg", "" };
1334     int i=0;
1335    
1336     while(exts[i][0]) {
1337     char *filename = g_strdup_printf("%s/wms.%s", appdata->project->path,
1338     exts[i]);
1339    
1340     if(g_file_test(filename, G_FILE_TEST_EXISTS)) {
1341     appdata->map->bg.offset.x = appdata->project->wms_offset.x;
1342     appdata->map->bg.offset.y = appdata->project->wms_offset.y;
1343    
1344     map_set_bg_image(appdata->map, filename);
1345    
1346     /* restore image to saved position */
1347     gint x = appdata->osm->bounds->min.x + appdata->map->bg.offset.x;
1348     gint y = appdata->osm->bounds->min.y + appdata->map->bg.offset.y;
1349     canvas_image_move(appdata->map->bg.item, x, y,
1350     appdata->map->bg.scale.x, appdata->map->bg.scale.y);
1351    
1352     g_free(filename);
1353    
1354     gtk_widget_set_sensitive(appdata->menu_item_wms_clear, TRUE);
1355     gtk_widget_set_sensitive(appdata->menu_item_wms_adjust, TRUE);
1356    
1357     return;
1358     }
1359     g_free(filename);
1360     i++;
1361     }
1362     }
1363    
1364 harbaum 326 void wms_remove_file(project_t *project) {
1365 harbaum 1 const char *exts[] = { "png", "gif", "jpg", "" };
1366     int i=0;
1367    
1368 harbaum 326 while(exts[i][0]) {
1369     char *filename =
1370     g_strdup_printf("%s/wms.%s", project->path, exts[i]);
1371    
1372     if(g_file_test(filename, G_FILE_TEST_EXISTS))
1373     g_remove(filename);
1374    
1375     g_free(filename);
1376     i++;
1377     }
1378     }
1379    
1380     void wms_remove(appdata_t *appdata) {
1381    
1382 harbaum 1 /* this cancels any wms adjustment in progress */
1383     if(appdata->map->action.type == MAP_ACTION_BG_ADJUST)
1384     map_action_cancel(appdata);
1385    
1386     gtk_widget_set_sensitive(appdata->menu_item_wms_clear, FALSE);
1387     gtk_widget_set_sensitive(appdata->menu_item_wms_adjust, FALSE);
1388    
1389     map_remove_bg_image(appdata->map);
1390    
1391 harbaum 326 wms_remove_file(appdata->project);
1392 harbaum 1 }
1393    
1394 harbaum 14 struct server_preset_s {
1395     char *name, *server, *path;
1396     } default_servers[] = {
1397     { "NASA landsat", "http://onearth.jpl.nasa.gov", "/wms.cgi" },
1398     /* add more servers here ... */
1399     { NULL, NULL, NULL }
1400     };
1401    
1402     wms_server_t *wms_server_get_default(void) {
1403     wms_server_t *server = NULL, **cur = &server;
1404     struct server_preset_s *preset = default_servers;
1405    
1406     while(preset->name) {
1407     *cur = g_new0(wms_server_t, 1);
1408     (*cur)->name = g_strdup(preset->name);
1409     (*cur)->server = g_strdup(preset->server);
1410     (*cur)->path = g_strdup(preset->path);
1411     cur = &(*cur)->next;
1412     preset++;
1413     }
1414    
1415     return server;
1416     }
1417    
1418     void wms_server_free(wms_server_t *wms_server) {
1419     if(wms_server->name) g_free(wms_server->name);
1420     if(wms_server->server) g_free(wms_server->server);
1421     if(wms_server->path) g_free(wms_server->path);
1422     g_free(wms_server);
1423     }
1424    
1425     void wms_servers_free(wms_server_t *wms_server) {
1426     while(wms_server) {
1427     wms_server_t *next = wms_server->next;
1428     wms_server_free(wms_server);
1429     wms_server = next;
1430     }
1431     }