Published
Edited
May 10, 2022
2 forks
7 stars
Global temperature trendsBeeswarm, Log ScaleContoursDensity contoursBubble chart componentCircle packing componentExploring Data with Vega-LiteStreaming data into VegaVega-Lite Line ChartBrushable scatterplotVega-Lite ScatterplotNight Skies — Lights and Light Pollution WebGL GlobePhysics based t-SNEGraphvizRidgeline plotZoomable sunburstHierarchical edge bundlingMethods of Comparison, ComparedChord diagramA Guide to Guides: Axes & Legends in VegaPSR B1919+21HexbinStacked-to-grouped barsTree of LifeHeat indexZoomable circle packingMarimekkoVizsla and Vega-LiteDirectly labelling linesParallel coordinatesCollapsible treeTangled tree visualizationMarey’s TrainsWorld History TimelineSmall multiple chart cartogramThe Real MVP in the NBAAnimated treemapDensity Contour Matrix with BrushingStars and constellationsHertzsprung–Russell diagram
The Coronavirus landscape
GitHub BurndownCandlestick ChartConcentration values vs. TimeA few days of CO2 levels in my homeIrregular bins histogramELD ViewerDistributions and summary statistics - a collection of Plot examplesMermaidBivariate Bubble MapD3 galleryCalendarStacked bar chartDot plotConnected scatterplotCandlestick chartHistogramForce-directed graphDisjoint force-directed graphIndex chartSankey diagramLine chart, percent changeStacked area chartBubble chartArea chartHorizontal bar chartRadial tidy treeCircle packingIcicleStreamgraphTidy treeCluster treeSunburstHorizon chartBox plotScatterplotDifference chartBand chartBar chart transitionsTreemapLine chart, multiple seriesScatterplot matrixBrushable scatterplot matrixPlayfair's Wheat and Wages
Also listed in…
Plot Examples
Insert cell
Insert cell
Type JavaScript, then Shift-Enter. Ctrl-space for more options. Arrow ↑/↓ to switch modes.

