javascript - 我怎样才能用 Leaflet.draw 抓取一些标记?

标签 javascript leaflet leaflet.draw

上下文:

我制作了一张 map ,并在其中填充了大约 300 个随机标记。我可以通过单击弹出窗口中的链接来“选择”标记,并激活选择以显示来自的数据。我还有 Leaflet.draw 插件来绘制圆形、矩形和自定义形状等形状,我想用它来“选择”几个标记。

问题

如何获取落在绘制的 leaflet.draw 形状内的标记的传单标记对象,以便我可以编辑它们?我似乎无法做出选择,它要么不选择任何标记,要么选择所有标记。

代码片段,去除了不必要的代码:

const drawControl = new L.Control.Draw({
    draw: {
        marker   : false,
        polygon  : true,
        polyline : false,
        rectangle: true,
        circle   : {
            metric: 'metric'
        }
    },
    edit: false
});

const map = L.map('map', {
    layers: [streets, light]
}).setView([CONFIG.MAP.LATITUDE, CONFIG.MAP.LONGITUDE], CONFIG.MAP.ZOOMLEVEL)

map.addControl(drawControl);

map.on(L.Draw.Event.DRAWSTOP, e => {

    const hello = e.target;

    console.log(hello);
    e.target.eachLayer(layer => {
        if (layer.options.icon) {
            console.log(layer);
        }
    });

});

最佳答案

您想要的大部分内容都可以使用 Leaflet 的实用方法轻松完成。如果你想用像 L.Polygon 这样的复杂形状来做到这一点,你将需要像 TurfJS 这样的东西。

对于 L.Circle,您需要计算圆心之间的距离并将其与半径进行比较:

var marker = new L.Marker(...),
    circle = new L.Circle(...);

var contains = circle.getLatLng().distanceTo(marker.getLatLng()) < circle.getRadius();

对于 L.Rectangle,您需要获取它的边界对象并使用 contains 方法:

var marker = new L.Marker(...),
    rectangle = new L.Rectangle(...);

var contains = rectangle.getBounds().contains(marker.getLatLng());

对于复杂的多边形,我会使用 Turf,但那里有更多的库和插件。下面是一个使用 Turf 的 inside 方法的示例。它以 GeoJSON 点和多边形特征作为参数,因此请注意转换:

var marker = new L.Marker(...),
    polygon = new L.Polygon(...);

var contains = turf.inside(marker.toGeoJSON(), polygon.toGeoJSON());

您可以将它们包装到每个类的便捷方法中:

L.Polygon.include({
    contains: function (latLng) {
        return turf.inside(new L.Marker(latLng).toGeoJSON(), this.toGeoJSON());
    } 
});

L.Rectangle.include({
    contains: function (latLng) {
        return this.getBounds().contains(latLng);
    }
});

L.Circle.include({
    contains: function (latLng) {
        return this.getLatLng().distanceTo(latLng) < this.getRadius();
    }
});

var marker = new L.Marker(...),
    polygon = new L.Polygon(...),
    rectangle = new L.Rectangle(...),
    circle = new L.Circle(...);

polygon.contains(marker.getLatLng());
rectangle.contains(marker.getLatLng());
circle.contains(marker.getLatLng());

请注意,如果您实现多边形方法,则不需要矩形方法。由于矩形是从多边形扩展而来的,它将继承该方法。我把它留在那儿是为了完整。

现在迭代你的标记并比较它们很容易:

map.on(L.Draw.Event.CREATED, function (e) {
    markers.eachLayer(function (marker) {
        if (!e.layer.contains(marker.getLatLng())) {
            marker.remove();
        }
    });
});

希望对您有所帮助,这是一个有效的片段:

var map = new L.Map('leaflet', {
    'center': [0, 0],
    'zoom': 0
});

var markers = new L.LayerGroup().addTo(map);

for (var i = 0; i < 300; i++) {
    var marker = new L.Marker([
        (Math.random() * (90 - -90) + -90).toFixed(5) * 1,
        (Math.random() * (180 - -180) + -180).toFixed(5) * 1
    ]).addTo(markers);
}

new L.Control.Draw({
    draw: {
        marker   : false,
        polygon  : true,
        polyline : false,
        rectangle: true,
        circle   : {
            metric: 'metric'
        }
    },
    edit: false
}).addTo(map);

L.Polygon.include({
    contains: function (latLng) {
        return turf.inside(new L.Marker(latLng).toGeoJSON(), this.toGeoJSON());
    } 
});

L.Rectangle.include({
    contains: function (latLng) {
        return this.getBounds().contains(latLng);
    }
});

L.Circle.include({
    contains: function (latLng) {
        return this.getLatLng().distanceTo(latLng) < this.getRadius();
    }
});

map.on(L.Draw.Event.CREATED, function (e) {
    markers.eachLayer(function (marker) {
        if (!e.layer.contains(marker.getLatLng())) {
            marker.remove();
        }
    });
});
body {
    margin: 0;
}

html, body, #leaflet {
    height: 100%;
}
<!DOCTYPE html>
<html>
  <head>
    <title>Leaflet 1.0.3</title>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <link type="text/css" rel="stylesheet" href="//unpkg.com/leaflet@1.0.3/dist/leaflet.css" />
    <link type="text/css" rel="stylesheet" href="//cdnjs.cloudflare.com/ajax/libs/leaflet.draw/0.4.9/leaflet.draw.css" />
  </head>
  <body>
    <div id="leaflet"></div>
    <script type="application/javascript" src="//unpkg.com/leaflet@1.0.3/dist/leaflet.js"></script>
    <script type="application/javascript" src="//cdnjs.cloudflare.com/ajax/libs/leaflet.draw/0.4.9/leaflet.draw.js"></script>
    <script type="application/javascript" src="//unpkg.com/@turf/turf@latest/turf.min.js"></script>
  </body>
</html>

关于javascript - 我怎样才能用 Leaflet.draw 抓取一些标记?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44261122/

相关文章:

javascript - Leaflet.Draw编辑和删除按钮不起作用

javascript - 在标记mapbox.js上添加自定义图标

javascript - jQuery:检查 BR 作为第一个或最后一个元素是否存在,包括段落中的文本节点

javascript - 使用 jszip 将 pdf 的 zip 保存到文件夹中

javascript - 摆脱滚动条

javascript - 如何更改传单属性

python - 如何在 folium 中创建相对于 map 的固定大小的标记?

javascript - 识别何时单击 Leaflet Draw 取消按钮

javascript - jQuery 链如何在返回数组的同时发挥作用?

javascript - 从 SQL 获取数据到 Leaflet