Published
Edited
Dec 19, 2020
Importers
6 stars
SQLite Geospatial - intersections!London Isodemographic CartogramGolden Tonal Palette AnalysisHexbin Transform / Observable Plot
NBA Finals, Game 4
Disk Sampling on a SphereNBA Finals, Game 1Build your first scatterplot with Observable PlotDisplaying IBM Carbon charts in ObservableModelo de Cúpula CatenáriaBivariate Bubble MapAnalyzing Star Wars movies + the first anniversary of Observable Plot!Single-family homes owned by large corporate landlords in North CarolinaThe MoMA Collection DataMapping the Vietnamese DiasporaElection map as striped circles #2They heard the callDiagrammatic EmbeddingsHegel's Complete System of PhilosophyGeneration Arc DiagramVulnerability of Mountain PeoplesVisualizing Air Raid Sirens in UkraineUkrainian refugees welcomeConcentric Circles in D3Daylight Saving Time Gripe Assistant ToolPerceptually uniform color models and their implications
Top Notebooks of February 2022Zillow Home PricesBlack History Month ActivityExamples from: Three Simple Flexible tools for Empowered Data VisualizationFull Covid Vaccination Rate in CA by County Over TimePlot CheatsheetsHunga Tonga–Hunga Ha'apai Global Pressure WaveVisualizing The New York Times’s Mini Crossword100 Beautiful and Informative Notebooks of 2021When Presidents Fade AwayThe Most Frequently Used Emoji of 2021Upset Plots with Observable PlotLucy mission animationSpreadsheet-based data collaboration gets supercharged with Observable30 Day Map Challenge: Day 4 - HexagonsDisputed territoriesMapping the Cities of US Highway SignsThirty day Map Challenge: Day 1 - PointsCube Pushing Loop 💪 🧊 🔄Guided Tour of an Infinite Sphere GridCanvas, P5.js and circle packing (collision and cluster force) on a mapWomen's History Month DataViz ContestWomen in DataViz EventSVG flowersMultiple Set Venn with Color BlendingA Stupid Emoji Utility FunctionQuilting with d3-curveCurvilinear RosettesStrange AttractorsSolo board gameDouble pendilumVertical SlidersDrawing and animating blobs with svg and javascriptOrbiting Particles III#genuary 2021 ~ Do some golf!Imitation Game 🖐Happy New Year (2019) (And other celebrations)Earth Temperature SpiralStar MapCorrelation FilteringBattle of Colors
Papercraft Christmas Ornaments Generator
Christmas Tree Perspective 🎄bouncing circles2D (Non-physical) N-body Gravity with Poisson's EquationWaterfall Fill2020 TimelineTime Spiral with a COVID DemoTransition between Three.js camerasJulio Le Parc Replications and VariationsFlexible HeatmapSquare PackingSierpiński curve animationTrainsWhy use a radial data visualization?Earthquakes from the past 30 daysThe US COVID SyringeHello OGL - Minimal WebGL libraryVersor MultiplicationLabyrinthTruncated-octahedron-based 3D-space-filling curveNUTS regions (The Nomenclature of territorial units for statistics): PerspectiveStretchy FishTP3: Power Diagram and Semi-Discrete Optimal TransportEmoji ParticlesFirma de color de BogotáCovid19 WorldwideWeb 3.0 Explorable # 3: NFTs -- Game Items With Real World ValueUnemployment ExplorationSierpinski curveEight QueensVoronoi StipplingSpinning out of controlFragment shader raytracerSunny day, rainy day in SeattleAs votes are countedOrbit of the dayVoronoi ClothTry to impeach this? Challenge accepted!Dispersion in Water Surface WavesSelf-Organizing Maps meet DelaunayGenerative ArtReturn to a squareCreating a Canvas Noise Map for Generative ArtMARTINI: Real-Time RTIN Terrain MeshMunsell spinGenerative artAs votes come in, what would it take for the trailing candidate to win?3a. Historical participation in early voting vs. Election Day votingWaffle ChartSun settingRaymarch HelperF1 Constructor Standings 2010-2019Electoral College Decision TreeThe Woman Citizen's Wheel of Progress, 1917Equisurface bull's eye
Insert cell
Insert cell
Insert cell
viewof sides = html`<input type=range min=4 max=12 step=1 value=5>`
Insert cell
viewof segmentCount = html`<input type=range min=4 max=12 step=1 value=5>`
Insert cell
viewof jointSegment = html`<input type=range min=1 max=${segmentCount} step=1 value=3>`
Insert cell
noTabSegments = [0, 3, 4]
Insert cell
Insert cell
Insert cell
bulbOutline = {
const n = segmentCount
const r = 115 // Radius of the Ornament
const o = r + hangerLength // 50 is how tall the hanger tab is
const t = 0.13 // The initial segment's angle
const main = d3.range(n).map(i => {
const theta = Math.PI * (0.5 + t) + i * (Math.PI - t) / (n)
return {
theta,
z: o - Math.sin(theta) * r,
w: -Math.cos(theta) * r
}
})
return [{ z: 0, w: 0}, { z: hangerLength, w: 12 }].concat(main).concat([{ z: o + r, w: 0 }])
}
Insert cell
Insert cell
Insert cell
Insert cell
sides
Insert cell
Insert cell
segments = d3.pairs(bulbOutline).map(p => {
return {
points: p,
length: Math.hypot(p[0].z - p[1].z, p[0].w - p[1].w) // Math.hypot is just the pythagoras formula
}
})
Insert cell
Insert cell
Insert cell
anglesBetweenSides = 2 * Math.PI / sides
Insert cell
Insert cell
horizontalEdgeCalc = (hypot) => {
return Math.sin(anglesBetweenSides / 2) * hypot * 2
}
Insert cell
horizontalEdgeLengths = bulbOutline.map((p) => {
return horizontalEdgeCalc(p.w)
})
Insert cell
horizontalEdgePairs = d3.pairs(horizontalEdgeLengths)
Insert cell
Insert cell
md`The red lines there represent the horizontal edges.

We now have the lengths of both horizontal and the vertical edges of facets on the ornament. We know that the horizontal edges should be parallel to each other, and that the facets should be symetrical. These constraints will allow us to again use trigonometry compute how tall each facet is.`
Insert cell
facetHeightCalc = (top, bottom, side) => {
const diff = Math.abs(top - bottom)
const theta = Math.asin(diff / 2 / side)
return Math.cos(theta) * side
}
Insert cell
facets = segments.reduce((memo, s, i) => {
const h = horizontalEdgePairs[i]
const facetHeight = facetHeightCalc(h[0], h[1], s.length)
memo.push({
points: s.points,
side: s.length,
top: h[0],
bottom: h[1],
height: facetHeight,
culmulativeHeight: i === 0 ? facetHeight : memo[i - 1].culmulativeHeight + facetHeight
})
return memo
}, [])
Insert cell
Insert cell
stretchedOutHeight = facets.reduce((memo, d) => {
return memo + d.height
}, 0)
Insert cell
midlineOffset = d3.max(horizontalEdgeLengths) / 2
Insert cell
pointPairsOnASide = (facets) => {
const lines = []
facets.forEach((f, i) => {
if (i > 0) {
// Regular Facet
lines.push({
type: "cut",
isExterior: true,
side: "right",
segment: i,
x1: f.top / 2,
y1: f.culmulativeHeight - f.height,
x2: f.bottom / 2,
y2: f.culmulativeHeight
})

lines.push({
type: "cut",
isExterior: true,
side: "left",
segment: i,
x1: -f.top / 2,
y1: f.culmulativeHeight - f.height,
x2: -f.bottom / 2,
y2: f.culmulativeHeight
})

// the Fold Line
lines.push({
type: "fold",
isExterior: false,
segment: i,
x1: +f.bottom / 2,
x2: -f.bottom / 2,
y1: f.culmulativeHeight,
y2: f.culmulativeHeight
})
} else {
lines.push({
type: "skip",
isExterior: true,
side: "right",
segment: i,
x1: f.top / 2,
y1: f.culmulativeHeight - f.height,
x2: f.bottom / 2,
y2: f.culmulativeHeight
})

lines.push({
type: "skip",
isExterior: true,
side: "left",
segment: i,
x1: -f.top / 2,
y1: f.culmulativeHeight - f.height,
x2: -f.bottom / 2,
y2: f.culmulativeHeight
})
// the Hanger Fold Line
lines.push({
type: "fold",
isExterior: false,
segment: i,
x1: +f.bottom / 2,
x2: -f.bottom / 2,
y1: f.culmulativeHeight,
y2: f.culmulativeHeight
})
// hanger
lines.push({
type: "hanger",
isExterior: true,
segment: i,
x1: +f.bottom / 2,
x2: -f.bottom / 2,
y1: f.culmulativeHeight,
y2: f.culmulativeHeight
})
}
})
return lines
}
Insert cell
Insert cell
Insert cell
jointSegment
Insert cell
rotationRequired = {
// facets[jointSegment]
const v = {
x: facets[jointSegment].top / 2,
y: facets[jointSegment].culmulativeHeight - facets[jointSegment].height
}
const v2 = {
x: facets[jointSegment].bottom / 2,
y: facets[jointSegment].culmulativeHeight
}
const diff = {
x: v2.x - v.x,
y: v2.y - v.y
}
const v2theta = Math.atan2(v2.x, v2.y)
const theta = Math.PI / 2 - 2 * Math.atan2(diff.x, diff.y) + v2theta
const length = Math.hypot(v2.x, v2.y)
const translation = {
x: v2.x - Math.cos(theta) * length,
y: v2.y - Math.sin(theta) * length
}
return {
f: facets[jointSegment],
v,
v2,
length,
rotation: -2 * Math.atan2(diff.x, diff.y),
translation
}
}
Insert cell
Insert cell
Insert cell
linesToDraw = {
const basicPointsPairs = pointPairsOnASide(facets).map((pair) => {
return {
...pair,
face: 0
}
})
let results = []
const extent = {
x: [Infinity, -Infinity],
y: [Infinity, -Infinity]
}
d3.range(sides).reduce((memo, i) => {
const transform = (pair) => {
const v1 = rotateVector({ x: pair.x1, y: pair.y1 }, rotationRequired.rotation)
const v2 = rotateVector({ x: pair.x2, y: pair.y2 }, rotationRequired.rotation)
const x1 = v1.x + rotationRequired.translation.x
const y1 = v1.y + rotationRequired.translation.y
const x2 = v2.x + rotationRequired.translation.x
const y2 = v2.y + rotationRequired.translation.y
if (x1 < extent.x[0] || x2 < extent.x[0]) { extent.x[0] = Math.min(x1, x2) }
if (x1 > extent.x[1] || x2 > extent.x[1]) { extent.x[1] = Math.max(x1, x2) }
if (y1 < extent.y[0] || y2 < extent.y[0]) { extent.y[0] = Math.min(y1, y2) }
if (y1 > extent.y[1] || y2 > extent.y[1]) { extent.y[1] = Math.max(y1, y2) }
return {
...pair,
x1,
y1,
x2,
y2,
face: i,
}
}
const linesGivenSide = (i === 0) ? memo : memo.map(transform)
results = results.concat(linesGivenSide)
return linesGivenSide
}, basicPointsPairs)
const xShift = (extent.x[0] < 0) ? Math.abs(extent.x[0]) : 0
const yShift = (extent.y[0] < 0) ? Math.abs(extent.y[0]) : 0
return results.map((pair) => {
const switchFromCutToFold = (pair.segment === jointSegment)
&& !(pair.face === 0 && pair.side === "left")
&& !(pair.face === sides - 1 && pair.side === "right")
return {
...pair,
x1: pair.x1 + midlineOffset + 22 + xShift,
x2: pair.x2 + midlineOffset + 22 + xShift,
y1: pair.y1 + 22 + yShift,
y2: pair.y2 + 22 + yShift,
type: switchFromCutToFold ? "fold" : pair.type,
tabs: !noTabSegments.includes(pair.segment)
}
})
}
Insert cell
Insert cell
drawingExtent = computeDrawingExtent(linesToDraw)
Insert cell
Insert cell
Insert cell
noTabSegments
Insert cell
md`With the use of tabs, I now have to distinguish between types of lines, and the "renderer" for the types of lines. The following is a bunch of line type implementations.

Starting with a "regular" line.`
Insert cell
regularLine = (svg, line) => {
const theta = Math.atan2(line.y1 - line.y2, line.x1 - line.x2)
// Overcut to ensure separation
const o1 = {
x: line.x1 + Math.cos(theta) * 1,
y: line.y1 + Math.sin(theta) * 1
}
const o2 = {
x: line.x2 - Math.cos(theta) * 1,
y: line.y2 - Math.sin(theta) * 1
}
svg.append("path")
.attr("d", `M ${o1.x} ${o1.y} L ${o2.x} ${o2.y}`)
.attr("stroke", "black")
}
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
md`After a while it ocurred to me that I needed a way to hang them.

To accomodate a variety of widths that the attachment might come at, I wrote a bit of code that adjusted the tab design according to the width. Just a bit of tangent calculations.`
Insert cell
Insert cell
hangerLength = 50
Insert cell
Insert cell
{
const svg = d3.create("svg")
.attr("viewBox", [0, 0, drawingExtent.x[1] + 10, drawingExtent.y[1] + 10]);
const cutLayer = svg.append("g")
const scoreLayer = svg.append("g")
linesToDraw.forEach((p) => {
switch (p.type) {
case "hanger":
hangerLine(cutLayer, p)
break;
case "fold":
foldCreaseLine(scoreLayer, p)
break;
default:
if (p.tabs) {
latchAttachmentLine(cutLayer, p)
} else {
regularLine(cutLayer, p)
}
}
})
return svg.node()
}
Insert cell
Insert cell
Insert cell
notchCreaseLine = (svg, line) => {
const theta = Math.atan2(line.y1 - line.y2, line.x1 - line.x2)
const length = Math.hypot(line.y1 - line.y2, line.x1 - line.x2)
const s = {
x: line.x1 - 5 * Math.cos(theta),
y: line.y1 - 5 * Math.sin(theta)
}
const e = {
x: line.x2 + 5 * Math.cos(theta),
y: line.y2 + 5 * Math.sin(theta)
}
svg.append("path")
.attr("d", `M ${line.x1} ${line.y1} L ${s.x} ${s.y}`)
.attr("stroke", "black")
svg.append("path")
.attr("d", `M ${line.x2} ${line.y2} L ${e.x} ${e.y}`)
.attr("stroke", "black")
}
Insert cell
foldCreaseLine = (svg, line) => {
svg.append("path")
.attr("d", `M ${line.x1} ${line.y1} L ${line.x2} ${line.y2}`)
.attr("stroke", "blue")
}
Insert cell
foldCutLine = (svg, line) => {
const theta = Math.atan2(line.y1 - line.y2, line.x1 - line.x2)
const length = Math.hypot(line.y1 - line.y2, line.x1 - line.x2)
const lengthWithPadding = length - 6
const unit = 12
const counts = Math.round(lengthWithPadding / unit)
const remainder = length - counts * unit
d3.range(counts).forEach(i => {
const p = {
x: line.x1 - Math.cos(theta) * (i * unit + remainder + 3),
y: line.y1 - Math.sin(theta) * (i * unit + remainder + 3)
}
const p2 = {
x: line.x1 - Math.cos(theta) * (i * unit + remainder - 3),
y: line.y1 - Math.sin(theta) * (i * unit + remainder - 3)
}
svg.append("path")
.attr("d", `M ${p.x} ${p.y} L ${p2.x} ${p2.y}`)
.attr("stroke", "black")
})
}
Insert cell
md`Below here are utils and things`
Insert cell
computeDrawingExtent = (linesToDraw) => {
const results = {
x: [Infinity, -Infinity],
y: [Infinity, -Infinity]
}
const linesExtents = linesToDraw.map(l => {
if (l.x1 < results.x[0] || l.x2 < results.x[0]) { results.x[0] = Math.min(l.x1, l.x2) }
if (l.x1 > results.x[1] || l.x2 > results.x[1]) { results.x[1] = Math.max(l.x1, l.x2) }
if (l.y1 < results.y[0] || l.y2 < results.y[0]) { results.y[0] = Math.min(l.y1, l.y2) }
if (l.y1 > results.y[1] || l.y2 > results.y[1]) { results.y[1] = Math.max(l.y1, l.y2) }
})
return results
}
Insert cell
rotateVector = (vector, theta) => {
const { x, y } = vector
// first column of the matrix
const xa = x * Math.cos(theta)
const xb = x * Math.sin(theta)
// second column of the matrix
const yc = y * -Math.sin(theta)
const yd = y * Math.cos(theta)
// sum the components
return {
x: xa + yc,
y: xb + yd
}
}
Insert cell
maxWOffset = d3.max(bulbOutline, d => d.w)
Insert cell
height = bulbOutline[bulbOutline.length - 1].z
Insert cell
d3 = require("d3@6")
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