I am using mapbox-gl and downloaded vector tiles from osm2vectortiles.org
I'd like to have the map visible only within an known polygon, and cannot find any way to pull this off.
I can imagine a couple ways to approach the problem, and every path has lead me to no answers. Here are some approaches I've tried to research:
What's the right way to approach this problem?
All of the options you've mentioned could work. You can cut the original vector data to fit the mask region(as in option 4) or you can draw a mask layer on top of your data(option 3).
Here is how you can do it with a mask layer:
https://jsfiddle.net/kmandov/cr2rav7v/
Add your mask as a fill layer:
map.addSource('mask', {
"type": "geojson",
"data": polyMask(mask, bounds)
});
map.addLayer({
"id": "zmask",
"source": "mask",
"type": "fill",
"paint": {
"fill-color": "white",
'fill-opacity': 0.999
}
});
You can create your mask in advance (say with QGis, or turf) or you can cut the mask directly in the browser using turf:
function polyMask(mask, bounds) {
var bboxPoly = turf.bboxPolygon(bounds);
return turf.difference(bboxPoly, mask);
}
var bounds = [-122.5336, 37.7049, -122.3122, 37.8398]; // wsen
var mask = turf.polygon([
[
[-122.43764877319336,
37.78645343442073
],
[-122.40056991577148,
37.78930232286027
],
[-122.39172935485838,
37.76630458915842
],
[-122.43550300598145,
37.75646561597495
],
[-122.45378494262697,
37.7781096293495
],
[-122.43764877319336,
37.78645343442073
]
]
]);
Probably, you would like to prevent the user from going outside of the mask boundaries by providing maxBounds
:
var map = new mapboxgl.Map({
container: 'map',
style: 'mapbox://styles/mapbox/light-v9',
center: [-122.42116928100586, 37.77532815168286],
maxBounds: bounds,
zoom: 12
});
Note that there seems to be a bug and you need to set fill-opacity
to something like 0.999