Insert cell
Insert cell
Insert cell
p = Plot.plot({
caption: html`Data from Johns Hopkins: chart by Ben Schmidt:<br>https://observablehq.com/@bmschmidt/the-coronavirus-landscape`,
height: places.size * 25,
width: width,
color: {
scheme: "dark2",
domain: ordering.filter(d => places.has(d)),

},
// marginTop: 50,
// marginBottom: 50,
x: {
tickFormat: "%b",
grid: true
},
y: {
position: "right",
range: [20, -80],
axis: null
},
fy: {
domain: ordering.filter(d => places.has(d)),
//padding: 0,
round: true,
axis: null,
display: false,
"fontSize": 0
},
facet: {
data: smooth_filtered,
y: "place",
marginLeft: 30,
marginRight: 150
},
marks: [
Plot.ruleY([0])
,
Plot.dot(smooth_filtered, {
filter: d => '' + d.date == cap_date_string,
r: 5,
x: "date",
y: 0,
dy: -4,
dx: 3,
fillOpacity: .5,
fy: "place",
fill: "place",
// curve: "catmull-rom"
}),
Plot.text(smooth_filtered, {
filter: d => '' + d.date == cap_date_string,
r: 5,
x: "date",
y: 0,
dy: -3,
textAnchor: 'start',
dx: 7,
fontSize: 14,
fy: "place",
text: "place",
// curv
}),
Plot.areaY(smooth_filtered, {
x: "date",
y: quantity,
fy: "place",
fill: "place",
title: "place",
fillOpacity: 0.5,
// curve: "catmull-rom"
}),
Plot.line(smooth_filtered, {
x: "date",
fill: "avg",
fy: "place",
y: quantity,
title: quantity
// rx: 20 // uncomment for circles
}),
]
})
Insert cell
smooth_filtered = smoothed.filter(d => d.date <= cap_date)
Insert cell
cap_date = new Date(maxdate)
Insert cell
cap_date_string = '' + cap_date
Insert cell
maxdate = "" + d3.max(smoothed, d => d.date)
Insert cell
ordering = {
if (dataset == "States") return linear_us_state_order
if (dataset == "Countries") return root_ordering
if (dataset == "US Metros") return metro_ordering
throw "Bad dataset"
}
Insert cell
import {Scrubber} from "@mbostock/scrubber"
Insert cell
async function insert_json(name, array, client) {
await client.query(`DROP TABLE IF EXISTS ${name}`)
const texted = d3.csvFormat(array)
await client.insertCSV(name, new TextEncoder("utf-8").encode(texted))
return
}
Insert cell
fips_msa = {
client
const fips_msa = await fetch(`https://gist.githubusercontent.com/bmschmidt/62ed8da02750cbf0b75278cfe06f7bb7/raw/37cc7dfe54b71c041fbc2d3d9cd214443d206c3c/msa-csa.json`).then(d => d.json())
const rows = [...Object.entries(fips_msa)].map(([fips, msa]) => {
const metro_state = msa.split(", ").pop().split("-")[0]
return {fips, msa, metro_state}
})
const texted = d3.csvFormat(rows)
await insert_json("fips_msa", rows, client)
return client.table('SELECT * FROM fips_msa')
}
Insert cell
viewof metro_values = {
fips_msa
return client.table(`SELECT metro_state, msa as name, SUM(population) as pop FROM fips_msa NATURAL JOIN county_populations GROUP BY metro_state, msa`)

}
Insert cell
metro_values[0]["metro_state"]
Insert cell
metro_ordering = {
metro_values.sort((a, b) => linear_us_state_order.indexOf(a["metro_state"]) - linear_us_state_order.indexOf(b["metro_state"]))
return metro_values.map(d => d["name"])
}
Insert cell
d3 = require('d3-array', 'd3-fetch', 'd3-dsv')
Insert cell
import {DuckDBClient} from '@cmudig/duckdb'

Insert cell
datenames = dates.map(d => {const [a, b, c] = d.split("/"); return `20${c}-${a}-${b}`})
Insert cell
client.table("DESCRIBE global")
Insert cell
dates = client.query("DESCRIBE global").then(d => d.getChild("column_name").toArray().slice(4))

