在 Mapbox GL JS 中显示和隐藏图层的功能
Show and hide features of a layer in Mapbox GL JS
我发现以下代码可以在 Mapbox GL 中显示和隐藏图层:
https://www.mapbox.com/mapbox-gl-js/example/toggle-layers/
这很有用,但是,我只有一个包含所有必要数据的 .geojson 图层(折线),不需要创建单独的图层。
我想实现完全相同的功能,能够在地图菜单中显示和隐藏一层的要素。共有 12 种不同的特征类型,包含在名为 "Type" 的列中。我想打开和关闭类型,就像在示例中一样。
有没有一种简单的方法可以在 JS 中使用 set.Filter
来做到这一点?
https://github.com/mapbox/mapbox-gl-js/blob/e9386d2880cc2c33e9a5a16b9bcb58834026a078/js/ui/map.js#L559-L562
我一直想不出解决办法。
我的 .geojson 图层在这里:
https://iskandarblue.github.io/mapbox/data/simplify_prototype.geojson
<!DOCTYPE html>
<html>
<head>
<meta charset='utf-8' />
<title></title>
<meta name='viewport' content='initial-scale=1,maximum-scale=1,user-scalable=no' />
<script src='https://api.tiles.mapbox.com/mapbox-gl-js/v0.15.0/mapbox-gl.js'></script>
<link href='https://api.tiles.mapbox.com/mapbox-gl-js/v0.15.0/mapbox-gl.css' rel='stylesheet' />
<style>
body { margin:0; padding:0; }
#map { position:absolute; top:0; bottom:0; width:100%; }
</style>
</head>
<body>
<style>
#menu {
background: #fff;
position: absolute;
z-index: 1;
top: 10px;
right: 10px;
border-radius: 3px;
width: 120px;
border: 1px solid rgba(0,0,0,0.4);
font-family: 'Open Sans', sans-serif;
}
#menu a {
font-size: 13px;
color: #404040;
display: block;
margin: 0;
padding: 0;
padding: 10px;
text-decoration: none;
border-bottom: 1px solid rgba(0,0,0,0.25);
text-align: center;
}
#menu a:last-child {
border: none;
}
#menu a:hover {
background-color: #f8f8f8;
color: #404040;
}
#menu a.active {
background-color: #3887be;
color: #ffffff;
}
#menu a.active:hover {
background: #3074a4;
}
</style>
<nav id="menu"></nav>
<div id="map"></div>
<script>
mapboxgl.accessToken = 'pk.eyJ1IjoiaXNrYW5kYXJibHVlIiwiYSI6ImNpazE3MTJldjAzYzZ1Nm0wdXZnMGU2MGMifQ.i3E1_b9QXJS8xXuPy3OTcg';
var map = new mapboxgl.Map({
container: 'map',
style: 'mapbox://styles/mapbox/streets-v8',
zoom: 15,
center: [-71.97722138410576, -13.517379300798098]
});
map.on('style.load', function () {
map.addSource('museums', {
type: 'vector',
url: 'mapbox://mapbox.2opop9hr'
});
map.addLayer({
'id': 'museums',
'type': 'circle',
'source': 'museums',
'paint': {
'circle-radius': 8,
'circle-color': 'rgba(55,148,179,1)'
},
'source-layer': 'museum-cusco'
});
map.addSource('contours', {
type: 'vector',
url: 'mapbox://mapbox.mapbox-terrain-v2'
});
map.addLayer({
'id': 'contours',
'type': 'line',
'source': 'contours',
'source-layer': 'contour',
'layout': {
'line-join': 'round',
'line-cap': 'round'
},
'paint': {
'line-color': '#877b59',
'line-width': 1
}
});
});
addLayer('Contours', 'contours');
addLayer('Museums', 'museums');
function addLayer(name, id) {
var link = document.createElement('a');
link.href = '#';
link.className = 'active';
link.textContent = name;
link.onclick = function (e) {
e.preventDefault();
e.stopPropagation();
var visibility = map.getLayoutProperty(id, 'visibility');
if (visibility === 'visible') {
map.setLayoutProperty(id, 'visibility', 'none');
this.className = '';
} else {
this.className = 'active';
map.setLayoutProperty(id, 'visibility', 'visible');
}
};
var layers = document.getElementById('menu');
layers.appendChild(link);
}
</script>
</body>
</html>
我试图做同样的事情,但最终使用了组合过滤器。与您类似,我有一张带有 geoJSON 源的地图:
map.addSource( 'routes', {
'type': 'geojson',
'data': geoJSON
});
在那张地图上,我有一个包含我所有路线的图层:
map.addLayer({
'id': 'route-no-hover',
'type': 'line',
'source': 'routes',
'layout': {
'line-join': 'round',
'line-cap': 'round'
},
'paint': {
'line-color': '#333',
'line-opacity': 0.25,
'line-width': 4
}
});
我正在列出我的所有路线并为用户提供打开和关闭它们的可用性。听起来我的概念 "routes" 等同于你的 "types"。当用户单击路由名称将其关闭时,我将该路由的 ID 推送到一个名为 'hidden' 的数组中,然后根据这些 ID 创建一系列过滤器:
let hiddenFilters = ['all'];
_.forEach( mapMeta.hidden, ( route ) => {
hiddenFilters.push([
'!=',
'id',
route.toString()
]);
});
map.setFilter( 'route-no-hover', hiddenFilters );
请注意,我使用的是 lodash,但您可以只使用 Array.forEach()
我正在有效地创建一个看起来像 this combining filter described in the filtering docs:
的过滤器
[
"all",
["!=", "id", "routeIDHere"],
["!=", "id", "routeIDHere"],
...
]
这排除了所有具有我指定的 ID 的路由。在你的情况下,我想你会想要像
这样的东西
[
"all",
["!=", "type" "YourTypeNameHere"]
...
]
我发现以下代码可以在 Mapbox GL 中显示和隐藏图层:
https://www.mapbox.com/mapbox-gl-js/example/toggle-layers/
这很有用,但是,我只有一个包含所有必要数据的 .geojson 图层(折线),不需要创建单独的图层。
我想实现完全相同的功能,能够在地图菜单中显示和隐藏一层的要素。共有 12 种不同的特征类型,包含在名为 "Type" 的列中。我想打开和关闭类型,就像在示例中一样。
有没有一种简单的方法可以在 JS 中使用 set.Filter
来做到这一点?
https://github.com/mapbox/mapbox-gl-js/blob/e9386d2880cc2c33e9a5a16b9bcb58834026a078/js/ui/map.js#L559-L562
我一直想不出解决办法。
我的 .geojson 图层在这里: https://iskandarblue.github.io/mapbox/data/simplify_prototype.geojson
<!DOCTYPE html>
<html>
<head>
<meta charset='utf-8' />
<title></title>
<meta name='viewport' content='initial-scale=1,maximum-scale=1,user-scalable=no' />
<script src='https://api.tiles.mapbox.com/mapbox-gl-js/v0.15.0/mapbox-gl.js'></script>
<link href='https://api.tiles.mapbox.com/mapbox-gl-js/v0.15.0/mapbox-gl.css' rel='stylesheet' />
<style>
body { margin:0; padding:0; }
#map { position:absolute; top:0; bottom:0; width:100%; }
</style>
</head>
<body>
<style>
#menu {
background: #fff;
position: absolute;
z-index: 1;
top: 10px;
right: 10px;
border-radius: 3px;
width: 120px;
border: 1px solid rgba(0,0,0,0.4);
font-family: 'Open Sans', sans-serif;
}
#menu a {
font-size: 13px;
color: #404040;
display: block;
margin: 0;
padding: 0;
padding: 10px;
text-decoration: none;
border-bottom: 1px solid rgba(0,0,0,0.25);
text-align: center;
}
#menu a:last-child {
border: none;
}
#menu a:hover {
background-color: #f8f8f8;
color: #404040;
}
#menu a.active {
background-color: #3887be;
color: #ffffff;
}
#menu a.active:hover {
background: #3074a4;
}
</style>
<nav id="menu"></nav>
<div id="map"></div>
<script>
mapboxgl.accessToken = 'pk.eyJ1IjoiaXNrYW5kYXJibHVlIiwiYSI6ImNpazE3MTJldjAzYzZ1Nm0wdXZnMGU2MGMifQ.i3E1_b9QXJS8xXuPy3OTcg';
var map = new mapboxgl.Map({
container: 'map',
style: 'mapbox://styles/mapbox/streets-v8',
zoom: 15,
center: [-71.97722138410576, -13.517379300798098]
});
map.on('style.load', function () {
map.addSource('museums', {
type: 'vector',
url: 'mapbox://mapbox.2opop9hr'
});
map.addLayer({
'id': 'museums',
'type': 'circle',
'source': 'museums',
'paint': {
'circle-radius': 8,
'circle-color': 'rgba(55,148,179,1)'
},
'source-layer': 'museum-cusco'
});
map.addSource('contours', {
type: 'vector',
url: 'mapbox://mapbox.mapbox-terrain-v2'
});
map.addLayer({
'id': 'contours',
'type': 'line',
'source': 'contours',
'source-layer': 'contour',
'layout': {
'line-join': 'round',
'line-cap': 'round'
},
'paint': {
'line-color': '#877b59',
'line-width': 1
}
});
});
addLayer('Contours', 'contours');
addLayer('Museums', 'museums');
function addLayer(name, id) {
var link = document.createElement('a');
link.href = '#';
link.className = 'active';
link.textContent = name;
link.onclick = function (e) {
e.preventDefault();
e.stopPropagation();
var visibility = map.getLayoutProperty(id, 'visibility');
if (visibility === 'visible') {
map.setLayoutProperty(id, 'visibility', 'none');
this.className = '';
} else {
this.className = 'active';
map.setLayoutProperty(id, 'visibility', 'visible');
}
};
var layers = document.getElementById('menu');
layers.appendChild(link);
}
</script>
</body>
</html>
我试图做同样的事情,但最终使用了组合过滤器。与您类似,我有一张带有 geoJSON 源的地图:
map.addSource( 'routes', {
'type': 'geojson',
'data': geoJSON
});
在那张地图上,我有一个包含我所有路线的图层:
map.addLayer({
'id': 'route-no-hover',
'type': 'line',
'source': 'routes',
'layout': {
'line-join': 'round',
'line-cap': 'round'
},
'paint': {
'line-color': '#333',
'line-opacity': 0.25,
'line-width': 4
}
});
我正在列出我的所有路线并为用户提供打开和关闭它们的可用性。听起来我的概念 "routes" 等同于你的 "types"。当用户单击路由名称将其关闭时,我将该路由的 ID 推送到一个名为 'hidden' 的数组中,然后根据这些 ID 创建一系列过滤器:
let hiddenFilters = ['all'];
_.forEach( mapMeta.hidden, ( route ) => {
hiddenFilters.push([
'!=',
'id',
route.toString()
]);
});
map.setFilter( 'route-no-hover', hiddenFilters );
请注意,我使用的是 lodash,但您可以只使用 Array.forEach()
我正在有效地创建一个看起来像 this combining filter described in the filtering docs:
的过滤器[
"all",
["!=", "id", "routeIDHere"],
["!=", "id", "routeIDHere"],
...
]
这排除了所有具有我指定的 ID 的路由。在你的情况下,我想你会想要像
这样的东西[
"all",
["!=", "type" "YourTypeNameHere"]
...
]