如何从地图外与叶标记层交互?(How to interact with leaflet marker

2019-07-01 15:40发布

我有一个展示的公共艺术作品,从点呈现的单张地图GeoJSON的 。 旁边的地图,我创建了来自同一件列表GeoJSON的数据,并希望能够单击一个项目从列表中的地图之外,并有相关标志的弹出拿出地图上。

我怎样才能通过点击事件链接项目各自的标记列表?

我map.js文件看起来像这样:

var map;
var pointsLayer;

$(document).ready(function () {
    map = new L.Map('mapContainer');
    var url = 'http://{s}.tiles.mapbox.com/v3/mapbox.mapbox-streets/{z}/{x}/{y}.png';
    var copyright = 'Map data © 2011 OpenStreetMap contributors, Imagery © 2011 CloudMade';
    var tileLayer = new L.TileLayer(url, {
        attribution: copyright
    });
    var startPosition = new L.LatLng(41.883333, - 87.633333);
    map.on('load', function (e) {
        requestUpdatedPoints(e.target.getBounds())
    });
    map.setView(startPosition, 13).addLayer(tileLayer);
    map.on('moveend', function (e) {
        requestUpdatedPoints(e.target.getBounds())
    })
});

function requestUpdatedPoints(bounds) {
    $.ajax({
        type: 'GET',
        url: '/SeeAll',
        dataType: 'json',
        data: JSON.stringify(bounds),
        contentType: 'application/json; charset=utf-8',
        success: function (result) {
            parseNewPoints(result);
            addToList(result)
        },
        error: function (req, status, error) {
            alert('what happen? did you lose conn. to server ?')
        }
    })
}

function addToList(data) {
    for (var i = 0; i < data.features.length; i++) {
        var art = data.features[i];
        $('div#infoContainer').append('<a href="#" class="list-link" title="' + art.properties.descfin + '"><div class="info-list-item">' + '<div class="info-list-txt">' + '<div class="title">' + art.properties.wrknm + '</div>' + '<br />' + art.properties.location + '</div>' + '<div class="info-list-img">' + art.properties.img_src + '</div>' + '<br />' + '</div></a>')
    }
    $('a.list-link').click(function (e) {
        alert('now you see what happens when you click a list item!');
        e.preventDefault()
    })
}

function parseNewPoints(data) {
    if (pointsLayer != undefined) {
        map.removeLayer(pointsLayer)
    }
    pointsLayer = new L.GeoJSON();
    var geojsonMarkerOptions = {
        radius: 8,
        fillColor: "#FF6788",
        color: "YELLOW",
        weight: 1,
        opacity: 1,
        fillOpacity: 0.5
    };
    L.geoJson(data, {
        pointToLayer: function (feature, latlng) {
            return L.circleMarker(latlng, geojsonMarkerOptions)
        },
        onEachFeature: function (feature, pointsLayer) {
            pointsLayer.bindPopup(feature.properties.img_src + "<br />" + feature.properties.wrknm + "<br />" + feature.properties.artist + "<br />" + feature.properties.location + '<div class="description">' + feature.properties.descfin + '</div>')
        }
    }).addTo(map)
}

Answer 1:

费利克斯·克林是正确的,但我会在他的评论一点点扩大......

由于L.LayerGroup和L.FeatureGroup(其中L.GeoJSON从延伸)没有方法来检索各层你需要无论是从L.GeoJSON扩展,并添加这样的方法或保持从唯一ID,以自己独立的映射CircleMarker从GeoJSON的。

以GeoJSON不需要一个唯一的ID,但我会假设你的饲料中标记有一个名为“ID”的唯一ID属性。 您需要将这种独特的ID添加到用户可以点击,这样的链接可以选择地图上的正确标记的链接。 然后,你需要存储地图ID来标记,以便检索标记选择在地图上。

markerMap = {}; // a global variable unless you extend L.GeoJSON

// Add the marker id as a data item (called "data-artId") to the "a" element
function addToList(data) {
    for (var i = 0; i < data.features.length; i++) {
        var art = data.features[i];
        $('div#infoContainer').append('<a href="#" class="list-link" data-artId=\"'+art.id+'\" title="' + art.properties.descfin + '"><div class="info-list-item">' + '<div class="info-list-txt">' + '<div class="title">' + art.properties.wrknm + '</div>' + '<br />' + art.properties.location + '</div>' + '<div class="info-list-img">' + art.properties.img_src + '</div>' + '<br />' + '</div></a>')
    }
    $('a.list-link').click(function (e) {
        alert('now you see what happens when you click a list item!');

        //Get the id of the element clicked
        var artId = $(this).data( 'artId' );
        var marker = markerMap[artId];

        //since you're using CircleMarkers the OpenPopup method requires
        //a latlng so I'll just use the center of the circle
        marker.openPopup(marker.getLatLng());
        e.preventDefault()
    })
}

你需要建立markerMap当你从服务器获取数据。 你pointToLayer方法可以修改为做到这一点:

L.geoJson(data, {
    pointToLayer: function (feature, latlng) {
      var marker = new L.CircleMarker( latlng, geojsonMarkerOptions );
      markerMap[feature.id] = marker;
      return marker;
    },...


Answer 2:

我知道这是一个老问题,但传单是它的方式 ,以提供一个内置的解决方案,有一个(有点)内置的方式,现在去实现它?

该方法是使用layerGroup接口。 它提供了一种方法, getLayer ,这听起来像这将是完美的使用ID得到我们的标志。 然而,在这个时候,单张提供任何方法来指定自定义ID或名称

这个问题在Github上讨论如何应该这样做。 随着中说,你可以获取和保存任何标记的(或自动生成的ID iLayer对于这个问题),像这样:

let people = [...],
    group = L.layerGroup()

people.forEach(person => {
    let marker = // ... create marker

    group.addLayer( marker );
    person.marker_id = group.getLayerId(marker)
})

现在,我们有充分的标记的ID保存我们的数据的阵列中的每个支持对象,我们可以很容易地标记以后像这样:

group.getLayer(person.marker_id)

请参阅此笔一个完整的例子, 这个问题更多的选择...



文章来源: How to interact with leaflet marker layer from outside the map?