API Карт Google - проблема с массивами и InfoWindows
Извините, если это тривиально, но я новичок в JS и был в этой проблеме в течение нескольких часов безрезультатно.
function initialize() {
geocoder = new google.maps.Geocoder();
var latlng = new google.maps.LatLng(-34.397, 150.644);
var myOptions = {
zoom: 12,
center: latlng,
mapTypeId: google.maps.MapTypeId.ROADMAP
}
map = new google.maps.Map(document.getElementById("map_canvas"), myOptions);
}
var markersArray = [];
var infowindowArray = [];
function addMarker(location) {
marker = new google.maps.Marker({
position: location,
map: map
});
markersArray.push(marker);
}
function addInfowindow(string) {
infowindow = new google.maps.InfoWindow({content: string});
infowindowArray.push(infowindow);
}
function findvenues(latitudelongitude) {
$.get("venuefinder.php", { latlong:latitudelongitude.Ka+","+latitudelongitude.La }, function(data){
var myObject = eval('(' + data + ')');
for(x in myObject.response.venues){
var latlonglocation = new google.maps.LatLng(myObject.response.venues[x].location.lat,myObject.response.venues[x].location.lng);
addMarker(latlonglocation);
addInfowindow(myObject.response.venues[x].name);
google.maps.event.addListener(markersArray[x], 'click', function() {infowindowArray[x].open(map,markersArray[x]);});
console.log(markersArray[x],infowindowArray[x]);
}
});
При вызове findvenues необходимо добавить события с нажатием на инфунд. Когда я нажимаю значки, всегда открывается тот же самый инфоиндумент последнего элемента в массиве infowindows.
Однако, если я вручную введю следующий код, он работает:
google.maps.event.addListener(markersArray[0], 'click', function() {infowindowArray[0].open(map,markersArray[0]);});
google.maps.event.addListener(markersArray[1], 'click', function() {infowindowArray[1].open(map,markersArray[1]);});
Мне нужно, чтобы это было динамично, но... Что я делаю неправильно? Дайте мне знать, если вопрос неясен, и я постараюсь прояснить его.
Ответы
Ответ 1
Измените этот вызов на "addListener" в цикле:
google.maps.event.addListener(markersArray[x], 'click', function() {infowindowArray[x].open(map,markersArray[x]);});
:
google.maps.event.addListener(markersArray[x], 'click', (function(x) {
return function() {
infowindowArray[x].open(map,markersArray[x]);
}
})(x));
Введя еще одну такую функцию, вы создаете совершенно новую область. Это "замораживает" значение "x" , которое вы передаете, чтобы возвращаемая функция (фактическая функция обработчика, которая будет передана API Google, другими словами) имеет доступ к ее собственному "x" , отдельно от все остальные обработчики, установленные в этом цикле.
Если вы этого не сделаете, все обработчики разделяют то же самое маленькое "x" , и это явно не будет.
edit — есть и другие способы сделать это. Вы можете написать эту дополнительную функцию как совершенно отдельную вещь:
function makeMapListener(window, map, markers) {
return function() { window.open(map, markers); };
}
Тогда оператор в вашем цикле будет выглядеть так:
google.maps.event.addListener(markersArray[x], 'click', makeMapListener(inforwindowArray[x], map, markersArray[x]));
Важной частью этого является создание копии переменных (переменных), которые изменяются во время цикла перед созданием функции обработчика.