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:

Full Screen Map with Style Switcher Gatsby Mapbox GL Starter

<!--  https://raw.githubusercontent.com/astridx/mapboxexamples/master/plugins/mapboxgl-style-switcher.html -->

<html>
  <head>
    <title>Mapbox style switcher</title>
    <meta charset="UTF-8" />
    <link
      href="https://api.mapbox.com/mapbox-gl-js/v1.10.0/mapbox-gl.css"
      rel="stylesheet"
    />
    <script src="https://api.mapbox.com/mapbox-gl-js/v1.12.0/mapbox-gl.js"></script>
    <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'
          }
        }
      }
      MapboxStyleSwitcherControl.DEFAULT_STYLE = 'Streets'
      MapboxStyleSwitcherControl.DEFAULT_STYLES = [
        { title: 'Dark', uri: 'mapbox://styles/mapbox/dark-v10' },
        { title: 'Light', uri: 'mapbox://styles/mapbox/light-v10' },
        { title: 'Outdoors', uri: 'mapbox://styles/mapbox/outdoors-v11' },
        {
          title: 'Satellite',
          uri: 'mapbox://styles/mapbox/satellite-streets-v11',
        },
        { title: 'Streets', uri: 'mapbox://styles/mapbox/streets-v11' },
      ]
    </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>
      mapboxgl.accessToken = '<Zugriffstoken>'
      const map = new mapboxgl.Map({
        container: 'map',
        style: 'mapbox://styles/mapbox/streets-v9',
        center: [-122.4194, 37.7788],
        zoom: 12,
      })

      var styles = (MapboxStyleDefinition = [
        {
          title: 'Dark',
          uri: 'mapbox://styles/mapbox/dark-v9',
        },
        {
          title: 'Light',
          uri: 'mapbox://styles/mapbox/light-v9',
        },
      ])

      map.addControl(new MapboxStyleSwitcherControl(styles))
    </script>
  </body>
</html>

Demo
Quellcode
Gatsby Starter mit dieser Funktion - Gatsby Starter Demo

Comments

Vor oder zurück ...