Can you adjust the speed of the zoom when the user scrolls in and out using the mousewheel?
My understanding is that the zoom.on (https://github.com/mbostock/d3/wiki/Zoom-Behavior#wiki-on) listener produces the two events d3.event.translate & d3.event.zoom, which contain matrices or coordinates that when passed to the translate or scale functions, allow panning and rescaling of the graphic.
But how do I speed this up, so that if the user moves his mousewheel by a little, she rapidly zooms in or out? I have a large visualization that I want to allow users to zoom in and out of rapidly with the mousewheel. Can I simply modify/add arguments to the above existing events and functions or do I have to create my own? I have a feeling some of the above is inaccurate/patchy in terms of understanding, so please explain if so.
Very simple jsfiddle example here: http://jsfiddle.net/fiddler86/6jJe6/, with identical code below:
var svg = d3.select("body").append("svg:svg")
.attr("width", 1000)
.attr("height", 2000)
.append("svg:g")
.call(d3.behavior.zoom().on("zoom", redraw))
.append("svg:g");
svg.append("svg:rect")
.attr("width", 200)
.attr("height", 300)
.attr("fill", 'green');
function redraw() {
svg.attr("transform", "translate(" + d3.event.translate + ")" + " scale(" + d3.event.scale + ")");
};
You need adjust the scale inside the function with a mathematical function when you select the function the important is that for x=0 the y=0 you can use pow is easier in this case
Math.pow(d3.event.scale,.1)
the second parameter does the zoom more slowly when is smaller.It´s not a good idea use a very complicated function because the browser will turn slow.
When you have the new scale, you need recalculate the translation. You don´t complicate the problem, in SVG you have the actual height with
this.getBBox().height
this ok, but it is not exactly because you are one iteration behind. You could calculate the new height with(originalHeight * scale)
and the translate with(originalHeight - (originalHeight * scale))/2
Well origialHeight*scale is the newHeight
The originalHeight - newHeight is the difference, and you want the center, you need divide for 2, the half part of the square and the half part below.
Now we need do the action with the width. It is the same
The code:
Note that I put the 200 and 300 hardcoded, you can use a property, use constant...
I created a fiddler: http://jsfiddle.net/t0j5b3e2/
I've modified mbostock's drag+zoom example to have a 4x zoom speed and put it in a jsfiddle. I've explained my thinking below. This is my first attempt at a stack overflow answer, please be nice.
As explained in Raúl Martín's answer you can use a formula within the
redraw()
function to change the rate of zoom. You need to make some extra steps to make sure that the d3 behaviour still works nicely with the modified zoom rate.Zoom centered on a chosen point (e.g. cursor)
By default d3 behaviour focuses the zoom on the mouse pointer, e.g. if you have the mouse pointer at the top left of the image it zooms in on the top left rather than the center of the image. To get this effect it scales the image and then also changes the translation of the image so that the point under the mouse cursor stays at the same location on the screen. This is why the value of
zoom.translate()
changes when you scroll the mousewheel even though the image doesn't look like it is moving across the screen.If you change the zoom speed the d3
zoom.translate()
values are no longer correct. To work out the correct translation you need to know the following information (ignore the numeric values):The formula to work out the
new_translate
to apply to the image is then:You can apply this to the image along with your new scale with:
You'll then have to update
prev_scale = new_scale
andprev_translate = new_translate
ready for the next iteration ofredraw()
Pan without scaling
The d3 zoom behaviour allows you to pan without scaling by clicking and dragging. If you click and drag then
zoom.scale()
stays the same butzoom.translate()
changes. The newzoom.translate()
value is still correct even after you have modified the zoom speed. However, you need to know when to use thiszoom.translate()
value and when to use the translate value that you calculate for zooming in on a center point.You can work out whether a pan or a zoom is happening by looking at whether
prev_scale
is the same asnew scale
. If the two values are identical you know a pan is taking place and you can usenew_translate = zoom.translate()
to move the image. Otherwise, you know that a zoom is taking place and you can calculate thenew_translate
value as described above. I do this by adding a function to thezoomstart
event.