Unterschiedliche Karten mit Mapbox GL ermöglichen
Jeder hat seine Vorliebe. Mancher bevorzugt Satellitenaufnahmen. Ein anderer gemalte Karten. Zusätzlich bieten unterschiedliche Style je nach Anwendungszweck Vor- und Nachteile. Deshalb ist ideal, wenn es man es Websitebesuchern ermöglicht, die Lieblingskarte per Steuerelement auszuwählen. Hier ein Beispiel für einen Kartentypwechsler oder Style-Switcher:
<!-- https://raw.githubusercontent.com/astridx/maplibreexamples/main/plugins/maplibre-style-switcher.html -->
<html>
<head>
<title>Mapbox style switcher</title>
<meta charset="UTF-8" />
<script src="../.env"></script>
<script src="https://unpkg.com/maplibre-gl@1.14.0-rc.1/dist/maplibre-gl.js"></script>
<link
href="https://unpkg.com/maplibre-gl@1.14.0-rc.1/dist/maplibre-gl.css"
rel="stylesheet"
/>
<script>
class MapboxStyleSwitcherControl {
constructor(styles, defaultStyle) {
this.styles = styles || MapboxStyleSwitcherControl.DEFAULT_STYLES
this.defaultStyle =
defaultStyle || MapboxStyleSwitcherControl.DEFAULT_STYLE
this.onDocumentClick = this.onDocumentClick.bind(this)
}
getDefaultPosition() {
const defaultPosition = 'top-right'
return defaultPosition
}
onAdd(map) {
this.map = map
this.controlContainer = document.createElement('div')
this.controlContainer.classList.add('mapboxgl-ctrl')
this.controlContainer.classList.add('mapboxgl-ctrl-group')
this.mapStyleContainer = document.createElement('div')
this.styleButton = document.createElement('button')
this.styleButton.type = 'button'
this.mapStyleContainer.classList.add('mapboxgl-style-list')
for (const style of this.styles) {
const styleElement = document.createElement('button')
styleElement.type = 'button'
styleElement.innerText = style.title
styleElement.classList.add(style.title.replace(/[^a-z0-9-]/gi, '_'))
styleElement.dataset.uri = JSON.stringify(style.uri)
styleElement.addEventListener('click', (event) => {
const srcElement = event.srcElement
if (srcElement.classList.contains('active')) {
return
}
this.map.setStyle(JSON.parse(srcElement.dataset.uri))
this.mapStyleContainer.style.display = 'none'
this.styleButton.style.display = 'block'
const elms =
this.mapStyleContainer.getElementsByClassName('active')
while (elms[0]) {
elms[0].classList.remove('active')
}
srcElement.classList.add('active')
})
if (style.title === this.defaultStyle) {
styleElement.classList.add('active')
}
this.mapStyleContainer.appendChild(styleElement)
}
this.styleButton.classList.add('mapboxgl-ctrl-icon')
this.styleButton.classList.add('mapboxgl-style-switcher')
this.styleButton.addEventListener('click', () => {
this.styleButton.style.display = 'none'
this.mapStyleContainer.style.display = 'block'
})
document.addEventListener('click', this.onDocumentClick)
this.controlContainer.appendChild(this.styleButton)
this.controlContainer.appendChild(this.mapStyleContainer)
return this.controlContainer
}
onRemove() {
if (
!this.controlContainer ||
!this.controlContainer.parentNode ||
!this.map ||
!this.styleButton
) {
return
}
this.styleButton.removeEventListener('click', this.onDocumentClick)
this.controlContainer.parentNode.removeChild(this.controlContainer)
document.removeEventListener('click', this.onDocumentClick)
this.map = undefined
}
onDocumentClick(event) {
if (
this.controlContainer &&
!this.controlContainer.contains(event.target) &&
this.mapStyleContainer &&
this.styleButton
) {
this.mapStyleContainer.style.display = 'none'
this.styleButton.style.display = 'block'
}
}
}
// https://cloud.maptiler.com/maps/
MapboxStyleSwitcherControl.DEFAULT_STYLE = 'Streets'
MapboxStyleSwitcherControl.DEFAULT_STYLES = [
{
title: 'Baisc',
uri:
'https://api.maptiler.com/maps/basic/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Light',
uri:
'https://api.maptiler.com/maps/bright/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Outdoors',
uri:
'https://api.maptiler.com/maps/outdoor/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Satellite Hybrid',
uri:
'https://api.maptiler.com/maps/hybrid/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Streets',
uri:
'https://api.maptiler.com/maps/streets/style.json?key=' +
config.MAPTILER_TOKEN,
},
]
</script>
<style>
.mapboxgl-style-list {
display: none;
}
.mapboxgl-ctrl-group .mapboxgl-style-list button {
background: none;
border: none;
cursor: pointer;
display: block;
font-size: 14px;
padding: 8px 8px 6px;
text-align: right;
width: 100%;
height: auto;
}
.mapboxgl-style-list button.active {
font-weight: bold;
}
.mapboxgl-style-list button:hover {
background-color: rgba(0, 0, 0, 0.05);
}
.mapboxgl-style-list button + button {
border-top: 1px solid #ddd;
}
.mapboxgl-style-switcher {
background-image: url();
background-position: center;
background-repeat: no-repeat;
background-size: 70%;
}
</style>
</head>
<body>
<h2 style="font-family: sans-serif">Mapbox style switcher</h2>
<div id="map" style="height: 60%"></div>
<script>
const map = new maplibregl.Map({
container: 'map',
style:
'https://api.maptiler.com/maps/streets/style.json?key=' +
config.MAPTILER_TOKEN,
center: [-122.4194, 37.7788],
zoom: 12,
})
var styles = (MapboxStyleDefinition = [
{
title: 'Basic',
uri:
'https://api.maptiler.com/maps/streets/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Light',
uri:
'https://api.maptiler.com/maps/bright/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Satellite Hybrid',
uri:
'https://api.maptiler.com/maps/hybrid/style.json?key=' +
config.MAPTILER_TOKEN,
},
{
title: 'Outdoors',
uri:
'https://api.maptiler.com/maps/outdoor/style.json?key=' +
config.MAPTILER_TOKEN,
},
])
map.addControl(new MapboxStyleSwitcherControl(styles))
</script>
</body>
</html>
Demo
Quellcode
Gatsby Starter mit dieser Funktion - Gatsby Starter Demo