Public
Edited
Feb 7, 2024
13 stars
Map zooming with slippy tiles and a scale bar
Introducing d3-geo-scale-bar
Also listed in…
Maps
Insert cell
Insert cell
Insert cell
Insert cell
Insert cell
scaleBarBottom = 20
Insert cell
scaleBarRight = 165
Insert cell
scaleBarIndiaTopZoom = d3.geoScaleBar()
.projection(projectionIndia)
.top(1 - scaleBarBottom / mapHeight)
.left(1 - scaleBarRight / mapWidth)
.size(size)
.orient(d3.geoScaleTop)
.label(null)
.tickPadding(2)
.tickSize(0)
.tickFormat((d, i, e) => i === e.length - 1 ? `${d} km.` : d)
.zoomClamp(false);
Insert cell
scaleBarIndiaBottomZoom = d3.geoScaleBar()
.projection(projectionIndia)
.top(1 - scaleBarBottom / mapHeight)
.left(1 - scaleBarRight / mapWidth)
.size(size)
.units(d3.geoScaleMiles)
.label(null)
.tickPadding(2)
.tickSize(0)
.tickFormat((d, i, e) => i === e.length - 1 ? `${d} mi.` : d)
.zoomClamp(false);
Insert cell
function styleBar(bar){
bar.selectAll("rect")
.remove();

bar.select(".domain")
.attr("shape-rendering", "crispEdges");
const text = bar.select(".tick:last-of-type").text();

const ticks = bar.selectAll(".tick");
ticks.select("line")
.attr("y2", text.includes("km") ? -10 : 10)
.style("display", (d, i, e) => i === e.length - 1 ? "block" : "none")
ticks.select("text")
.attr("dy", text.includes("km") ? -3 : 12)
.attr("text-anchor", "start")
.attr("font-size", 14)
.style("user-select", "none")
.style("display", (d, i, e) => i === 0 ? "block" : "none")
.style("paint-order", "stroke fill")
.style("stroke", "white")
.style("stroke-linecap", "round")
.style("stroke-linejoin", "round")
.style("stroke-opacity", .5)
.style("stroke-width", 4);
bar.select(".tick text")
.text(text);
}
Insert cell
Insert cell
// © OpenStreetMap contributors https://www.openstreetmap.org/copyright
url = d => `http://${"abc"[d[1] % 3]}.tile.openstreetmap.org/${d[2]}/${d[0]}/${d[1]}.png`
Insert cell
deltas = [-100, -4, -1, 0]
Insert cell
Insert cell
india = {
const topo = await FileAttachment("india@1.json").json();
return topojson.feature(topo, topo.objects.india);
}
Insert cell
Insert cell
mapWidth = Math.min(width, 600);
Insert cell
mapHeight = mapWidth * 1.13;
Insert cell
size = [mapWidth, mapHeight];
Insert cell
Insert cell
projectionIndia = d3.geoMercator().fitSize(size, india);
Insert cell
Insert cell
d3 = require("d3@7", "d3-geo-scale-bar@1.2.5", "d3-tile@1");
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