Hi all,
Ik heb onderstaande JS-code toegepast ten behoeve van een store locator.
De store locator geeft een overzicht van shops op basis van reisafstand vanaf locatie.
Zie hier de link naar de demo-page: http://optimax.apphb.com/demo.html
Mijn vraag heeft betrekking op de foutieve weergave van de afstand.
Als je 'London' invoert in het zoekveld dan wordt bij de eerste hit '11 miles' weergegeven, waar dit 1.1 mile zou moeten zijn.
Bij lange afstanden treedt deze fout ook op. Bij een afstand boven de 1000 miles wordt deze door Google-maps weergeven als bijvoorbeeld 1,161 miles terwijl JS 1.161 mile genereert, alsof de afstand dus minder dan 2 mile zou zijn.
Mogelijk verduidelijkt onderstaande mijn vraag:
Voorbeeld A
Daadwerkelijke afstand: 1,1 mile
Notificatie Google: 1.1 mile
JS genereert foutief: 11 miles
Voorbeeld B
Daadwerkelijke afstand: 1161 miles
Notificatie Google: 1,161 miles
JS genereert foutief: 1.161
Gevolg: deze shop wordt als eerste weergeven (alsof de afstand minder dan 2 mile is) in de resultaten, terwijl deze pas achteraan genoemd zou moeten worden.
Wie zou mij kunnen helpen met de aanpassing in waarschijnlijk JS?
Thanks!
Hieronder de JS-code:
Ik heb onderstaande JS-code toegepast ten behoeve van een store locator.
De store locator geeft een overzicht van shops op basis van reisafstand vanaf locatie.
Zie hier de link naar de demo-page: http://optimax.apphb.com/demo.html
Mijn vraag heeft betrekking op de foutieve weergave van de afstand.
Als je 'London' invoert in het zoekveld dan wordt bij de eerste hit '11 miles' weergegeven, waar dit 1.1 mile zou moeten zijn.
Bij lange afstanden treedt deze fout ook op. Bij een afstand boven de 1000 miles wordt deze door Google-maps weergeven als bijvoorbeeld 1,161 miles terwijl JS 1.161 mile genereert, alsof de afstand dus minder dan 2 mile zou zijn.
Mogelijk verduidelijkt onderstaande mijn vraag:
Voorbeeld A
Daadwerkelijke afstand: 1,1 mile
Notificatie Google: 1.1 mile
JS genereert foutief: 11 miles
Voorbeeld B
Daadwerkelijke afstand: 1161 miles
Notificatie Google: 1,161 miles
JS genereert foutief: 1.161
Gevolg: deze shop wordt als eerste weergeven (alsof de afstand minder dan 2 mile is) in de resultaten, terwijl deze pas achteraan genoemd zou moeten worden.
Wie zou mij kunnen helpen met de aanpassing in waarschijnlijk JS?
Thanks!
Hieronder de JS-code:
code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
| (function ($) { $.fn.storeLocator = function (options) { var settings = $.extend({ 'mapDiv': 'map', 'listDiv': 'list', 'formID': 'user-location', 'pinColor': 'fe7569', 'startPinColor': '66bd4a', 'pinTextColor': '000000', 'storeLimit': 10, 'distanceAlert': 60, 'xmlLocation': 'data/stores.xml', 'addressErrorMsg': 'Please enter valid UK address address or postcode', 'googleDistanceMatrixDestinationLimit': 25, 'defaultLat': 52.3038165, 'defaultLng': -1.081117, 'defaultLocationName': 'Northampton, United Kingdom' }, options); return this.each(function () { var $this = $(this); // global array of shop objects var _locationset = new Array(); var geocoder; // Calculate distances from passed in origin to all locations in the [_locationset] array // using Google Maps Distance Matrix Service https://developers.google.com/maps/documentation/javascript/reference#DistanceMatrixService var GeoCodeCalc = {}; GeoCodeCalc.CalcDistanceGoogle = function (origin, callback) { var destCoordArr = new Array(); var subFunctionTokens = []; $.each(_locationset, function (ix, loc) { destCoordArr.push(loc.LatLng); }); for (var i = 0; i < destCoordArr.length; i = i + settings.googleDistanceMatrixDestinationLimit) { // Google Distance Matrix allows up to 25 destinations to be passed in var tempArr = destCoordArr.slice(i, Math.min(i + settings.googleDistanceMatrixDestinationLimit)); subFunctionTokens.push(this.CallGoogleDistanceMatrix(i, origin, tempArr)); } $.when.apply($, subFunctionTokens) .then(function () { callback(true); }); }; GeoCodeCalc.CallGoogleDistanceMatrix = function (startIndex, origin, destinations) { var token = $.Deferred(); var service = new google.maps.DistanceMatrixService(); service.getDistanceMatrix( { origins: [origin], destinations: destinations, travelMode: google.maps.TravelMode.DRIVING, unitSystem: google.maps.UnitSystem.METRIC }, function (response, status) { if (response && response.rows.length) { var results = response.rows[0].elements; $.each(results, function (j, val) { if (results[j].status != "ZERO_RESULTS") { _locationset[startIndex + j].Distance = GoogleMapDistanceTextToNumber(results[j].distance.text); } }); token.resolve(); } }); return token.promise(); }; // Converts "123.45 mi" into 123.45 function GoogleMapDistanceTextToNumber(str) { return Number(str.replace(/[^0-9.]/g, "")); } // removes Google Maps URL unfriendly chars from a string function formatGoogleMapUrlString(str) { return str.replace("&", "%26").replace(" ", "+"); } //Geocode function for the origin location geocoder = new google.maps.Geocoder(); function GoogleGeocode() { this.geocode = function (address, callbackFunction) { geocoder.geocode({ 'address': address }, function (results, status) { if (status == google.maps.GeocoderStatus.OK) { var result = {}; result.latitude = results[0].geometry.location.lat(); result.longitude = results[0].geometry.location.lng(); result.formatted_address = results[0].formatted_address; result.address_components = results[0].address_components; callbackFunction(result); } else { handleError("Geocode was not successful for the following reason: " + status); callbackFunction(null); } }); }; this.geocodeLatLng = function (LatLng, callbackFunction) { geocoder.geocode({ 'location': LatLng }, function (results, status) { if (status == google.maps.GeocoderStatus.OK && results.length) { callbackFunction(results[0]); } else { handleError("Geocode was not successful for the following reason: " + status); callbackFunction(null); } }); }; } //Process form input $(function () { $(document).on('submit', '#' + settings.formID, function (e) { $("#lblError").html(""); //Stop the form submission e.preventDefault(); //Get the user input and use it var userinput = $('form').serialize(); userinput = userinput.replace("address=", ""); if (userinput == "") { handleError(settings.addressErrorMsg); } var g = new GoogleGeocode(); var address = userinput; g.geocode(address, function (data) { if (data != null) { showAddress(data); mapping(data.latitude, data.longitude); } else { //Unable to geocode handleError(settings.addressErrorMsg); } }); //Replace spaces in user input userinput = formatGoogleMapUrlString(userinput); }); }); $(document).ready(function () { // Try HTML5 geolocation if (navigator.geolocation) { navigator.geolocation.getCurrentPosition(function (position) { //map.setCenter(pos); var g = new GoogleGeocode(); var latlng = new google.maps.LatLng(position.coords.latitude, position.coords.longitude); g.geocodeLatLng(latlng, function (address) { if (address) { showAddress(address); } else { //Unable to geocode handleNoGeolocation('Error: Unable to geocode address'); } }); // do the mapping stuff mapping(position.coords.latitude, position.coords.longitude); }, function () { handleNoGeolocation("Tracking of location was not allowed."); }); } else { // Browser doesn't support Geolocation handleNoGeolocation(false); } }); function showAddress(address) { $("#lblAddress").html(address.formatted_address); // find a postcode and show it in the address textbox $.each(address.address_components, function (i, val) { if (val.types[0] == "postal_code") { $("#address").val(val.short_name); return false; // breaks the each() loop } }); } function handleNoGeolocation(error) { if (error) { var content = error; } else { var content = 'Error: Your browser doesn\'t support geolocation.'; } handleError(content + " Using default location."); mapping(settings.defaultLat, settings.defaultLng); $("#lblAddress").html(settings.defaultLocationName); } function handleError(error) { $("#lblError").html(error); } //Now all the mapping stuff function mapping(orig_lat, orig_lng) { $(function () { //Parse xml with jQuery $.ajax({ type: "GET", url: settings.xmlLocation, dataType: "xml", success: function (xml) { _locationset = new Array(); $(xml).find('Placemark').each(function (i) { var shop = { Name: $(this).find('name').text(), //Take the lat lng from the user, geocoded above LatLng: new google.maps.LatLng( $(this).find('coordinates').text().split(",")[1], $(this).find('coordinates').text().split(",")[0]), Description: $(this).find('description').text(), Marker: null, Distance: null }; _locationset.push(shop); }); // Calc Distances from user's location GeoCodeCalc.CalcDistanceGoogle(new google.maps.LatLng(orig_lat, orig_lng), function (success) { if (!success) { //something went wrong handleError("Unable to calculate distances at this time"); } else { //Sort the multi-dimensional array numerically _locationset.sort(function (a, b) { return ((a.Distance < b.Distance) ? -1 : ((a.Distance > b.Distance) ? 1 : 0)); }); // take "N" closest shops _locationset = _locationset.slice(0, settings.storeLimit); //Check the closest marker if (_locationset[0].Distance > settings.distanceAlert) { handleError("Unfortunately, our closest location is more than " + settings.distanceAlert + " miles away."); } //Create the map with jQuery $(function () { var orig_LatLng = new google.maps.LatLng(orig_lat, orig_lng); //Google maps settings var myOptions = { center: orig_LatLng, mapTypeId: google.maps.MapTypeId.ROADMAP }; var map = new google.maps.Map(document.getElementById(settings.mapDiv), myOptions); //Create one infowindow to fill later var infowindow = new google.maps.InfoWindow(); //Add user location marker var marker = createMarker(orig_LatLng, "0", settings.startPinColor); marker.setAnimation(google.maps.Animation.DROP); var bounds = new google.maps.LatLngBounds(); bounds.extend(orig_LatLng); $("#" + settings.listDiv).empty(); $(_locationset).each(function (i, location) { bounds.extend(location.LatLng); letter = String.fromCharCode("A".charCodeAt(0) + i); location.Marker = createMarker(location.LatLng, letter, settings.pinColor); create_infowindow(location); listClick(letter, location); }); // zoom in/out to show all markers map.fitBounds(bounds); function listClick(letter, shop) { $('<li />').html("<div class=\"list-details\"><div class=\"list-content\">" + "<div class=\"list-label\">" + letter + "<\/div>" + "<div class=\"loc-name\">" + shop.Name + "<\/div> <div class=\"loc-addr\">" + shop.Description + "<\/div>" + (shop.Distance ? "<div class=\"loc-addr2\"><i>approx. " + shop.Distance + " miles</i><\/div>" : "") + "<div class=\"loc-web\"><a href=\"http://maps.google.co.uk/maps?saddr=" + formatGoogleMapUrlString($("#address").val()) + "+%40" + orig_lat + "," + orig_lng + "&daddr=" + formatGoogleMapUrlString(shop.Name) + "+%40" + shop.LatLng.lat() + "," + shop.LatLng.lng() + "&hl=en" + "\" target=\"_blank\">>Get directions</a><\/div><\/div><\/div>") .click(function () { create_infowindow(shop, "left"); }).appendTo("#" + settings.listDiv); }; //Custom marker function - aplhabetical function createMarker(point, letter, pinColor) { //Set up pin icon with the Google Charts API for all of our markers var pinImage = new google.maps.MarkerImage("http://chart.apis.google.com/chart?chst=d_map_pin_letter&chld=" + letter + "|" + pinColor + "|" + settings.pinTextColor, new google.maps.Size(21, 34), new google.maps.Point(0, 0), new google.maps.Point(10, 34)); var pinShadow = new google.maps.MarkerImage("http://chart.apis.google.com/chart?chst=d_map_pin_shadow", new google.maps.Size(40, 37), new google.maps.Point(0, 0), new google.maps.Point(12, 35)); //Create the markers return new google.maps.Marker({ position: point, map: map, icon: pinImage, shadow: pinShadow, draggable: false }); }; //Infowindows function create_infowindow(shop, listLocation) { var formattedAddress = "<div class=\"infoWindow\"><b>" + shop.Name + "<\/b>" + "<div>" + shop.Description + "<\/div>" + (shop.Distance ? "<div><i>" + shop.Distance + " miles<\/i><\/div><\/div>" : "<\/div>"); //Opens the infowindow when list item is clicked if (listLocation == "left") { infowindow.setContent(formattedAddress); infowindow.open(shop.Marker.get(settings.mapDiv), shop.Marker); } //Opens the infowindow when the marker is clicked else { google.maps.event.addListener(shop.Marker, 'click', function () { infowindow.setContent(formattedAddress); infowindow.open(shop.Marker.get(settings.mapDiv), shop.Marker); }) } }; }); } }); } }); }); } }); }; })(jQuery); |