Insert cell
client = {
const client = new DuckDBClient()
const db = await client.db()
await client.insertCSV("global", r)
await client.insertCSV("us", us_raw)
return client
}
Insert cell
places = new Set(smoothed.map(d => d.place).filter(d => ordering.indexOf(d) > -1))
Insert cell
mindate = new Date("2020-03-05")
Insert cell
smoothed = smoothed_unfiltered.filter(d => new Set(ordering).has(d.place)).filter(d => d.date > mindate)//.filter(d=>d.place.match("New ") )
Insert cell
viewof smoothed_unfiltered = {
wide
await client.query(`DROP TABLE IF EXISTS pops`)
await client.query(`CREATE TABLE pops (place CHAR, population INT)`)
let pop_set;
if (dataset === "States") {
pop_set = state_pops
} else if (dataset == "Countries") {
pop_set = populations
} else {
pop_set = metro_values.map( d=> {
return {
name : d.get("name"),
"population_total" : d.get("pop")
}
})
}
let popmin = 0
if (dataset == "Countries") popmin = 10000000
if (dataset == "US Metros") popmin = 150000
const pop_vals = pop_set.map(({name, population_total}) => `('${name.replace("'", "")}', ${population_total})`).join(", ")
await client.query(`INSERT INTO pops VALUES ${pop_vals}`)
return client.table(`SELECT *, CASE WHEN smoothed > 0 THEN smoothed*1000000/population ELSE 0 END as avg FROM ( SELECT *, (AVG(new_cases) OVER (
PARTITION BY place
ORDER BY date
RANGE BETWEEN INTERVAL 13 DAYS PRECEDING
AND INTERVAL 0 DAYS FOLLOWING
)) as smoothed FROM tmp NATURAL JOIN pops WHERE population > ${popmin} order by place, date) t1 `)
}
Insert cell
d = client.db()
Insert cell
metros = {
wide;
fips_msa;

await client.query(`DROP TABLE IF EXISTS metro_long`)

// Create metro counts
await client.query(`CREATE TABLE metro_long AS SELECT
msa place, date, sum(cases) as cases FROM us_long_raw NATURAL JOIN "fips_msa" GROUP BY place, date`)
return client.table(`SELECT * FROM metro_long`)
}
Insert cell
wide = {
await client.query(`DROP TABLE IF EXISTS long`)
await client.query(`DROP TABLE IF EXISTS us_long`)
await client.query(`DROP TABLE IF EXISTS us_long_raw`)
await client.query(`
CREATE TABLE IF NOT EXISTS long AS
SELECT "place", date, SUM(cases) cases FROM (
SELECT
"Country/Region" place,
unnest(${JSON.stringify(datenames).replaceAll('"', "'")})::DATE as date,
unnest(${JSON.stringify(dates)}) as cases
FROM global)
as t1
GROUP BY place, "date" ORDER BY place, date
`)
await client.query(`
CREATE TABLE us_long_raw AS
SELECT
"Province_State", FIPS,
unnest(${JSON.stringify(datenames).replaceAll('"', "'")})::DATE as date,
unnest(${JSON.stringify(dates)}) as cases
FROM us
`)
await client.query(`
CREATE TABLE us_long AS SELECT "Province_State" place, date, SUM(cases) cases
FROM "us_long_raw" GROUP BY place, date ORDER BY place, date`)
}
Insert cell
{
wide
let upper_tab = "long"
if (dataset == "States") upper_tab = "us_long"
if (dataset == "US Metros") upper_tab = "metro_long"
await client.query(`DROP TABLE IF EXISTS tmp`)
await client.query(`CREATE TABLE tmp AS
SELECT cases::FLOAT - LAG(cases, 1) OVER
(PARTITION BY 'place' ORDER BY 'date')::FLOAT as new_cases,
*
FROM ${upper_tab}
`)
return client.table(`SELECT * FROM tmp LIMIT 30`)
}
Insert cell
r = fetch(`https://raw.githubusercontent.com/CSSEGISandData/COVID-19/master/csse_covid_19_data/csse_covid_19_time_series/time_series_covid19_confirmed_global.csv`).then(d => d.arrayBuffer())
Insert cell
client.query(`CREATE TABLE
Insert cell
us_raw = fetch(`https://raw.githubusercontent.com/CSSEGISandData/COVID-19/master/csse_covid_19_data/csse_covid_19_time_series/time_series_covid19_confirmed_US.csv`).then(d => d.arrayBuffer())
Insert cell
import { linear_us_state_order } from '@bmschmidt/useful-linear-orders-for-countries-and-states'
Insert cell
import { populations, root_ordering } from '@bmschmidt/well-ordered-coronavirus-heatmaps-for-us-and-the-world'
Insert cell
root_ordering
Insert cell
import { state_info } from "@bmschmidt/state-info"

Insert cell
state_info.get("Alabama")
Insert cell
county_populations =
{
client;
const data = await d3.json('https://raw.githubusercontent.com/Zoooook/CoronavirusTimelapse/master/static/population.json')
for (let datum of data) {
datum.population = +datum.population
datum.fips = +datum.us_county_fips
}
await insert_json("county_populations", data, client)
return
}
Insert cell
state_pops = Object.entries(population.states).map(([state, pop]) => ({name: state, population_total: pop}))
Insert cell
import { population } from "@youbastard/us-state-populations"
Insert cell
[...state_info.entries()][3]
Insert cell
import { duckdb, db } from '@cmudig/duckdb'
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