Skip to content

Instantly share code, notes, and snippets.

Last active August 5, 2019 23:43
Show Gist options
  • Save mbostock/9656675 to your computer and use it in GitHub Desktop.
Save mbostock/9656675 to your computer and use it in GitHub Desktop.
Zoom to Bounding Box II
license: gpl-3.0

A variant of the Zoom to Bounding Box example that uses zoom transitions to smoothly interpolate between different views. This example also allows you to freely pan and zoom with the mouse (or touch).

<!DOCTYPE html>
<meta charset="utf-8">
.background {
fill: none;
pointer-events: all;
.feature {
fill: #ccc;
cursor: pointer;
} {
fill: orange;
.mesh {
fill: none;
stroke: #fff;
stroke-linecap: round;
stroke-linejoin: round;
<script src="//"></script>
<script src="//"></script>
var width = 960,
height = 500,
active =;
var projection = d3.geo.albersUsa()
.translate([width / 2, height / 2]);
var zoom = d3.behavior.zoom()
.translate([0, 0])
.scaleExtent([1, 8])
.on("zoom", zoomed);
var path = d3.geo.path()
var svg ="body").append("svg")
.attr("width", width)
.attr("height", height)
.on("click", stopped, true);
.attr("class", "background")
.attr("width", width)
.attr("height", height)
.on("click", reset);
var g = svg.append("g");
.call(zoom) // delete this line to disable free zooming
d3.json("/mbostock/raw/4090846/us.json", function(error, us) {
if (error) throw error;
.data(topojson.feature(us, us.objects.states).features)
.attr("d", path)
.attr("class", "feature")
.on("click", clicked);
.datum(topojson.mesh(us, us.objects.states, function(a, b) { return a !== b; }))
.attr("class", "mesh")
.attr("d", path);
function clicked(d) {
if (active.node() === this) return reset();
active.classed("active", false);
active ="active", true);
var bounds = path.bounds(d),
dx = bounds[1][0] - bounds[0][0],
dy = bounds[1][1] - bounds[0][1],
x = (bounds[0][0] + bounds[1][0]) / 2,
y = (bounds[0][1] + bounds[1][1]) / 2,
scale = Math.max(1, Math.min(8, 0.9 / Math.max(dx / width, dy / height))),
translate = [width / 2 - scale * x, height / 2 - scale * y];
function reset() {
active.classed("active", false);
active =;
.call(zoom.translate([0, 0]).scale(1).event);
function zoomed() {"stroke-width", 1.5 / d3.event.scale + "px");
g.attr("transform", "translate(" + d3.event.translate + ")scale(" + d3.event.scale + ")");
// If the drag behavior prevents the default click,
// also stop propagation so we don’t click-to-zoom.
function stopped() {
if (d3.event.defaultPrevented) d3.event.stopPropagation();
Copy link

Hello, thank you for your great work. There seems to be one small problem: clicking New England states sends the map flying off screen. This doesn't happen for your Bounding Box 1 example:

Copy link

As a newbie to D3, I'm loving this work. Any chance this will be updated with the new Zoom behavior in D3 V4?

Copy link

Also a d3 newcomer, and trying to digest what i might be doing wrong when trying to use the same sort of zoom transition to a bounding box under the v4 codebase.

Copy link

iamkevinv commented Oct 7, 2016

@bhrutledge / @mbostock - I've gone ahead and made a working fork to using d3 v4 here instead of the v3 example.
I'm still new to this, so i don't claim to definitively know this to be perfect, but it appears to be 100% ported over. Revisions can be reviewed here:

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment