Routino SVN Repository Browser

Check out the latest version of Routino: svn co http://routino.org/svn/trunk routino

ViewVC logotype

Contents of /trunk/extras/find-fixme/web/www/fixme.openlayers2.js

Parent Directory Parent Directory | Revision Log Revision Log


Revision 2006 - (show annotations) (download) (as text)
Sat Aug 31 14:31:34 2019 UTC (5 years, 7 months ago) by amb
File MIME type: application/javascript
File size: 15699 byte(s)
Update Leaflet and OpenLayers v2 scripts based on improvements made in
the new OpenLayers scripts.

1 //
2 // Routino (extras) fixme web page Javascript
3 //
4 // Part of the Routino routing software.
5 //
6 // This file Copyright 2008-2014, 2019 Andrew M. Bishop
7 //
8 // This program is free software: you can redistribute it and/or modify
9 // it under the terms of the GNU Affero General Public License as published by
10 // the Free Software Foundation, either version 3 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 // GNU Affero General Public License for more details.
17 //
18 // You should have received a copy of the GNU Affero General Public License
19 // along with this program. If not, see <http://www.gnu.org/licenses/>.
20 //
21
22
23 ////////////////////////////////////////////////////////////////////////////////
24 /////////////////////////////// Initialisation /////////////////////////////////
25 ////////////////////////////////////////////////////////////////////////////////
26
27 // Process the URL query string and extract the arguments
28
29 var legal={"^lon" : "^[-0-9.]+$",
30 "^lat" : "^[-0-9.]+$",
31 "^zoom" : "^[0-9]+$"};
32
33 var args={};
34
35 if(location.search.length>1)
36 {
37 var query,queries;
38
39 query=location.search.replace(/^\?/,"");
40 query=query.replace(/;/g,"&");
41 queries=query.split("&");
42
43 for(var i=0;i<queries.length;i++)
44 {
45 queries[i].match(/^([^=]+)(=(.*))?$/);
46
47 var k=RegExp.$1;
48 var v=decodeURIComponent(RegExp.$3);
49
50 for(var l in legal)
51 {
52 if(k.match(RegExp(l)) && v.match(RegExp(legal[l])))
53 args[k]=v;
54 }
55 }
56 }
57
58
59 ////////////////////////////////////////////////////////////////////////////////
60 ///////////////////////////////// Map handling /////////////////////////////////
61 ////////////////////////////////////////////////////////////////////////////////
62
63 var map;
64 var layerMap=[], layerHighlights, layerVectors, layerBoxes;
65 var epsg4326, epsg900913;
66 var box;
67 var select;
68
69 //
70 // Initialise the 'map' object
71 //
72
73 function map_init() // called from fixme.html
74 {
75 // Create the map (Map URLs and limits are in mapprops.js)
76
77 epsg4326=new OpenLayers.Projection("EPSG:4326");
78 epsg900913=new OpenLayers.Projection("EPSG:900913");
79
80 map = new OpenLayers.Map ("map",
81 {
82 controls:[
83 new OpenLayers.Control.Navigation(),
84 new OpenLayers.Control.PanZoomBar(),
85 new OpenLayers.Control.ScaleLine(),
86 new OpenLayers.Control.LayerSwitcher()
87 ],
88
89 projection: epsg900913,
90 displayProjection: epsg4326,
91
92 minZoomLevel: mapprops.zoomout,
93 numZoomLevels: mapprops.zoomin-mapprops.zoomout+1,
94 maxResolution: 156543.03390625 / Math.pow(2,mapprops.zoomout),
95
96 restrictedExtent: new OpenLayers.Bounds(mapprops.westedge,mapprops.southedge,mapprops.eastedge,mapprops.northedge).transform(epsg4326,epsg900913)
97 });
98
99 // Get a URL for the tile (mostly copied from OpenLayers/Layer/XYZ.js).
100
101 function limitedUrl(bounds)
102 {
103 var res = this.map.getResolution();
104
105 var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w));
106 var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h));
107 var z = this.map.getZoom() + this.map.minZoomLevel;
108
109 var limit = Math.pow(2, z);
110 x = ((x % limit) + limit) % limit;
111
112 var xyz = {"x": x, "y": y, "z": z};
113 var url = this.url;
114
115 if (OpenLayers.Util.isArray(url))
116 {
117 var s = "" + xyz.x + xyz.y + xyz.z;
118 url = this.selectUrl(s, url);
119 }
120
121 return OpenLayers.String.format(url, xyz);
122 }
123
124 // Add map tile layers
125
126 for(var l=0; l<mapprops.mapdata.length; l++)
127 {
128 var urls;
129
130 if(OpenLayers.Util.isArray(mapprops.mapdata[l].tiles.subdomains))
131 {
132 urls=[];
133
134 for(var s=0; s<mapprops.mapdata[l].tiles.subdomains.length; s++)
135 urls.push(mapprops.mapdata[l].tiles.url.replace(/\${s}/,mapprops.mapdata[l].tiles.subdomains[s]));
136 }
137 else
138 urls=mapprops.mapdata[l].tiles.url;
139
140 layerMap[l] = new OpenLayers.Layer.TMS(mapprops.mapdata[l].label,
141 urls,
142 {
143 getURL: limitedUrl,
144 displayOutsideMaxExtent: true,
145 buffer: 1
146 });
147 map.addLayer(layerMap[l]);
148 }
149
150 // Update the attribution if the layer changes
151
152 function change_attribution_event(event)
153 {
154 for(var l=0; l<mapprops.mapdata.length; l++)
155 if(layerMap[l] == event.layer)
156 change_attribution(l);
157 }
158
159 map.events.register("changelayer",layerMap,change_attribution_event);
160
161 function change_attribution(l)
162 {
163 var data_url =mapprops.mapdata[l].attribution.data_url;
164 var data_text=mapprops.mapdata[l].attribution.data_text;
165 var tile_url =mapprops.mapdata[l].attribution.tile_url;
166 var tile_text=mapprops.mapdata[l].attribution.tile_text;
167
168 document.getElementById("attribution_data").innerHTML="<a href=\"" + data_url + "\" target=\"data_attribution\">" + data_text + "</a>";
169 document.getElementById("attribution_tile").innerHTML="<a href=\"" + tile_url + "\" target=\"tile_attribution\">" + tile_text + "</a>";
170 }
171
172 change_attribution(0);
173
174 // Add two vectors layers (one for highlights that display behind the vectors)
175
176 layerHighlights = new OpenLayers.Layer.Vector("Highlights",{displayInLayerSwitcher: false});
177 map.addLayer(layerHighlights);
178
179 layerVectors = new OpenLayers.Layer.Vector("Markers",{displayInLayerSwitcher: false});
180 map.addLayer(layerVectors);
181
182 // Handle feature selection and popup
183
184 select = new OpenLayers.Control.SelectFeature(layerVectors,
185 {onSelect: selectFeature, onUnselect: unselectFeature});
186
187 map.addControl(select);
188 select.activate();
189
190 createPopup();
191
192 // Add a boxes layer
193
194 layerBoxes = new OpenLayers.Layer.Boxes("Boundary",{displayInLayerSwitcher: false});
195 map.addLayer(layerBoxes);
196
197 box=null;
198
199 // Move the map
200
201 map.events.register("moveend", map, updateURLs);
202
203 var lon =args["lon"];
204 var lat =args["lat"];
205 var zoom=args["zoom"];
206
207 if(lon !== undefined && lat !== undefined && zoom !== undefined)
208 {
209 if(lon<mapprops.westedge) lon=mapprops.westedge;
210 if(lon>mapprops.eastedge) lon=mapprops.eastedge;
211
212 if(lat<mapprops.southedge) lat=mapprops.southedge;
213 if(lat>mapprops.northedge) lat=mapprops.northedge;
214
215 if(zoom<mapprops.zoomout) zoom=mapprops.zoomout;
216 if(zoom>mapprops.zoomin) zoom=mapprops.zoomin;
217
218 var lonlat = new OpenLayers.LonLat(lon,lat);
219 lonlat.transform(epsg4326,epsg900913);
220
221 map.moveTo(lonlat,zoom-map.minZoomLevel);
222 }
223 else
224 {
225 map.setCenter(map.restrictedExtent.getCenterLonLat(), map.getZoomForExtent(map.restrictedExtent,true));
226 map.maxResolution = map.getResolution();
227 }
228
229 // Unhide editing URL if variable set
230
231 if(mapprops.editurl !== undefined && mapprops.editurl !== "")
232 {
233 var edit_url=document.getElementById("edit_url");
234
235 edit_url.style.display="";
236 edit_url.href=mapprops.editurl;
237 }
238
239 updateURLs();
240 }
241
242
243 //
244 // Format a number in printf("%.5f") format.
245 //
246
247 function format5f(number)
248 {
249 var newnumber=Math.floor(number*100000+0.5);
250 var delta=0;
251
252 if(newnumber>=0 && newnumber<100000) delta= 100000;
253 if(newnumber<0 && newnumber>-100000) delta=-100000;
254
255 var string=String(newnumber+delta);
256
257 var intpart =string.substring(0,string.length-5);
258 var fracpart=string.substring(string.length-5,string.length);
259
260 if(delta>0) intpart="0";
261 if(delta<0) intpart="-0";
262
263 return(intpart + "." + fracpart);
264 }
265
266
267 //
268 // Build a set of URL arguments for the map location
269 //
270
271 function buildMapArguments()
272 {
273 var lonlat = map.getCenter().clone();
274 lonlat.transform(epsg900913,epsg4326);
275
276 var zoom = map.getZoom() + map.minZoomLevel;
277
278 return "lat=" + format5f(lonlat.lat) + ";lon=" + format5f(lonlat.lon) + ";zoom=" + zoom;
279 }
280
281
282 //
283 // Update the URLs
284 //
285
286 function updateURLs()
287 {
288 var mapargs=buildMapArguments();
289
290 var links=document.getElementsByTagName("a");
291
292 for(var i=0; i<links.length; i++)
293 {
294 var element=links[i];
295
296 if(element.id == "permalink_url")
297 element.href=location.pathname + "?" + mapargs;
298
299 if(element.id == "edit_url")
300 element.href=mapprops.editurl + "?" + mapargs;
301 }
302 }
303
304
305 ////////////////////////////////////////////////////////////////////////////////
306 ///////////////////////// Popup and selection handling /////////////////////////
307 ////////////////////////////////////////////////////////////////////////////////
308
309 var popup=null;
310
311 //
312 // Create a popup - independent of map because want it fixed on screen not fixed on map.
313 //
314
315 function createPopup()
316 {
317 popup=document.createElement("div");
318
319 popup.className = "popup";
320
321 popup.innerHTML = "<span></span>";
322
323 popup.style.display = "none";
324
325 popup.style.position = "fixed";
326 popup.style.top = "-4000px";
327 popup.style.left = "-4000px";
328 popup.style.zIndex = "100";
329
330 popup.style.padding = "5px";
331
332 popup.style.opacity=0.85;
333 popup.style.backgroundColor="#C0C0C0";
334 popup.style.border="4px solid #404040";
335
336 document.body.appendChild(popup);
337 }
338
339
340 //
341 // Draw a popup - independent of map because want it fixed on screen not fixed on map.
342 //
343
344 function drawPopup(html)
345 {
346 if(html===null)
347 {
348 popup.style.display="none";
349 return;
350 }
351
352 if(popup.style.display=="none")
353 {
354 var map_div=document.getElementById("map");
355
356 popup.style.left =map_div.offsetParent.offsetLeft+map_div.offsetLeft+60 + "px";
357 popup.style.top = map_div.offsetTop +30 + "px";
358 popup.style.width =map_div.clientWidth-120 + "px";
359
360 popup.style.display="";
361 }
362
363 var close="<span style='float: right; cursor: pointer;' onclick='drawPopup(null)'>X</span>";
364
365 popup.innerHTML=close+html;
366 }
367
368
369 //
370 // Select a feature
371 //
372
373 function selectFeature(feature)
374 {
375 if(feature.attributes.dump)
376 ajaxGET("fixme.cgi?dump=" + feature.attributes.dump, runDumpSuccess);
377
378 layerHighlights.destroyFeatures();
379
380 var highlight_style = new OpenLayers.Style({},{strokeColor: "#F0F000",strokeWidth: 8,
381 fillColor: "#F0F000",pointRadius: 4});
382
383 var highlight = new OpenLayers.Feature.Vector(feature.geometry.clone(),{},highlight_style);
384
385 layerHighlights.addFeatures([highlight]);
386 }
387
388
389 //
390 // Un-select a feature
391 //
392
393 function unselectFeature(feature)
394 {
395 layerHighlights.destroyFeatures();
396
397 drawPopup(null);
398 }
399
400
401 //
402 // Display the dump data
403 //
404
405 function runDumpSuccess(response)
406 {
407 var string=response.responseText;
408
409 if(mapprops.editurl !== undefined && mapprops.editurl !== "")
410 {
411 var types=["node", "way", "relation"];
412
413 for(var t in types)
414 {
415 var type=types[t];
416
417 var regexp=RegExp(type + " id=&#39;([0-9]+)&#39;");
418
419 var match=string.match(regexp);
420
421 if(match !== null)
422 {
423 var id=match[1];
424
425 string=string.replace(regexp,type + " id=&#39;<a href='" + mapprops.browseurl + "/" + type + "/" + id + "' target='" + type + id + "'>" + id + "</a>&#39;");
426 }
427 }
428 }
429
430 drawPopup(string.split("&gt;&lt;").join("&gt;<br>&lt;").split("<br>&lt;tag").join("<br>&nbsp;&nbsp;&lt;tag"));
431 }
432
433
434 ////////////////////////////////////////////////////////////////////////////////
435 /////////////////////////////// Server handling ////////////////////////////////
436 ////////////////////////////////////////////////////////////////////////////////
437
438 //
439 // Define an AJAX request object
440 //
441
442 function ajaxGET(url,success,failure,state)
443 {
444 var ajaxRequest=new XMLHttpRequest();
445
446 function ajaxGOT(options) {
447 if(this.readyState==4)
448 if(this.status==200)
449 { if(typeof(options.success)=="function") options.success(this,options.state); }
450 else
451 { if(typeof(options.failure)=="function") options.failure(this,options.state); }
452 }
453
454 ajaxRequest.onreadystatechange = function(){ ajaxGOT.call(ajaxRequest,{success: success, failure: failure, state: state}); };
455 ajaxRequest.open("GET", url, true);
456 ajaxRequest.send(null);
457 }
458
459
460 //
461 // Display the status
462 //
463
464 function displayStatus(type,subtype,content)
465 {
466 var child=document.getElementById("result_status").firstChild;
467
468 do
469 {
470 if(child.id !== undefined)
471 child.style.display="none";
472
473 child=child.nextSibling;
474 }
475 while(child !== null);
476
477 var chosen_status=document.getElementById("result_status_" + type);
478
479 chosen_status.style.display="";
480
481 if(subtype !== undefined)
482 {
483 var format_status=document.getElementById("result_status_" + subtype).innerHTML;
484
485 chosen_status.innerHTML=format_status.replace("#",String(content));
486 }
487 }
488
489
490 //
491 // Display data statistics
492 //
493
494 function displayStatistics()
495 {
496 // Use AJAX to get the statistics
497
498 ajaxGET("fixme.cgi?statistics=yes", runStatisticsSuccess);
499 }
500
501
502 //
503 // Success in running data statistics generation.
504 //
505
506 function runStatisticsSuccess(response)
507 {
508 document.getElementById("statistics_data").innerHTML="<pre>" + response.responseText + "</pre>";
509 document.getElementById("statistics_link").style.display="none";
510 }
511
512
513 //
514 // Get the requested data
515 //
516
517 function displayData(datatype) // called from fixme.html
518 {
519 // Delete the old data
520
521 unselectFeature();
522
523 select.deactivate();
524
525 layerVectors.destroyFeatures();
526 layerHighlights.destroyFeatures();
527
528 if(box !== null)
529 layerBoxes.removeMarker(box);
530 box=null;
531
532 // Print the status
533
534 displayStatus("no_data");
535
536 // Return if just here to clear the data
537
538 if(datatype === "")
539 return;
540
541 // Get the new data
542
543 var mapbounds=map.getExtent().clone();
544 mapbounds.transform(epsg900913,epsg4326);
545
546 var url="fixme.cgi";
547
548 url=url + "?lonmin=" + format5f(mapbounds.left);
549 url=url + ";latmin=" + format5f(mapbounds.bottom);
550 url=url + ";lonmax=" + format5f(mapbounds.right);
551 url=url + ";latmax=" + format5f(mapbounds.top);
552 url=url + ";data=" + datatype;
553
554 // Use AJAX to get the data
555
556 ajaxGET(url, runFixmeSuccess, runFailure);
557 }
558
559
560 //
561 // Add a bounding box
562 //
563
564 function addBox(words)
565 {
566 var lat1=words[0];
567 var lon1=words[1];
568 var lat2=words[2];
569 var lon2=words[3];
570
571 var bounds = new OpenLayers.Bounds(lon1,lat1,lon2,lat2).transform(epsg4326,epsg900913);
572
573 box = new OpenLayers.Marker.Box(bounds);
574
575 layerBoxes.addMarker(box);
576 }
577
578
579 //
580 // Success in getting the error log data
581 //
582
583 function runFixmeSuccess(response)
584 {
585 var lines=response.responseText.split("\n");
586
587 var style = new OpenLayers.Style({},{stroke: false,
588 pointRadius: 3,fillColor: "#FF0000",
589 cursor: "pointer"});
590
591 var features=[];
592
593 for(var line=0;line<lines.length;line++)
594 {
595 var words=lines[line].split(" ");
596
597 if(line === 0)
598 addBox(words);
599 else if(words[0] !== "")
600 {
601 var dump=words[0];
602 var lat=words[1];
603 var lon=words[2];
604
605 var lonlat = new OpenLayers.LonLat(lon,lat).transform(epsg4326,epsg900913);
606
607 var point = new OpenLayers.Geometry.Point(lonlat.lon,lonlat.lat);
608
609 features.push(new OpenLayers.Feature.Vector(point,{dump: dump},style));
610 }
611 }
612
613 select.activate();
614
615 layerVectors.addFeatures(features);
616
617 displayStatus("data","fixme",lines.length-2);
618 }
619
620
621 //
622 // Failure in getting data.
623 //
624
625 function runFailure(response)
626 {
627 displayStatus("failed");
628 }