If your chart contains items sized according to a data variable (such as a bubble chart) hovering over (or clicking) tiny items can be difficult.
Try hovering over the tiny circles:
You can make picking small items easier by searching for the closest item to the mouse pointer each time the mouse is moved. This can be an expensive operation but can be made more efficient using D3's quadtree module.
A quadtree is a tree data structure that recursively divides an area into smaller and smaller areas and can make searching for items more efficient.
With D3's quadtree function (from the d3-quadtree module) you can create a quadtree, add some points to it, then find the closest point to a given coordinate.
Create a quadtree by calling quadtree() then add points to it using .add:
let quadtree =quadtree();
quadtree.add([50,100]); quadtree.add([100,100]);
In the above example two points have been added: 50, 100 and 100, 100.
Given a coodinate x, y you can find the nearest point in the quadtree using .find(x, y):
The first .find returns [50, 100] because this is the closest point and is within a distance of 20. The second .find returns undefined because the closest point [50, 100] is more than 5 away. This is useful for ensuring the returned point is close to the requested point. Without this constraint, outlier points can get selected even though the pointer isn't very close.
You can add an array of points using .addAll:
quadtree.addAll([[10,50],[60,30],[80,20]]);
The .add and .addAll methods are cumulative i.e. the existing quadtree points persist.
If you've an array of objects, you can specify accessor functions using the .x and .y methods:
let data =[ {x:50,y:100}, {x:100,y:100} ];
let quadtree =quadtree() .x(function(d){return d.x;}) .y(function(d){return d.y;});
quadtree.addAll(data);
quadtree.find(60,100);// returns {x: 50, y: 100}
Example
Let's look at an example where we create some randomised points (updateData) and add them to a quadtree (updateQuadtree). We draw the points (update) and set up a mousemove event on the svg element (initEvents).
When the mouse moves (see handleMousemove) we search for the nearest point to the mouse pointer using the quadtree. We update hoveredId with the found point's id then call update again so that the hovered point is coloured red:
let data =[], width =600, height =400, numPoints =100;
let quadtree =quadtree() .x(function(d){return d.x;}) .y(function(d){return d.y;});
In handleMousemove we use pointer which given the event object e and an HTML/SVG element, returns the mouse position relative to the HTML/SVG element. In our example, this is the SVG element because this is the element .on was called on (see initEvents). pointer is imported from the d3-selection module.
Now your mouse pointer only needs to be within 20 pixels of its nearest circle:
D3's quadtree is also used by forceCollide when detecting collisions in the force layout.
\n\n
\n \n
\n","css":"circle {\n\topacity: 0.5;\n}\n","js":"// For bundlers such as Vite and Webpack omit https://esm.sh/\nimport { select, pointer } from 'https://esm.sh/d3-selection';\nimport { quadtree } from 'https://esm.sh/d3-quadtree';\n\nlet data = [], width = 600, height = 400, numPoints = 100;\nlet triangles;\nlet hoveredId;\n\nfunction updateData() {\n\tdata = [];\n\tfor(let i=0; i
-->
Dragging
D3 has a module d3-drag for adding drag behaviour to elements. Dragging is where you hover over an element, press the mouse button, move the pointer, then release the mouse button, in order to move the element. D3's drag module also supports touch gestures.
There's three steps to making HTML/SVG elements draggable:
call drag() to create a drag behaviour function
add an event handler that's called when a drag event occurs. The event handler receives an event object with which you can update the position of the dragged element
attach the drag behaviour to the elements you want to make draggable
Calling drag() creates a drag behaviour:
let drag =drag();
A drag behaviour is a function that adds event listeners to elements. It also has methods such as .on defined on it.
You can attach an event handler to the drag behaviour by calling the .on method. This accepts two arguments:
the event type ('drag', 'start' or 'end')
the name of your event handler function
functionhandleDrag(e){ // update the dragged element with its new position }
let drag =drag() .on('drag', handleDrag);
The event types are 'drag', 'start' and 'end'. 'drag' indicates a drag. 'start' indicates the start of the drag (e.g. the user has pressed the mouse button). 'end' indicates the end of the drag (e.g. the user has released the mouse button).
handleDrag receives a single parameter e which is an object representing the drag event. The drag event object has several properties, the most useful of which are:
Property name
Description
.subject
The joined data of the dragged element (or a fallback object)
.x & .y
The new coordinates of the dragged element
.dx & .dy
The new coordinates of the dragged element, relative to the previous coordinates
If the dragged element was created by a data join and the joined data has x and y properties, the x and y properties of the drag event object are computed such that the relative position of the element and pointer are maintained. (This prevents the element's center 'snapping' to the pointer position.) Otherwise x and y are the pointer position relative to the dragged element's parent element.
You attach the drag behaviour to elements by selecting the elements and passing the drag behaviour into the .call method.
For example to add drag behaviour to circle elements:
select('svg') .selectAll('circle') .call(drag);
The drag behaviour is a function that sets up event listeners on the selected elements (each circle element in the above example). When drag events occur the event handler (handleDrag in the above examples) is called.
Example
In the following code an array of random coordinates is joined to circle elements (updateData and update).
A drag behaviour is created using drag() and attached to the circle elements (initDrag).
When a circle is dragged, handleDrag gets called and an event object e is passed in as the first argument. e.subject represents the joined data of the dragged element. The x and y properties of the joined data are updated to e.x and e.y. update is then called to update the circle positions.
let data =[], width =600, height =400, numPoints =10;
Notice that the relative position of the pointer and dragged circle is maintained (try initiating a drag near the edge of a circle):
Brushing
Brushing lets you user specify an area (by pressing the mouse button, moving the mouse, then releasing) in order to, for example, select a group of elements.
Try selecting circles by pressing the mouse button, dragging, then releasing the button:
D3 has a module d3-brush for adding brushing behaviour to an element (or, less commonly, multiple elements).
There's three steps to adding brush behaviour to an HTML or SVG element:
call brush() to create a brush behaviour function
add an event handler that's called when a brush event occurs. The event handler receives the brush extent which can then be used to select elements, define a zoom area etc.
attach the brush behaviour to an element (or elements)
Calling brush() creates a brush behaviour:
let brush =brush();
A brush behaviour is a function that has methods such as .on defined on it. The function itself adds event listeners to an element as well as additional elements (mainly rect elements) for rendering the brush extent.
You can attach an event handler to a brush behaviour by calling the .on method. This accepts two arguments:
the event type ('brush', 'start' or 'end')
the name of your event handler function
functionhandleBrush(e){ // get the brush extent and use it to, for example, select elements }
let brush =brush() .on('brush', handleBrush);
The event types are 'brush', 'start' and 'end'. 'brush' indicates that the brush extent has changed. 'start' indicates the brushing has started (e.g. the user has pressed the mouse button). 'end' indicates the end of brushing (e.g. the user has released the mouse button).
handleBrush receives a single parameter e which is an object representing the brush event. The most useful property on the brush event is .selection which represents the extent of the brush as an array [[x0, y0], [x1, y1]] where x0, y0 and x1, y1 are the opposite corners of the brush. Typically handleBrush will compute which elements are within the brush extent and update them accordingly.
You attach the brush behaviour to an element by selecting the element and passing the brush behaviour into the .call method:
select('svg') .call(brush);
Examples
Basic example
In the following example a brush is created using brush(). An event handler handleBrush is added to the brush behaviour using the .on method.
handleBrush gets called whenever brushing starts (the 'start' event type) or the brush extent changes (the 'brush' event type).
The brush behaviour is attached to the svg element by calling .call and passing in the brush behaviour (see initBrush).
let brush =brush() .on('start brush', handleBrush);
functionhandleBrush(e){ // Use the brush extent e.selection to compute, for example, which elements to select }
Here's a complete example where an array of random coordinates is joined to circle elements (updateData and update). When the brush is active, the circles within the brush extent are coloured red.
The brush is initialised in initBrush. (Note that it's attached to a g element within the svg element, in order to keep the elements used to render the brush separate to the circles.)
When brushing occurs, handleBrush is called. This receives a brush event object e which has a property selection that defines the extent of the brush. This is saved to the variable brushExtent and update is called.
update performs the data join and colours circles red if they're within the extent defined by brushExtent (see isInBrushExtent):
let data =[], width =600, height =400, numPoints =100;
let brush =brush() .on('start brush', handleBrush);
D3 also provides brushes brushX and brushY that constrain the brush to a single dimension.
They work in a simlar fashion to brush, the main difference being the event object's .selection property which is an array of two numbers [min, max] which represent the extent of the brush.
Here's an example using brushX:
Programmatic control of brushing
You can also set the brush extent programmatically. For example you can create a button that sets the brush to maximum size.
The brush behaviour has two methods for setting the brush extent .move and .clear. The first sets the brush extent to [[x0, y0], [x1, y1]] and the second clears the brush.
.move and .clear should be called on the element that receives the brush gestures. For example: