Published
Edited
Sep 13, 2020
Importers
11 stars
Insert cell
Insert cell
Insert cell
import { hexdata } from '@enjalot/purpleair-historical-data'
Insert cell
hexdata
Insert cell
groupedData = d3.group(hexdata, d => d.day)
Insert cell
times = Array.from(groupedData.keys())
Insert cell
timeData = groupedData.get(currentTime)
Insert cell
points = timeData.map(d => [d.lng, d.lat, d.pm1])
Insert cell
Insert cell
Insert cell
Insert cell
md`### ${timeFormat(new Date(currentTime*1000))}`
Insert cell
mapSVG = svg`<svg width=${width} height=${height}>
<defs>
<clipPath id="californiaClip">
<path></path>
</clipPath>
</defs>
<path class="state"></path>
<g class="voronoi"></g>
</svg>`
Insert cell
legend({color: colorPM1})
Insert cell
import {legend} from "@d3/color-legend"
Insert cell
projection = d3.geoAlbersUsa()
// magically fits map geography to width/height!
.fitSize([width, height], california[0])
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
path = d3.geoPath(projection)
Insert cell
colorPM1 = d3.scaleLinear()
.domain([0,50, 100, 150, 200, 250])
.range(["green", "yellow", "orange", "red", "maroon", "maroon"])
Insert cell
function render(svg) {
let svgSelection = d3.select(svg)
// california is an array with 1 element in it (the GeoJSON of its boundaries)
// the path function takes in a GeoJSON element and returns an SVG path string
// we can use this both to define the clippath and draw a boundary
let cali = path(california[0])

svgSelection.select("#californiaClip")
.select("path")
.attr('d', cali)
.attr('stroke', 'black')
.attr('fill', 'none');

// create the outline of california to clip our voronoi
// using https://observablehq.com/@manishearth/uc-california-voronoi-map
//what are defs
svgSelection.select("path.state")
// whats the dif between .data and .datum?
// When do you use each?
.attr("fill", "none")
.attr("stroke", "#111")
.attr("stroke-width", 6)
.attr("stroke-linejoin", "round")
.attr("d", cali);

svgSelection.select("g.voronoi")
// .append("g")
.attr("stroke", "none")
.attr("pointer-events", "all")
.selectAll("path")
.data(d3.geoVoronoi().polygons(points).features)
.join("path")
.attr("d", d3.geoPath(projection))
.attr("fill", d => colorPM1(d.properties.site[2]))
.attr("stroke", "black")
.attr("clip-path", "url(#californiaClip)");

return svgSelection;
}
Insert cell
Insert cell
timeFormat = d3.timeFormat("%Y-%m-%d %H:%M")
Insert cell

rendered = render(mapSVG)
Insert cell
height = 600
Insert cell
Insert cell

One platform to build and deploy the best data apps

Experiment and prototype by building visualizations in live JavaScript notebooks. Collaborate with your team and decide which concepts to build out.
Use Observable Framework to build data apps locally. Use data loaders to build in any language or library, including Python, SQL, and R.
Seamlessly deploy to Observable. Test before you ship, use automatic deploy-on-commit, and ensure your projects are always up-to-date.
Learn more