block by micahstubbs 89c484e87298425f3d8c09f37362243e

deck.gl + Mapbox Extruded-Hex Map

Full Screen

a visualization of personal injury road accidents in the U.K. from 1979 through 2016. It uses deck.gl’s HexagonLayer to aggregate data within the boundary of each hexagon cell. The HexagonLayer is inserted underneath Mapbox’s label layers. source data


this iteration moves the controls to the North Sea. It also gives the controls a darker background and aligns the text.


an iteration that makes the code nice to work with, for my personal definition of nice.

an iteration on the deck.gl + Mapbox HexagonLayer Example from @cxiaoji, discovered from the Uber Visualization Newsletter

in github repo form at https://github.com/micahstubbs/deck-gl-experiments/tree/master/hexagon-layer/02

index.js

mapboxgl.accessToken =
  'pk.eyJ1IjoidWJlcmRhdGEiLCJhIjoiY2pudzRtaWloMDAzcTN2bzN1aXdxZHB5bSJ9.2bkj3IiRC8wj3jLThvDGdA'

const { MapboxLayer, HexagonLayer } = deck

const map = new mapboxgl.Map({
  container: document.body,
  style: 'mapbox://styles/mapbox/dark-v9',
  center: [-1.4157, 52.2324],
  zoom: 6,
  pitch: 40.5
})

const OPTIONS = ['radius', 'coverage', 'upperPercentile']
const COLOR_RANGE = [
  [1, 152, 189],
  [73, 227, 206],
  [216, 254, 181],
  [254, 237, 177],
  [254, 173, 84],
  [209, 55, 78]
]
const LIGHT_SETTINGS = {
  lightsPosition: [-0.144528, 49.739968, 8000, -3.807751, 54.104682, 8000],
  ambientRatio: 0.4,
  diffuseRatio: 0.6,
  specularRatio: 0.2,
  lightsStrength: [0.8, 0.0, 0.8, 0.0],
  numberOfLights: 2
}

let hexagonLayer

map.on('load', () => {
  hexagonLayer = new MapboxLayer({
    type: HexagonLayer,
    id: 'heatmap',
    data: d3.csv('./heatmap-data.csv'),
    radius: 1000,
    coverage: 1,
    upperPercentile: 100,
    colorRange: COLOR_RANGE,
    elevationRange: [0, 1000],
    elevationScale: 250,
    extruded: true,
    getPosition: d => [Number(d.lng), Number(d.lat)],
    lightSettings: LIGHT_SETTINGS,
    opacity: 1
  })

  map.addLayer(hexagonLayer, 'waterway-label')
})

OPTIONS.forEach(key => {
  document.getElementById(key).oninput = evt => {
    const value = Number(evt.target.value)
    document.getElementById(key + '-value').innerHTML = value
    if (hexagonLayer) {
      hexagonLayer.setProps({ [key]: value })
    }
  }
})

index.html

<html>
  <head>
    <title>deck.gl + Mapbox HexagonLayer</title>
    <meta
      name="viewport"
      content="initial-scale=1,maximum-scale=1,user-scalable=no"
    />

    <script src="https://unpkg.com/deck.gl@^6.2.0/deckgl.min.js"></script>
    <script src="https://api.tiles.mapbox.com/mapbox-gl-js/v0.50.0/mapbox-gl.js"></script>
    <link
      rel="stylesheet"
      type="text/css"
      href="https://api.tiles.mapbox.com/mapbox-gl-js/v0.50.0/mapbox-gl.css"
    />
    <script src="https://d3js.org/d3.v5.min.js"></script>
    <link rel="stylesheet" type="text/css" href="./index.css" />
  </head>

  <body>
    <div id="control-panel">
      <div>
        <label>Radius</label>
        <input
          id="radius"
          type="range"
          min="1000"
          max="20000"
          step="1000"
          value="1000"
        />
        <span id="radius-value"></span>
      </div>
      <div>
        <label>Coverage</label>
        <input
          id="coverage"
          type="range"
          min="0"
          max="1"
          step="0.1"
          value="1"
        />
        <span id="coverage-value"></span>
      </div>
      <div>
        <label>Upper Percentile</label>
        <input
          id="upperPercentile"
          type="range"
          min="90"
          max="100"
          step="1"
          value="100"
        />
        <span id="upperPercentile-value"></span>
      </div>
    </div>
  </body>
  <script src="./index.js" type="text/javascript"></script>
</html>

index.css

body {
  font-family: Helvetica, Arial, sans-serif;
  width: 100vw;
  height: 100vh;
  margin: 0;
}
#control-panel {
  position: absolute;
  background: #737373;
  top: 0;
  right: 0;
  margin: 12px;
  padding: 20px;
  font-size: 12px;
  line-height: 1.5;
  z-index: 1;
  border-radius: 8px;
  color: #f0f0f0;
}
label {
  display: inline-block;
  width: 90px;
  text-align: right;
  vertical-align: top;
  line-height: 20px;
  margin-right: 8px;
}