Routino SVN Repository Browser

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

ViewVC logotype

Contents of /trunk/web/www/routino/router.leaflet.js

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1949 - (show annotations) (download) (as text)
Fri Jun 15 18:33:09 2018 UTC (6 years, 9 months ago) by amb
File MIME type: application/javascript
File size: 52801 byte(s)
Initialise the router map web page with the search form for locations
rather than coordinates.  Dragging a marker to the map converts it to
coordinates.

1 //
2 // Routino router web page Javascript
3 //
4 // Part of the Routino routing software.
5 //
6 // This file Copyright 2008-2018 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 var vismarkers, markers, icons, markersmoved, paramschanged;
24 var homelat=null, homelon=null;
25
26
27 ////////////////////////////////////////////////////////////////////////////////
28 /////////////////////////////// Initialisation /////////////////////////////////
29 ////////////////////////////////////////////////////////////////////////////////
30
31 // Make a deep copy of the routino profile.
32
33 var routino_default={};
34 for(var l1 in routino)
35 if(typeof(routino[l1])!="object")
36 routino_default[l1]=routino[l1];
37 else
38 {
39 routino_default[l1]={};
40 for(var l2 in routino[l1])
41 if(typeof(routino[l1][l2])!="object")
42 routino_default[l1][l2]=Number(routino[l1][l2]);
43 else
44 {
45 routino_default[l1][l2]={};
46 for(var l3 in routino[l1][l2])
47 routino_default[l1][l2][l3]=Number(routino[l1][l2][l3]);
48 }
49 }
50
51 // Store the latitude and longitude in the routino variable
52
53 routino.point=[];
54 for(var marker=1;marker<=mapprops.maxmarkers;marker++)
55 {
56 routino.point[marker]={};
57
58 routino.point[marker].lon="";
59 routino.point[marker].lat="";
60 routino.point[marker].search="";
61 routino.point[marker].active=false;
62 routino.point[marker].used=false;
63 routino.point[marker].home=false;
64 }
65
66 // Process the URL query string and extract the arguments
67
68 var legal={"^lon" : "^[-0-9.]+$",
69 "^lat" : "^[-0-9.]+$",
70 "^zoom" : "^[0-9]+$",
71
72 "^lon[1-9]" : "^[-0-9.]+$",
73 "^lat[1-9]" : "^[-0-9.]+$",
74 "^search[1-9]" : "^.+$",
75 "^transport" : "^[a-z]+$",
76 "^highway-[a-z]+" : "^[0-9.]+$",
77 "^speed-[a-z]+" : "^[0-9.]+$",
78 "^property-[a-z]+" : "^[0-9.]+$",
79 "^oneway" : "^(1|0|true|false|on|off)$",
80 "^turns" : "^(1|0|true|false|on|off)$",
81 "^weight" : "^[0-9.]+$",
82 "^height" : "^[0-9.]+$",
83 "^width" : "^[0-9.]+$",
84 "^length" : "^[0-9.]+$",
85
86 "^language" : "^[-a-zA-Z]+$",
87
88 "^reverse" : "(1|0|true|false|on|off)",
89 "^loop" : "(1|0|true|false|on|off)"};
90
91 var args={};
92
93 if(location.search.length>1)
94 {
95 var query,queries;
96
97 query=location.search.replace(/^\?/,"");
98 query=query.replace(/;/g,"&");
99 queries=query.split("&");
100
101 for(var i=0;i<queries.length;i++)
102 {
103 queries[i].match(/^([^=]+)(=(.*))?$/);
104
105 var k=RegExp.$1;
106 var v=decodeURIComponent(RegExp.$3);
107
108 for(var l in legal)
109 {
110 if(k.match(RegExp(l)) && v.match(RegExp(legal[l])))
111 args[k]=v;
112 }
113 }
114 }
115
116
117 //
118 // Fill in the HTML - add the missing waypoints
119 //
120
121 function html_init() // called from router.html
122 {
123 var waypoints=document.getElementById("waypoints");
124
125 var waypoint_html=waypoints.firstElementChild.outerHTML.split("XXX");
126
127 waypoints.removeChild(waypoints.firstElementChild);
128
129 for(var marker=1;marker<=mapprops.maxmarkers;marker++)
130 {
131 var waypoint=document.createElement('div');
132
133 waypoints.appendChild(waypoint);
134
135 waypoint.outerHTML=waypoint_html.join(marker);
136 }
137
138 waypoints.addEventListener('dragstart',dragWaypointStart,false);
139 waypoints.addEventListener('dragend' ,dragWaypointEnd ,false);
140 waypoints.addEventListener('dragenter',dragWaypointEnter,false);
141 waypoints.addEventListener('dragover' ,dragWaypointOver ,false);
142 waypoints.addEventListener('dragleave',dragWaypointLeave,false);
143 waypoints.addEventListener('drop' ,dragWaypointDrop ,false);
144
145
146 var map=document.getElementById("map");
147
148 map.addEventListener('dragenter',dragWaypointMapEnter,false);
149 map.addEventListener('dragover' ,dragWaypointMapOver ,false);
150 map.addEventListener('dragleave',dragWaypointMapLeave,false);
151 map.addEventListener('drop' ,dragWaypointMapDrop ,false);
152 }
153
154
155 ////////////////////////////////////////////////////////////////////////////////
156 //////////////////////////////// Form handling /////////////////////////////////
157 ////////////////////////////////////////////////////////////////////////////////
158
159 //
160 // Form initialisation - fill in the uninitialised parts
161 //
162
163 function form_init() // called from router.html
164 {
165 // Fill in the waypoints
166
167 vismarkers=0;
168
169 for(var marker=mapprops.maxmarkers;marker>=1;marker--)
170 {
171 var lon=args["lon" + marker];
172 var lat=args["lat" + marker];
173 var search=args["search" + marker];
174
175 if(lon !== undefined && lat !== undefined && search !== undefined && lon !== "" && lat !== "" && search !== "")
176 {
177 markerAddForm(marker);
178
179 formSetSearch(marker,search);
180 formSetCoords(marker,lon,lat);
181
182 markerAddMap(marker);
183
184 markerSearch(marker);
185
186 vismarkers++;
187 }
188 else if(lon !== undefined && lat !== undefined && lon !== "" && lat !== "")
189 {
190 markerAddForm(marker);
191
192 formSetCoords(marker,lon,lat);
193
194 markerAddMap(marker);
195
196 markerCoords(marker);
197
198 vismarkers++;
199 }
200 else if(search !== undefined && search !== "")
201 {
202 markerAddForm(marker);
203
204 formSetSearch(marker,search);
205
206 markerSearch(marker);
207
208 DoSearch(marker);
209
210 vismarkers++;
211 }
212 else if(vismarkers || marker<=2)
213 {
214 markerAddForm(marker);
215
216 vismarkers++;
217 }
218
219 var searchfield=document.forms["form"].elements["search" + marker];
220
221 if(searchfield.addEventListener)
222 searchfield.addEventListener("keyup", searchOnReturnKey, false);
223 else if(searchfield.attachEvent)
224 searchfield.attachEvent("keyup", searchOnReturnKey); // Internet Explorer
225 }
226
227 if(args["loop"] !== undefined)
228 formSetLoopReverse("loop",args["loop"]);
229 else
230 formSetLoopReverse("loop",false);
231
232 if(args["reverse"] !== undefined)
233 formSetLoopReverse("reverse",args["reverse"]);
234 else
235 formSetLoopReverse("reverse",false);
236
237 // Update the transport type with the URL settings which updates all HTML forms to defaults.
238
239 var transport=routino.transport;
240
241 if(args["transport"] !== undefined)
242 transport=args["transport"];
243
244 formSetTransport(transport);
245
246 // Update the HTML with the URL settings
247
248 if(args["language"] !== undefined)
249 formSetLanguage(args["language"]);
250 else
251 formSetLanguage();
252
253 for(var key in routino.profile_highway)
254 if(args["highway-" + key] !== undefined)
255 formSetHighway(key,args["highway-" + key]);
256
257 for(var key in routino.profile_speed)
258 if(args["speed-" + key] !== undefined)
259 formSetSpeed(key,args["speed-" + key]);
260
261 for(var key in routino.profile_property)
262 if(args["property-" + key] !== undefined)
263 formSetProperty(key,args["property-" + key]);
264
265 for(var key in routino.restrictions)
266 {
267 if(key=="oneway" || key=="turns")
268 {
269 if(args[key] !== undefined)
270 formSetRestriction(key,args[key]);
271 }
272 else
273 {
274 if(args["restrict-" + key] !== undefined)
275 formSetRestriction(key,args["restrict-" + key]);
276 }
277 }
278
279 // Get the home location cookie and compare to each waypoint
280
281 var cookies=document.cookie.split("; ");
282
283 for(var cookie=0;cookie<cookies.length;cookie++)
284 if(cookies[cookie].substr(0,"Routino-home".length)=="Routino-home")
285 {
286 var data=cookies[cookie].split(/[=:;]/);
287
288 if(data[1]=="lon") homelon=Number(data[2]);
289 if(data[3]=="lat") homelat=Number(data[4]);
290 }
291
292 if(homelon!==null && homelat!==null)
293 {
294 for(var m=1;m<=vismarkers;m++)
295 markerCheckHome(m);
296
297 // If the first location is empty and the cookie is set then fill it.
298
299 if(!routino.point[1].used)
300 markerMoveHome(1);
301 }
302
303 updateURLs();
304
305 updateSearchButtons();
306 }
307
308
309 //
310 // Function to perform the search if the return key is pressed.
311 // (using 'onchange' only triggers once and is confusing when clicking outside the field).
312 //
313
314 function searchOnReturnKey(ev)
315 {
316 if(ev.keyCode==13)
317 if(this.name.match(/^search([0-9]+)$/))
318 formSetSearch(RegExp.$1);
319
320 return(true);
321 }
322
323
324 //
325 // Change of language in the form
326 //
327
328 function formSetLanguage(value) // called from router.html (with no arguments)
329 {
330 if(value === undefined)
331 {
332 for(var lang=0;lang<document.forms["form"].elements["language"].length;lang++)
333 if(document.forms["form"].elements["language"][lang].checked)
334 routino.language=document.forms["form"].elements["language"][lang].value;
335 }
336 else
337 {
338 for(var lang=0;lang<document.forms["form"].elements["language"].length;lang++)
339 if(document.forms["form"].elements["language"][lang].value==value)
340 document.forms["form"].elements["language"][lang].checked=true;
341 else
342 document.forms["form"].elements["language"][lang].checked=false;
343
344 routino.language=value;
345 }
346
347 updateURLs();
348 }
349
350
351 //
352 // Change of transport in the form
353 //
354
355 function formSetTransport(value) // called from router.html
356 {
357 routino.transport=value;
358
359 for(var key in routino.transports)
360 document.forms["form"].elements["transport"][routino.transports[key]-1].checked=(key==routino.transport);
361
362 for(var key in routino.profile_highway)
363 document.forms["form"].elements["highway-" + key].value=routino.profile_highway[key][routino.transport];
364
365 for(var key in routino.profile_speed)
366 document.forms["form"].elements["speed-" + key].value=routino.profile_speed[key][routino.transport];
367
368 for(var key in routino.profile_property)
369 document.forms["form"].elements["property-" + key].value=routino.profile_property[key][routino.transport];
370
371 for(var key in routino.restrictions)
372 {
373 if(key=="oneway" || key=="turns")
374 document.forms["form"].elements["restrict-" + key].checked=routino.profile_restrictions[key][routino.transport];
375 else
376 document.forms["form"].elements["restrict-" + key].value=routino.profile_restrictions[key][routino.transport];
377 }
378
379 paramschanged=true;
380
381 updateURLs();
382 }
383
384
385 //
386 // Change of highway in the form
387 //
388
389 function formSetHighway(type,value) // called from router.html (with one argument)
390 {
391 if(value == "+")
392 {
393 value=routino.profile_highway[type][routino.transport];
394 value=10*Math.floor(value/10)+10;
395 }
396 else if(value == "-")
397 {
398 value=routino.profile_highway[type][routino.transport]-10;
399 value=10*Math.ceil(value/10)-10;
400 }
401 else if(value == "=")
402 value=document.forms["form"].elements["highway-" + type].value;
403
404 value=Number(value);
405 if(isNaN(value)) value= 50;
406 if(value>100) value=100;
407 if(value< 0) value= 0;
408
409 document.forms["form"].elements["highway-" + type].value=value;
410 routino.profile_highway[type][routino.transport]=value;
411
412 paramschanged=true;
413
414 updateURLs();
415 }
416
417
418 //
419 // Change of Speed in the form
420 //
421
422 function formSetSpeed(type,value) // called from router.html (with one argument)
423 {
424 if(value == "+")
425 {
426 value=routino.profile_speed[type][routino.transport];
427 if(value<10) value=2*Math.floor(value/2)+2;
428 else if(value<30) value=5*Math.floor(value/5)+5;
429 else value=10*Math.floor(value/10)+10;
430 }
431 else if(value == "-")
432 {
433 value=routino.profile_speed[type][routino.transport];
434 if(value<=10) value=2*Math.ceil(value/2)-2;
435 else if(value<=30) value=5*Math.ceil(value/5)-5;
436 else value=10*Math.ceil(value/10)-10;
437 }
438 else if(value == "=")
439 value=document.forms["form"].elements["speed-" + type].value;
440
441 value=Number(value);
442 if(isNaN(value)) value= 60;
443 if(value>150) value=150;
444 if(value< 0) value= 0;
445
446 document.forms["form"].elements["speed-" + type].value=value;
447 routino.profile_speed[type][routino.transport]=value;
448
449 paramschanged=true;
450
451 updateURLs();
452 }
453
454
455 //
456 // Change of Property in the form
457 //
458
459 function formSetProperty(type,value) // called from router.html (with one argument)
460 {
461 if(value == "+")
462 {
463 value=routino.profile_property[type][routino.transport];
464 if(value>=40 && value<60) value=2*Math.floor(value/2)+2;
465 else value=5*Math.floor(value/5)+5;
466 }
467 else if(value == "-")
468 {
469 value=routino.profile_property[type][routino.transport];
470 if(value>40 && value<=60) value=2*Math.ceil(value/2)-2;
471 else value=5*Math.ceil(value/5)-5;
472 }
473 else if(value == "=")
474 value=document.forms["form"].elements["property-" + type].value;
475
476 value=Number(value);
477 if(isNaN(value)) value= 50;
478 if(value>100) value=100;
479 if(value< 0) value= 0;
480
481 document.forms["form"].elements["property-" + type].value=value;
482 routino.profile_property[type][routino.transport]=value;
483
484 paramschanged=true;
485
486 updateURLs();
487 }
488
489
490 //
491 // Change of Restriction rule in the form
492 //
493
494 function formSetRestriction(type,value) // called from router.html (with one argument)
495 {
496 if(type=="oneway" || type=="turns")
497 {
498 if(value === undefined)
499 value=document.forms["form"].elements["restrict-" + type].checked;
500
501 document.forms["form"].elements["restrict-" + type].checked=value;
502 routino.profile_restrictions[type][routino.transport]=value;
503 }
504 else if(type=="weight")
505 {
506 if(value == "+")
507 value=routino.profile_restrictions[type][routino.transport]+5;
508 else if(value == "-")
509 value=routino.profile_restrictions[type][routino.transport]-5;
510 else if(value == "=")
511 value=document.forms["form"].elements["restrict-" + type].value;
512
513 value=Number(value);
514 if(isNaN(value)) value= 0;
515 if(value>50) value=50;
516 if(value< 0) value= 0;
517
518 document.forms["form"].elements["restrict-" + type].value=value;
519 routino.profile_restrictions[type][routino.transport]=value;
520 }
521 else /* if(type=="height" || type=="width" || type=="length") */
522 {
523 if(value == "+")
524 value=routino.profile_restrictions[type][routino.transport]+1;
525 else if(value == "-")
526 value=routino.profile_restrictions[type][routino.transport]-1;
527 else if(value == "=")
528 value=document.forms["form"].elements["restrict-" + type].value;
529
530 value=Number(value);
531 if(isNaN(value)) value= 0;
532 if(value>25) value=25;
533 if(value< 0) value= 0;
534
535 document.forms["form"].elements["restrict-" + type].value=value;
536 routino.profile_restrictions[type][routino.transport]=value;
537 }
538
539 paramschanged=true;
540
541 updateURLs();
542 }
543
544
545 //
546 // Set the feature coordinates from the form when the form changes.
547 //
548
549 function formSetCoords(marker,lon,lat) // called from router.html (with one argument)
550 {
551 clearSearchResult(marker);
552
553 if(lon === undefined && lat === undefined)
554 {
555 lon=document.forms["form"].elements["lon" + marker].value;
556 lat=document.forms["form"].elements["lat" + marker].value;
557 }
558
559 if(lon === "" && lat === "")
560 {
561 document.forms["form"].elements["lon" + marker].value="";
562 document.forms["form"].elements["lat" + marker].value="";
563
564 routino.point[marker].lon="";
565 routino.point[marker].lat="";
566
567 updateURLs();
568 }
569 else
570 {
571 var lonlat;
572
573 if(lon==="")
574 {
575 lonlat=map.getCenter();
576
577 lon=lonlat.lon;
578 }
579
580 if(lon<-180) lon=-180;
581 if(lon>+180) lon=+180;
582
583 if(lat==="")
584 {
585 lonlat=map.getCenter();
586
587 lat=lonlat.lat;
588 }
589
590 if(lat<-90 ) lat=-90 ;
591 if(lat>+90 ) lat=+90 ;
592
593 lonlat = L.latLng(lat,lon);
594
595 markers[marker].setLatLng(lonlat);
596
597 markersmoved=true;
598
599 document.forms["form"].elements["lon" + marker].value=format5f(lon);
600 document.forms["form"].elements["lat" + marker].value=format5f(lat);
601
602 routino.point[marker].lon=lon;
603 routino.point[marker].lat=lat;
604 routino.point[marker].used=true;
605
606 markerCheckHome(marker);
607 }
608 }
609
610
611 //
612 // Set the search field from the form when the form changes.
613 //
614
615 function formSetSearch(marker,search) // called from event handler linked to router.html (with one argument)
616 {
617 clearSearchResult(marker);
618
619 if(search === undefined)
620 {
621 routino.point[marker].search=document.forms["form"].elements["search" + marker].value;
622
623 DoSearch(marker);
624 }
625 else
626 {
627 document.forms["form"].elements["search" + marker].value=search;
628
629 routino.point[marker].search=search;
630 }
631 }
632
633
634 //
635 // Change of loop or reverse option in the form
636 //
637
638 function formSetLoopReverse(type,value) // called from router.html (with one argument)
639 {
640 if(value === undefined)
641 value=document.forms["form"].elements[type].checked;
642
643 document.forms["form"].elements[type].checked=value;
644
645 if(type == "loop")
646 routino.loop=value;
647 else
648 routino.reverse=value;
649
650 updateURLs();
651 }
652
653
654 //
655 // Format a number in printf("%.5f") format.
656 //
657
658 function format5f(number)
659 {
660 var newnumber=Math.floor(number*100000+0.5);
661 var delta=0;
662
663 if(newnumber>=0 && newnumber<100000) delta= 100000;
664 if(newnumber<0 && newnumber>-100000) delta=-100000;
665
666 var string=String(newnumber+delta);
667
668 var intpart =string.substring(0,string.length-5);
669 var fracpart=string.substring(string.length-5,string.length);
670
671 if(delta>0) intpart="0";
672 if(delta<0) intpart="-0";
673
674 return(intpart + "." + fracpart);
675 }
676
677
678 //
679 // Build a set of URL arguments
680 //
681
682 function buildURLArguments(lang)
683 {
684 var url= "transport=" + routino.transport;
685
686 for(var marker=1;marker<=vismarkers;marker++)
687 if(routino.point[marker].active)
688 {
689 url=url + ";lon" + marker + "=" + format5f(routino.point[marker].lon);
690 url=url + ";lat" + marker + "=" + format5f(routino.point[marker].lat);
691 if(routino.point[marker].search !== "")
692 url=url + ";search" + marker + "=" + encodeURIComponent(routino.point[marker].search);
693 }
694
695 for(var key in routino.profile_highway)
696 if(routino.profile_highway[key][routino.transport]!=routino_default.profile_highway[key][routino.transport])
697 url=url + ";highway-" + key + "=" + routino.profile_highway[key][routino.transport];
698
699 for(var key in routino.profile_speed)
700 if(routino.profile_speed[key][routino.transport]!=routino_default.profile_speed[key][routino.transport])
701 url=url + ";speed-" + key + "=" + routino.profile_speed[key][routino.transport];
702
703 for(var key in routino.profile_property)
704 if(routino.profile_property[key][routino.transport]!=routino_default.profile_property[key][routino.transport])
705 url=url + ";property-" + key + "=" + routino.profile_property[key][routino.transport];
706
707 for(var key in routino.restrictions)
708 if(routino.profile_restrictions[key][routino.transport]!=routino_default.profile_restrictions[key][routino.transport])
709 url=url + ";" + key + "=" + routino.profile_restrictions[key][routino.transport];
710
711 if(routino.loop)
712 url=url + ";loop=true";
713
714 if(routino.reverse)
715 url=url + ";reverse=true";
716
717 if(lang && routino.language)
718 url=url + ";language=" + routino.language;
719
720 return(url);
721 }
722
723
724 //
725 // Build a set of URL arguments for the map location
726 //
727
728 function buildMapArguments()
729 {
730 var lonlat = map.getCenter();
731
732 var zoom = map.getZoom();
733
734 return "lat=" + format5f(lonlat.lat) + ";lon=" + format5f(lonlat.lng) + ";zoom=" + zoom;
735 }
736
737
738 //
739 // Update the URLs
740 //
741
742 function updateURLs()
743 {
744 var urlargs1=buildURLArguments(true);
745 var urlargs2=buildURLArguments(false);
746 var mapargs=buildMapArguments();
747
748 var links=document.getElementsByTagName("a");
749
750 for(var i=0; i<links.length; i++)
751 {
752 var element=links[i];
753
754 if(element.id == "permalink_url")
755 element.href=location.pathname + "?" + urlargs1 + ";" + mapargs;
756
757 if(element.id == "visualiser_url")
758 element.href="visualiser.html" + "?" + mapargs;
759
760 if(element.id == "edit_url")
761 element.href=mapprops.editurl + "?" + mapargs;
762
763 if(element.id.match(/^lang_([a-zA-Z-]+)_url$/))
764 element.href="router.html" + "." + RegExp.$1 + "?" + urlargs2 + ";" + mapargs;
765 }
766 }
767
768
769 ////////////////////////////////////////////////////////////////////////////////
770 ///////////////////////////////// Map handling /////////////////////////////////
771 ////////////////////////////////////////////////////////////////////////////////
772
773 var map;
774 var layerMap=[], layerVectors, layerGPX;
775 var routing_type;
776
777 //
778 // Initialise the 'map' object
779 //
780
781 function map_init() // called from router.html
782 {
783 // Create the map (Map URLs and limits are in mapprops.js)
784
785 map = L.map("map",
786 {
787 attributionControl: false,
788 zoomControl: false,
789
790 minZoom: mapprops.zoomout,
791 maxZoom: mapprops.zoomin,
792
793 maxBounds: L.latLngBounds(L.latLng(mapprops.southedge,mapprops.westedge),L.latLng(mapprops.northedge,mapprops.eastedge))
794 });
795
796 // Add map tile layers
797
798 var baselayers={};
799
800 for(var l=0; l<mapprops.mapdata.length; l++)
801 {
802 var urls=mapprops.mapdata[l].tiles.url.replace(/\${/g,"{");
803
804 if(mapprops.mapdata[l].tiles.subdomains===undefined)
805 layerMap[l] = L.tileLayer(urls);
806 else
807 layerMap[l] = L.tileLayer(urls, {subdomains: mapprops.mapdata[l].tiles.subdomains});
808
809 baselayers[mapprops.mapdata[l].label]=layerMap[l];
810
811 if(l===0)
812 map.addLayer(layerMap[l]);
813 }
814
815 // Add the controls
816
817 map.addControl(L.control.zoom());
818 map.addControl(L.control.scale());
819 map.addControl(L.control.layers(baselayers));
820
821 // Update the attribution if the layer changes
822
823 function change_attribution_event(event)
824 {
825 for(var l=0; l<mapprops.mapdata.length; l++)
826 if(layerMap[l] == event.layer)
827 change_attribution(l);
828 }
829
830 map.on("baselayerchange",change_attribution_event);
831
832 function change_attribution(l)
833 {
834 var data_url =mapprops.mapdata[l].attribution.data_url;
835 var data_text=mapprops.mapdata[l].attribution.data_text;
836 var tile_url =mapprops.mapdata[l].attribution.tile_url;
837 var tile_text=mapprops.mapdata[l].attribution.tile_text;
838
839 document.getElementById("attribution_data").innerHTML="<a href=\"" + data_url + "\" target=\"data_attribution\">" + data_text + "</a>";
840 document.getElementById("attribution_tile").innerHTML="<a href=\"" + tile_url + "\" target=\"tile_attribution\">" + tile_text + "</a>";
841 }
842
843 change_attribution(0);
844
845 // Define a GPX layer but don't add it yet
846
847 layerGPX={shortest: null, quickest: null};
848
849 // Add a markers vectors layer
850
851 layerVectors = L.layerGroup();
852 map.addLayer(layerVectors);
853
854 // A set of markers
855
856 markers={};
857 icons={};
858 markersmoved=false;
859 paramschanged=false;
860
861 for(var marker=1;marker<=mapprops.maxmarkers;marker++)
862 {
863 icons[marker]=L.icon({iconUrl: "icons/marker-" + marker + "-red.png",
864 iconSize: L.point(21,25),
865 iconAnchor: L.point(10,25)});
866
867 markers[marker]=L.marker(L.point(0,0), {clickable: true, draggable: true, icon: icons[marker]});
868
869 markers[marker].on("drag" , (function(m) { return function(evt) { dragMarkerMove (m,evt); }; }(marker)));
870 markers[marker].on("dragend", (function(m) { return function(evt) { dragMarkerComplete(m,evt); }; }(marker)));
871 }
872
873 icons.home=L.icon({iconUrl: "icons/marker-home-red.png",
874 iconSize: L.point(21,25),
875 iconAnchor: L.point(10,25)});
876
877 // Markers to highlight a selected point
878
879 for(var highlight in highlights)
880 {
881 highlights[highlight]=L.circleMarker(L.latLng(0,0), {radius: 10, stroke: true, weight: 4, color: route_dark_colours[highlight], opacity: 1.0,
882 fill: false});
883 }
884
885 // A popup for routing results
886
887 for(var popup in popups)
888 popups[popup] = createPopup(popup);
889
890 // Move the map
891
892 map.on("moveend", updateURLs);
893
894 var lon =args["lon"];
895 var lat =args["lat"];
896 var zoom=args["zoom"];
897
898 if(lon !== undefined && lat !== undefined && zoom !== undefined)
899 {
900 if(lon<mapprops.westedge) lon=mapprops.westedge;
901 if(lon>mapprops.eastedge) lon=mapprops.eastedge;
902
903 if(lat<mapprops.southedge) lat=mapprops.southedge;
904 if(lat>mapprops.northedge) lat=mapprops.northedge;
905
906 if(zoom<mapprops.zoomout) zoom=mapprops.zoomout;
907 if(zoom>mapprops.zoomin) zoom=mapprops.zoomin;
908
909 map.setView(L.latLng(lat,lon),zoom);
910 }
911 else
912 map.fitBounds(map.options.maxBounds);
913
914 // Unhide editing URL if variable set
915
916 if(mapprops.editurl !== undefined && mapprops.editurl !== "")
917 {
918 var edit_url=document.getElementById("edit_url");
919
920 edit_url.style.display="";
921 edit_url.href=mapprops.editurl;
922 }
923
924 updateURLs();
925 }
926
927
928 //
929 // Callback for a marker drag occuring on the map.
930 //
931
932 function dragMarkerMove(marker,event)
933 {
934 dragMarkerSetForm(marker);
935 }
936
937
938 //
939 // Callback for completing a drag occuring on the map.
940 //
941
942 function dragMarkerComplete(marker,event)
943 {
944 dragMarkerSetForm(marker);
945
946 updateURLs();
947 }
948
949
950 //
951 // Set the feature coordinates in the form after dragging it on the map.
952 //
953
954 function dragMarkerSetForm(marker)
955 {
956 var lonlat = markers[marker].getLatLng();
957
958 formSetCoords(marker,lonlat.lng,lonlat.lat);
959 }
960
961
962 ////////////////////////////////////////////////////////////////////////////////
963 /////////////////////////////// Marker dragging ////////////////////////////////
964 ////////////////////////////////////////////////////////////////////////////////
965
966 var dragged_waypoint=null,dragged_marker=null;
967 var dragged_waypoint_over=null,dragged_marker_over=null;
968 var dragged_icon_x,dragged_icon_y;
969
970 //
971 // Drag a waypoint up or down the list.
972 //
973
974 function dragWaypointStart(e)
975 {
976 var w=e.target;
977
978 while(w!=null && w.className != "waypoint")
979 w=w.parentElement;
980
981 if(w===null)
982 return;
983
984 w.style.opacity = "0.5";
985
986 dragged_waypoint=w;
987 dragged_marker=Number.parseInt(dragged_waypoint.id.substring(8));
988
989 dragged_icon_x=e.clientX-e.target.offsetLeft;
990 dragged_icon_y=e.clientY-e.target.offsetTop;
991 }
992
993 function dragWaypointEnd(e)
994 {
995 e.preventDefault();
996
997 if(dragged_waypoint===null)
998 return;
999
1000 dragged_waypoint.style.opacity = "";
1001
1002 dragged_waypoint=null;
1003 dragged_marker=null;
1004
1005 if(dragged_waypoint_over===null)
1006 return;
1007
1008 dragged_waypoint_over.style.border = "";
1009
1010 dragged_waypoint_over=null;
1011 dragged_marker_over=null;
1012 }
1013
1014
1015 //
1016 // Drag a waypoint over another one up or down the list.
1017 //
1018
1019 function dragWaypointEnter(e)
1020 {
1021 var w=e.target;
1022
1023 while(w!=null && w.className != "waypoint")
1024 w=w.parentElement;
1025
1026 if(w===null)
1027 return;
1028
1029 if(dragged_waypoint_over!==null)
1030 dragged_waypoint_over.style.border = "";
1031
1032 if(w==dragged_waypoint)
1033 return;
1034
1035 dragged_waypoint_over=w;
1036 dragged_marker_over=Number.parseInt(dragged_waypoint_over.id.substring(8));
1037
1038 if(dragged_marker>dragged_marker_over)
1039 w.style.borderTop = "3px solid black";
1040 else
1041 w.style.borderBottom = "3px solid black";
1042 }
1043
1044 function dragWaypointOver(e)
1045 {
1046 e.preventDefault();
1047 }
1048
1049 function dragWaypointLeave(e)
1050 {
1051 var w=e.target;
1052
1053 while(w!=null && w.className != "waypoint")
1054 w=w.parentElement;
1055
1056 if(w===null)
1057 return;
1058
1059 if(w==dragged_waypoint_over)
1060 return;
1061
1062 w.style.border = "";
1063 }
1064
1065
1066 //
1067 // Drop the waypoint after dragging up or down the list.
1068 //
1069
1070 function dragWaypointDrop(e)
1071 {
1072 e.preventDefault();
1073
1074 if(dragged_marker_over===null)
1075 return;
1076
1077 if(dragged_marker_over>dragged_marker)
1078 for(var m=dragged_marker;m<dragged_marker_over;m++)
1079 markerSwap(m,m+1);
1080
1081 if(dragged_marker_over<dragged_marker)
1082 for(var m=dragged_marker;m>dragged_marker_over;m--)
1083 markerSwap(m,m-1);
1084 }
1085
1086
1087 //
1088 // Drag a waypoint over the map.
1089 //
1090
1091 function dragWaypointMapEnter(e)
1092 {
1093 e.preventDefault();
1094
1095 if(dragged_waypoint_over!==null)
1096 dragged_waypoint_over.style.border = "";
1097 }
1098
1099 function dragWaypointMapOver(e)
1100 {
1101 e.preventDefault();
1102 }
1103
1104 function dragWaypointMapLeave(e)
1105 {
1106 e.preventDefault();
1107 }
1108
1109
1110 //
1111 // Drop the waypoint after dragging it over the map.
1112 //
1113
1114 function dragWaypointMapDrop(e)
1115 {
1116 e.preventDefault();
1117
1118 var rect = document.getElementById("map").getBoundingClientRect();
1119
1120 var lonlat=map.containerPointToLatLng(L.point(e.clientX-rect.left-window.scrollX-dragged_icon_x+8,
1121 e.clientY-rect.top -window.scrollY-dragged_icon_y+21));
1122
1123 formSetCoords(dragged_marker,lonlat.lng,lonlat.lat);
1124
1125 if(!routino.point[dragged_marker].active)
1126 markerToggleMap(dragged_marker);
1127
1128 if(routino.point[dragged_marker].search=="")
1129 markerCoords(dragged_marker);
1130 }
1131
1132
1133 ////////////////////////////////////////////////////////////////////////////////
1134 /////////////////////////////// Marker handling ////////////////////////////////
1135 ////////////////////////////////////////////////////////////////////////////////
1136
1137 //
1138 // Toggle a marker on the map.
1139 //
1140
1141 function markerToggleMap(marker) // called from router.html
1142 {
1143 if(!routino.point[marker].used)
1144 {
1145 routino.point[marker].used=true;
1146 markerCentre(marker);
1147 markerCoords(marker);
1148 }
1149
1150 markerAddRemoveMap(marker,!routino.point[marker].active);
1151 }
1152
1153
1154 //
1155 // Show or hide a marker on the map.
1156 //
1157
1158 function markerAddRemoveMap(marker,active)
1159 {
1160 if(active)
1161 markerAddMap(marker);
1162 else
1163 markerRemoveMap(marker);
1164 }
1165
1166
1167 //
1168 // Show a marker on the map.
1169 //
1170
1171 function markerAddMap(marker)
1172 {
1173 clearSearchResult(marker);
1174
1175 layerVectors.addLayer(markers[marker]);
1176 routino.point[marker].active=true;
1177 routino.point[marker].used=true;
1178
1179 updateIcon(marker);
1180
1181 markersmoved=true;
1182
1183 updateURLs();
1184
1185 updateSearchButtons();
1186 }
1187
1188
1189 //
1190 // Remove a marker from the map.
1191 //
1192
1193 function markerRemoveMap(marker)
1194 {
1195 clearSearchResult(marker);
1196
1197 layerVectors.removeLayer(markers[marker]);
1198 routino.point[marker].active=false;
1199
1200 updateIcon(marker);
1201
1202 markersmoved=true;
1203
1204 updateURLs();
1205
1206 updateSearchButtons();
1207 }
1208
1209
1210 //
1211 // Display search string for the marker
1212 //
1213
1214 function markerSearch(marker) // called from router.html
1215 {
1216 clearSearchResult(marker);
1217
1218 document.getElementById("coords" + marker).style.display="none";
1219 document.getElementById("search" + marker).style.display="";
1220 }
1221
1222
1223 //
1224 // Display coordinates for the marker
1225 //
1226
1227 function markerCoords(marker) // called from router.html
1228 {
1229 clearSearchResult(marker);
1230
1231 document.getElementById("search" + marker).style.display="none";
1232 document.getElementById("coords" + marker).style.display="";
1233 }
1234
1235
1236 //
1237 // Centre the marker on the map
1238 //
1239
1240 function markerCentre(marker) // called from router.html
1241 {
1242 if(!routino.point[marker].used)
1243 return;
1244
1245 clearSearchResult(marker);
1246
1247 var lonlat=map.getCenter();
1248
1249 formSetCoords(marker,lonlat.lng,lonlat.lat);
1250 }
1251
1252
1253 //
1254 // Centre the map on the marker
1255 //
1256
1257 function markerRecentre(marker) // called from router.html
1258 {
1259 if(!routino.point[marker].used)
1260 return;
1261
1262 clearSearchResult(marker);
1263
1264 var lon=routino.point[marker].lon;
1265 var lat=routino.point[marker].lat;
1266
1267 var lonlat = L.latLng(lat,lon);
1268
1269 map.panTo(lonlat);
1270 }
1271
1272
1273 //
1274 // Clear the current marker.
1275 //
1276
1277 function markerRemove(marker) // called from router.html
1278 {
1279 clearSearchResult(marker);
1280
1281 for(var m=marker;m<vismarkers;m++)
1282 markerCopy(m,m+1);
1283
1284 markerRemoveForm(vismarkers--);
1285
1286 if(vismarkers==1)
1287 markerAddAfter(1);
1288
1289 updateSearchButtons();
1290 }
1291
1292
1293 //
1294 // Add a marker before the current one.
1295 //
1296
1297 function markerAddBefore(marker)
1298 {
1299 if(vismarkers==mapprops.maxmarkers || marker==1)
1300 return false;
1301
1302 clearSearchResult(marker);
1303
1304 markerAddForm(++vismarkers);
1305
1306 for(var m=vismarkers;m>marker;m--)
1307 markerCopy(m,m-1);
1308
1309 markerClearForm(marker-1);
1310 }
1311
1312
1313 //
1314 // Add a marker after the current one.
1315 //
1316
1317 function markerAddAfter(marker) // called from router.html
1318 {
1319 if(vismarkers==mapprops.maxmarkers)
1320 return false;
1321
1322 clearSearchResult(marker);
1323
1324 markerAddForm(++vismarkers);
1325
1326 for(var m=vismarkers;m>(marker+1);m--)
1327 markerCopy(m,m-1);
1328
1329 markerClearForm(marker+1);
1330 }
1331
1332
1333 //
1334 // Set this marker as the home location.
1335 //
1336
1337 function markerHome(marker) // called from router.html
1338 {
1339 if(!routino.point[marker].used)
1340 {
1341 markerMoveHome(marker);
1342 }
1343 else
1344 {
1345 clearSearchResult(marker);
1346
1347 markerSetClearHome(marker,!routino.point[marker].home);
1348 }
1349 }
1350
1351
1352 //
1353 // Set this marker as the current location.
1354 //
1355
1356 function markerLocate(marker) // called from router.html
1357 {
1358 clearSearchResult(marker);
1359
1360 function success(position)
1361 {
1362 formSetCoords(marker,position.coords.longitude,position.coords.latitude);
1363 markerAddMap(marker);
1364 }
1365
1366 function failure(error)
1367 {
1368 alert("Error: " + error.message);
1369 }
1370
1371 if(navigator.geolocation)
1372 navigator.geolocation.getCurrentPosition(success,failure);
1373 else
1374 alert("Error: Geolocation unavailable");
1375 }
1376
1377
1378 //
1379 // Update the search buttons enable/disable.
1380 //
1381
1382 function updateSearchButtons()
1383 {
1384 var markersactive=0;
1385
1386 for(var m=1;m<=vismarkers;m++)
1387 if(routino.point[m].active)
1388 markersactive++;
1389
1390 if(markersactive<2)
1391 {
1392 document.getElementById("shortest1").disabled="disabled";
1393 document.getElementById("quickest1").disabled="disabled";
1394 document.getElementById("shortest2").disabled="disabled";
1395 document.getElementById("quickest2").disabled="disabled";
1396 }
1397 else
1398 {
1399 document.getElementById("shortest1").disabled="";
1400 document.getElementById("quickest1").disabled="";
1401 document.getElementById("shortest2").disabled="";
1402 document.getElementById("quickest2").disabled="";
1403 }
1404 }
1405
1406
1407 //
1408 // Update an icon to set colours and home or normal marker.
1409 //
1410
1411 function updateIcon(marker)
1412 {
1413 if(routino.point[marker].home)
1414 {
1415 if(routino.point[marker].active)
1416 document.getElementById("icon" + marker).src="icons/marker-home-red.png";
1417 else
1418 document.getElementById("icon" + marker).src="icons/marker-home-grey.png";
1419
1420 markers[marker].setIcon(icons.home);
1421 }
1422 else
1423 {
1424 if(routino.point[marker].active)
1425 document.getElementById("icon" + marker).src="icons/marker-" + marker + "-red.png";
1426 else
1427 document.getElementById("icon" + marker).src="icons/marker-" + marker + "-grey.png";
1428
1429 markers[marker].setIcon(icons[marker]);
1430 }
1431
1432 markers[marker].update();
1433 }
1434
1435
1436 //
1437 // Move the marker to the home location
1438 //
1439
1440 function markerMoveHome(marker)
1441 {
1442 if(homelon===null || homelat===null)
1443 return;
1444
1445 routino.point[marker].home=true;
1446 routino.point[marker].used=true;
1447
1448 formSetCoords(marker,homelon,homelat);
1449 markerAddMap(marker);
1450 }
1451
1452
1453 //
1454 // Set or clear the home marker icon
1455 //
1456
1457 function markerSetClearHome(marker,home)
1458 {
1459 var cookie;
1460 var date = new Date();
1461
1462 if(home)
1463 {
1464 homelat=format5f(routino.point[marker].lat);
1465 homelon=format5f(routino.point[marker].lon);
1466
1467 cookie="Routino-home=lon:" + homelon + ":lat:" + homelat;
1468
1469 date.setUTCFullYear(date.getUTCFullYear()+5);
1470
1471 routino.point[marker].home=true;
1472 }
1473 else
1474 {
1475 homelat=null;
1476 homelon=null;
1477
1478 cookie="Routino-home=unset";
1479
1480 date.setUTCFullYear(date.getUTCFullYear()-1);
1481
1482 routino.point[marker].home=false;
1483 }
1484
1485 document.cookie=cookie + ";expires=" + date.toGMTString();
1486
1487 updateIcon(marker);
1488
1489 for(var m=1;m<=mapprops.maxmarkers;m++)
1490 markerCheckHome(m);
1491 }
1492
1493
1494 //
1495 // Check if a marker is the home marker
1496 //
1497
1498 function markerCheckHome(marker)
1499 {
1500 var home=routino.point[marker].home;
1501
1502 if(routino.point[marker].lon==homelon && routino.point[marker].lat==homelat)
1503 routino.point[marker].home=true;
1504 else
1505 routino.point[marker].home=false;
1506
1507 if(home!=routino.point[marker].home)
1508 updateIcon(marker);
1509 }
1510
1511
1512 //
1513 // Move this marker up.
1514 //
1515
1516 function markerMoveUp(marker) // called from router.html
1517 {
1518 if(marker==1)
1519 {
1520 for(var m=1;m<vismarkers;m++)
1521 markerSwap(m,m+1);
1522 }
1523 else
1524 markerSwap(marker,marker-1);
1525 }
1526
1527
1528 //
1529 // Move this marker down.
1530 //
1531
1532 function markerMoveDown(marker) // called from router.html
1533 {
1534 if(marker==vismarkers)
1535 {
1536 for(var m=vismarkers;m>1;m--)
1537 markerSwap(m,m-1);
1538 }
1539 else
1540 markerSwap(marker,marker+1);
1541 }
1542
1543
1544 //
1545 // Copy a marker from one place to another.
1546 //
1547
1548 function markerCopy(marker1,marker2)
1549 {
1550 for(var element in routino.point[marker2])
1551 routino.point[marker1][element]=routino.point[marker2][element];
1552
1553 document.getElementById("search" + marker1).style.display=document.getElementById("search" + marker2).style.display;
1554
1555 document.getElementById("coords" + marker1).style.display=document.getElementById("coords" + marker2).style.display;
1556
1557 document.forms["form"].elements["search" + marker1].value=document.forms["form"].elements["search" + marker2].value;
1558
1559 formSetCoords(marker1,routino.point[marker1].lon,routino.point[marker1].lat);
1560
1561 markerAddRemoveMap(marker1,routino.point[marker1].active);
1562 }
1563
1564
1565 //
1566 // Swap a pair of markers.
1567 //
1568
1569 function markerSwap(marker1,marker2)
1570 {
1571 for(var element in routino.point[marker2])
1572 {
1573 var temp=routino.point[marker1][element];
1574 routino.point[marker1][element]=routino.point[marker2][element];
1575 routino.point[marker2][element]=temp;
1576 }
1577
1578 var search_display=document.getElementById("search" + marker1).style.display;
1579 document.getElementById("search" + marker1).style.display=document.getElementById("search" + marker2).style.display;
1580 document.getElementById("search" + marker2).style.display=search_display;
1581
1582 var coords_display=document.getElementById("coords" + marker1).style.display;
1583 document.getElementById("coords" + marker1).style.display=document.getElementById("coords" + marker2).style.display;
1584 document.getElementById("coords" + marker2).style.display=coords_display;
1585
1586 var search_value=document.forms["form"].elements["search" + marker1].value;
1587 document.forms["form"].elements["search" + marker1].value=document.forms["form"].elements["search" + marker2].value;
1588 document.forms["form"].elements["search" + marker2].value=search_value;
1589
1590 formSetCoords(marker1,routino.point[marker1].lon,routino.point[marker1].lat);
1591 formSetCoords(marker2,routino.point[marker2].lon,routino.point[marker2].lat);
1592
1593 markerAddRemoveMap(marker1,routino.point[marker1].active);
1594 markerAddRemoveMap(marker2,routino.point[marker2].active);
1595 }
1596
1597
1598 //
1599 // Reverse the markers.
1600 //
1601
1602 function markersReverse() // called from router.html
1603 {
1604 for(var marker=1;marker<=vismarkers/2;marker++)
1605 markerSwap(marker,vismarkers+1-marker);
1606
1607 markersmoved=true;
1608
1609 updateURLs();
1610 }
1611
1612
1613 //
1614 // Close the loop.
1615 //
1616
1617 function markersLoop() // called from router.html
1618 {
1619 if(vismarkers==mapprops.maxmarkers)
1620 return false;
1621
1622 if(routino.point[vismarkers].lon==routino.point[1].lon && routino.point[vismarkers].lat==routino.point[1].lat)
1623 {
1624 if(routino.point[vismarkers].active)
1625 return false;
1626 else
1627 {
1628 markerToggleMap(vismarkers);
1629 return true;
1630 }
1631 }
1632
1633 if(routino.point[vismarkers].used)
1634 markerAddForm(++vismarkers);
1635
1636 markerCopy(vismarkers,1);
1637
1638 markersmoved=true;
1639
1640 updateURLs();
1641
1642 updateSearchButtons();
1643 }
1644
1645
1646 //
1647 // Display the form for a marker
1648 //
1649
1650 function markerAddForm(marker)
1651 {
1652 document.getElementById("waypoint" + marker).style.display="";
1653 }
1654
1655
1656 //
1657 // Hide the form for a marker
1658 //
1659
1660 function markerRemoveForm(marker)
1661 {
1662 document.getElementById("waypoint" + marker).style.display="none";
1663
1664 markerClearForm(marker);
1665 }
1666
1667
1668 //
1669 // Clear the form for a marker
1670 //
1671
1672 function markerClearForm(marker)
1673 {
1674 markerRemoveMap(marker);
1675 markerSearch(marker);
1676
1677 formSetCoords(marker,"","");
1678 formSetSearch(marker,"");
1679
1680 updateIcon(marker);
1681
1682 routino.point[marker].used=false;
1683 routino.point[marker].home=false;
1684 routino.point[marker].active=false;
1685 }
1686
1687
1688 ////////////////////////////////////////////////////////////////////////////////
1689 //////////////////////////// Route results handling ////////////////////////////
1690 ////////////////////////////////////////////////////////////////////////////////
1691
1692 var route_light_colours={shortest: "#60C060", quickest: "#6060C0"};
1693 var route_dark_colours ={shortest: "#408040", quickest: "#404080"};
1694
1695 var highlights={shortest: null, quickest: null};
1696 var popups={shortest: null, quickest: null};
1697 var routepoints={shortest: {}, quickest: {}};
1698
1699 //
1700 // Highlight a specific item in the route
1701 //
1702
1703 function highlight(type,line,action)
1704 {
1705 if(action == "clear")
1706 {
1707 layerVectors.removeLayer(highlights[type]);
1708
1709 drawPopup(type,null);
1710 }
1711 else if(action == "zoom")
1712 {
1713 var lonlat = L.latLng(routepoints[type][line].lat,routepoints[type][line].lon);
1714
1715 map.setView(lonlat,mapprops.zoomin-2);
1716 }
1717 else
1718 {
1719 // Marker
1720
1721 var lonlat = L.latLng(routepoints[type][line].lat,routepoints[type][line].lon);
1722
1723 highlights[type].setLatLng(lonlat);
1724
1725 layerVectors.addLayer(highlights[type]);
1726
1727 // Popup
1728
1729 drawPopup(type,"<table>" + routepoints[type][line].html + "</table>");
1730 }
1731
1732 highlights[type].redraw();
1733 }
1734
1735
1736 //
1737 // Create a popup - independent of map because want it fixed on screen not fixed on map.
1738 //
1739
1740 function createPopup(type)
1741 {
1742 var popup=document.createElement("div");
1743
1744 popup.className = "popup";
1745
1746 popup.innerHTML = "<span></span>";
1747
1748 popup.style.display = "none";
1749
1750 popup.style.position = "fixed";
1751 popup.style.top = "-4000px";
1752 popup.style.left = "-4000px";
1753 popup.style.zIndex = "100";
1754
1755 popup.style.padding = "5px";
1756
1757 popup.style.opacity=0.85;
1758 popup.style.backgroundColor=route_light_colours[type];
1759 popup.style.border="4px solid " + route_dark_colours[type];
1760
1761 document.body.appendChild(popup);
1762
1763 return(popup);
1764 }
1765
1766
1767 //
1768 // Draw a popup - independent of map because want it fixed on screen not fixed on map.
1769 //
1770
1771 function drawPopup(type,html)
1772 {
1773 var popup=popups[type];
1774
1775 if(html===null)
1776 {
1777 popup.style.display="none";
1778 return;
1779 }
1780
1781 if(popup.style.display=="none")
1782 {
1783 var map_div=document.getElementById("map");
1784
1785 popup.style.left =map_div.offsetParent.offsetLeft+map_div.offsetLeft+60 + "px";
1786 popup.style.top = map_div.offsetTop +30 + "px";
1787 popup.style.width =map_div.clientWidth-120 + "px";
1788
1789 popup.style.display="";
1790 }
1791
1792 var close="<span style='float: right; cursor: pointer;' onclick='highlight(\""+type+"\",-1,\"clear\")'>X</span>";
1793
1794 popup.innerHTML=close+html;
1795 }
1796
1797
1798 //
1799 // Remove a GPX trace
1800 //
1801
1802 function removeGPXTrace(type)
1803 {
1804 map.removeLayer(layerGPX[type]);
1805 layerGPX[type]=null;
1806
1807 displayStatus(type,"no_info");
1808
1809 document.getElementById(type + "_links").style.display = "none";
1810
1811 document.getElementById(type + "_route").innerHTML = "";
1812
1813 hideshow_hide(type);
1814 }
1815
1816
1817 ////////////////////////////////////////////////////////////////////////////////
1818 /////////////////////////////// Server handling ////////////////////////////////
1819 ////////////////////////////////////////////////////////////////////////////////
1820
1821 //
1822 // Define an AJAX request object
1823 //
1824
1825 function ajaxGET(url,success,failure,state)
1826 {
1827 var ajaxRequest=new XMLHttpRequest();
1828
1829 function ajaxGOT(options) {
1830 if(this.readyState==4)
1831 if(this.status==200)
1832 { if(typeof(options.success)=="function") options.success(this,options.state); }
1833 else
1834 { if(typeof(options.failure)=="function") options.failure(this,options.state); }
1835 }
1836
1837 ajaxRequest.onreadystatechange = function(){ ajaxGOT.call(ajaxRequest,{success: success, failure: failure, state: state}); };
1838 ajaxRequest.open("GET", url, true);
1839 ajaxRequest.send(null);
1840 }
1841
1842
1843 //
1844 // Display data statistics
1845 //
1846
1847 function displayStatistics() // called from router.html
1848 {
1849 // Use AJAX to get the statistics
1850
1851 ajaxGET("statistics.cgi", runStatisticsSuccess);
1852 }
1853
1854
1855 //
1856 // Success in running data statistics generation.
1857 //
1858
1859 function runStatisticsSuccess(response)
1860 {
1861 document.getElementById("statistics_data").innerHTML="<pre>" + response.responseText + "</pre>";
1862 document.getElementById("statistics_link").style.display="none";
1863 }
1864
1865
1866 //
1867 // Submit form - perform the routing
1868 //
1869
1870 function findRoute(type) // called from router.html
1871 {
1872 tab_select("results");
1873
1874 hideshow_hide("help_options");
1875 hideshow_hide("shortest");
1876 hideshow_hide("quickest");
1877
1878 displayStatus("result","running");
1879
1880 var url="router.cgi" + "?" + buildURLArguments(true) + ";type=" + type;
1881
1882 // Destroy the existing layer(s)
1883
1884 highlight("shortest",-1,"clear");
1885 highlight("quickest",-1,"clear");
1886
1887 if(markersmoved || paramschanged)
1888 {
1889 if(layerGPX.shortest!==null)
1890 removeGPXTrace("shortest");
1891 if(layerGPX.quickest!==null)
1892 removeGPXTrace("quickest");
1893 markersmoved=false;
1894 paramschanged=false;
1895 }
1896 else if(layerGPX[type]!==null)
1897 removeGPXTrace(type);
1898
1899 // Use AJAX to run the router
1900
1901 routing_type=type;
1902
1903 ajaxGET(url, runRouterSuccess, runRouterFailure);
1904 }
1905
1906
1907 //
1908 // Success in running router.
1909 //
1910
1911 function runRouterSuccess(response)
1912 {
1913 var lines=response.responseText.split("\n");
1914
1915 var uuid=lines[0];
1916 var success=lines[1];
1917
1918 var link;
1919
1920 // Update the status message
1921
1922 if(success=="ERROR")
1923 {
1924 displayStatus("result","error");
1925 hideshow_show("help_route");
1926
1927 link=document.getElementById("router_log_error");
1928 link.href="results.cgi?uuid=" + uuid + ";type=router;format=log";
1929
1930 return;
1931 }
1932 else
1933 {
1934 displayStatus("result","complete");
1935 hideshow_hide("help_route");
1936
1937 link=document.getElementById("router_log_complete");
1938 link.href="results.cgi?uuid=" + uuid + ";type=router;format=log";
1939 }
1940
1941 // Update the routing result message
1942
1943 link=document.getElementById(routing_type + "_html");
1944 link.href="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=html";
1945
1946 link=document.getElementById(routing_type + "_gpx_track");
1947 link.href="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=gpx-track";
1948
1949 link=document.getElementById(routing_type + "_gpx_route");
1950 link.href="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=gpx-route";
1951
1952 link=document.getElementById(routing_type + "_text_all");
1953 link.href="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=text-all";
1954
1955 link=document.getElementById(routing_type + "_text");
1956 link.href="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=text";
1957
1958 document.getElementById(routing_type + "_links").style.display = "";
1959
1960 // Add a GPX layer
1961
1962 var url="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=gpx-track";
1963
1964 ajaxGET(url, runGPXSuccess);
1965
1966 hideshow_show(routing_type);
1967
1968 displayResult(routing_type,uuid);
1969 }
1970
1971
1972 //
1973 // Success in getting GPX.
1974 //
1975
1976 function runGPXSuccess(response)
1977 {
1978 var lines=response.responseText.split("\n");
1979
1980 var coords=[];
1981 var segment=-1;
1982
1983 for(var line=0;line<lines.length;line++)
1984 {
1985 if(lines[line].match(/^<trkseg>/))
1986 {
1987 segment++;
1988 coords[segment]=[];
1989 }
1990 if(lines[line].match(/^<trkpt lat="([-0-9.]+)" lon="([-0-9.]+)"/))
1991 {
1992 var lat=RegExp.$1;
1993 var lon=RegExp.$2;
1994
1995 coords[segment].push(L.latLng(lat,lon));
1996 }
1997 }
1998
1999 var colour;
2000
2001 if(routing_type == "shortest")
2002 colour="#00FF00";
2003 else
2004 colour="#0000FF";
2005
2006 if(typeof(L.multiPolyline)=="function")
2007 layerGPX[routing_type] = L.multiPolyline(coords,{weight: 3, stroke: true, color: colour, opacity: 1.0,
2008 fill: false});
2009 else
2010 layerGPX[routing_type] = L.polyline(coords,{weight: 3, stroke: true, color: colour, opacity: 1.0,
2011 fill: false});
2012
2013 map.addLayer(layerGPX[routing_type]);
2014 }
2015
2016
2017 //
2018 // Failure in running router.
2019 //
2020
2021 function runRouterFailure(response)
2022 {
2023 displayStatus("result","failed");
2024 }
2025
2026
2027 //
2028 // Display the status
2029 //
2030
2031 function displayStatus(type,subtype,content)
2032 {
2033 var child=document.getElementById(type + "_status").firstChild;
2034
2035 do
2036 {
2037 if(child.id !== undefined)
2038 child.style.display="none";
2039
2040 child=child.nextSibling;
2041 }
2042 while(child !== null);
2043
2044 var chosen_status=document.getElementById(type + "_status_" + subtype);
2045
2046 chosen_status.style.display="";
2047
2048 if(content !== undefined)
2049 chosen_status.innerHTML=content;
2050 }
2051
2052
2053 //
2054 // Display the route
2055 //
2056
2057 function displayResult(type,uuid)
2058 {
2059 routing_type = type;
2060
2061 // Add the route
2062
2063 var url="results.cgi?uuid=" + uuid + ";type=" + routing_type + ";format=html";
2064
2065 // Use AJAX to get the route
2066
2067 ajaxGET(url, getRouteSuccess, getRouteFailure);
2068 }
2069
2070
2071 //
2072 // Success in getting route.
2073 //
2074
2075 function getRouteSuccess(response)
2076 {
2077 var lines=response.responseText.split("\n");
2078
2079 routepoints[routing_type]=[];
2080
2081 var points=routepoints[routing_type];
2082
2083 var table=0;
2084 var point=0;
2085 var total_table,total_word;
2086
2087 for(var line=0;line<lines.length;line++)
2088 {
2089 var thisline=lines[line];
2090
2091 if(table===0)
2092 {
2093 if(thisline.match("<table>"))
2094 table=1;
2095 else
2096 continue;
2097 }
2098
2099 if(thisline.match("</table>"))
2100 break;
2101
2102 if(thisline.match("<tr class='([a-z])'>"))
2103 {
2104 var rowtype=RegExp.$1;
2105
2106 if(rowtype=="c")
2107 {
2108 thisline.match(": *([-0-9.]+) *([-0-9.]+)");
2109 points[point]={lat: Number(RegExp.$1), lon: Number(RegExp.$2), html: "", highway: "", distance: "", total: ""};
2110
2111 point++;
2112 }
2113 else if(rowtype=="n")
2114 {
2115 points[point-1].html += thisline;
2116 }
2117 else if(rowtype=="s")
2118 {
2119 thisline.match("<span class='h'>([^<]+)</span>");
2120 points[point-1].highway = RegExp.$1;
2121
2122 thisline.match("<span class='d'>([^<]+)</span>");
2123 points[point-1].distance = RegExp.$1;
2124
2125 thisline.match("(<span class='j'>[^<]+</span>)");
2126 points[point-1].total = RegExp.$1;
2127
2128 thisline.match("^(.*).<span class='j'>");
2129
2130 points[point-1].html += RegExp.$1;
2131 }
2132 else if(rowtype=="t")
2133 {
2134 points[point-1].html += thisline;
2135
2136 thisline.match("<span class='j'>([^<]+)</span>");
2137 points[point-1].total = RegExp.$1;
2138
2139 thisline.match("<td>(.*)");
2140 points[point-1].highway = RegExp.$1;
2141 }
2142 }
2143 }
2144
2145 displayStatus(routing_type,"info",points[point-1].total.bold());
2146
2147 var result="<table onmouseout='highlight(\"" + routing_type + "\",-1,\"clear\")'>";
2148
2149 for(var p=0;p<point;p++)
2150 {
2151 if(p!=point-1)
2152 points[p].html += "<tr><td>" + points[p].total;
2153
2154 result=result + "<tr onmouseover='highlight(\"" + routing_type + "\"," + p + ",\"show\")' " +
2155 "onclick='highlight(\"" + routing_type + "\"," + p + ",\"zoom\")'>" +
2156 "<td class='distance' title='" + points[p].distance + "'>#" + (p+1) +
2157 "<td class='highway'>" + points[p].highway;
2158 }
2159
2160 result=result + "</table>";
2161
2162 document.getElementById(routing_type + "_route").innerHTML=result;
2163 }
2164
2165
2166 //
2167 // Failure in getting route.
2168 //
2169
2170 function getRouteFailure(response)
2171 {
2172 document.getElementById(routing_type + "_route").innerHTML = "";
2173 }
2174
2175
2176 //
2177 // Perform a search
2178 //
2179
2180 function DoSearch(marker)
2181 {
2182 // Use AJAX to get the search result
2183
2184 var search=routino.point[marker].search;
2185
2186 var mapbounds=map.getBounds();
2187
2188 var url="search.cgi";
2189
2190 url=url + "?marker=" + marker;
2191 url=url + ";lonmin=" + format5f(mapbounds.getWest());
2192 url=url + ";latmin=" + format5f(mapbounds.getSouth());
2193 url=url + ";lonmax=" + format5f(mapbounds.getEast());
2194 url=url + ";latmax=" + format5f(mapbounds.getNorth());
2195 url=url + ";search=" + encodeURIComponent(search);
2196
2197 ajaxGET(url,runSearchSuccess);
2198 }
2199
2200
2201 var searchresults=[];
2202
2203 //
2204 // Success in running search.
2205 //
2206
2207 function runSearchSuccess(response)
2208 {
2209 var lines=response.responseText.split("\n");
2210
2211 var marker=lines[0];
2212 var cpuinfo=lines[1]; // not used
2213 var message=lines[2];
2214
2215 if(message !== "")
2216 {
2217 alert(message);
2218 return;
2219 }
2220
2221 searchresults[marker]=[];
2222
2223 for(var line=3;line<lines.length;line++)
2224 {
2225 var thisline=lines[line];
2226
2227 if(thisline==="")
2228 break;
2229
2230 thisline.match("([-.0-9]+) ([-.0-9]+) (.*)");
2231
2232 searchresults[marker][searchresults[marker].length]={lat: Number(RegExp.$1), lon: Number(RegExp.$2), name: RegExp.$3};
2233 }
2234
2235 if(searchresults[marker].length==1)
2236 {
2237 formSetSearch(marker,searchresults[marker][0].name);
2238 formSetCoords(marker,searchresults[marker][0].lon,searchresults[marker][0].lat);
2239 markerAddMap(marker);
2240 }
2241 else
2242 {
2243 var results=document.getElementById("searchresults" + marker);
2244
2245 var innerHTML="<td colspan=\"3\">";
2246
2247 for(var n=0;n<searchresults[marker].length;n++)
2248 {
2249 if(n>0)
2250 innerHTML+="<br>";
2251
2252 innerHTML+="<a href=\"#\" onclick=\"choseSearchResult(" + marker + "," + n + ")\">" +
2253 searchresults[marker][n].name +
2254 "</a>";
2255 }
2256
2257 results.innerHTML=innerHTML;
2258
2259 results.style.display="";
2260 }
2261 }
2262
2263
2264 //
2265 // Display search results.
2266 //
2267
2268 function choseSearchResult(marker,n)
2269 {
2270 if(n>=0)
2271 {
2272 formSetSearch(marker,searchresults[marker][n].name);
2273 formSetCoords(marker,searchresults[marker][n].lon,searchresults[marker][n].lat);
2274 markerAddMap(marker);
2275 }
2276 }
2277
2278
2279 //
2280 // Clear search results.
2281 //
2282
2283 function clearSearchResult(marker)
2284 {
2285 document.getElementById("searchresults" + marker).style.display="none";
2286 }