An interactive React component for visualizing data as positioned dots with zoom, pan, and hover interactions. Extracted from the cybird visualization project.
- Interactive Dots: Display data points as SVG circles with customizable size, color, and stroke
- Zoom & Pan: Smooth zoom with mouse wheel + ctrl/cmd or trackpad pinch, pan with mouse wheel or trackpad
- Hover Interactions: Customizable hover callbacks with automatic debouncing during zoom operations
- Click Interactions: Handle dot clicks with custom callbacks
- Collision Detection: Optional D3 force simulation to prevent dot overlap
- Automatic Layout: Calculates optimal viewBox from data bounds with configurable margins
- Performance Optimized: Efficient rendering and interaction handling for large datasets
npm install react-dot-visualizationimport React, { useState } from 'react';
import { DotVisualization } from 'react-dot-visualization';
const MyComponent = () => {
// Just x, y coordinates - that's it!
const data = [
{ x: 100, y: 150 },
{ x: 200, y: 100 },
{ x: 150, y: 200 }
];
const [hovered, setHovered] = useState(null);
return (
<div style={{ width: '100%', height: '400px' }}>
<DotVisualization
data={data}
onHover={setHovered}
/>
{hovered && <div>Hovering: {hovered.name}</div>}
</div>
);
};That's literally all the code you need!
The component automatically provides:
- ✅ Zoom: Ctrl/Cmd + mouse wheel or trackpad pinch
- ✅ Pan: Mouse wheel or trackpad scroll
- ✅ Hover callbacks: Work during pan/zoom
- ✅ Collision detection: Prevents dot overlap
- ✅ Auto-generated IDs: No manual ID management
- ✅ Optimal layout: Calculates viewBox from data bounds
- ✅ Beautiful colors: Generated automatically
# Clone and install dependencies
git clone <your-repo>
cd react-dot-visualization
npm install
# Start development server for testing
npm run dev
# Opens http://localhost:3011 with demo
# Build library for distribution
npm run build:lib
# Link for local development in other projects
npm run link:localAfter running npm run link:local, you can use the library in other React projects:
# In your other project
cd ../my-other-project
npm link react-dot-visualizationThen import and use normally:
import { DotVisualization } from 'react-dot-visualization';- Edit source files in
src/ - Test changes with
npm run dev - Rebuild library with
npm run build:lib - Linked projects automatically get updates
To test the built package:
# Navigate to the package directory
cd react-dot-visualization
# Start development server
npm run dev
# Open in browser
open http://localhost:3011Test the interactions:
- Hover: Move mouse over dots to see hover callbacks
- Zoom: Ctrl/Cmd + mouse wheel or trackpad pinch
- Pan: Mouse wheel or trackpad scroll
- Click: Click dots to test click callbacks
| Prop | Type | Default | Description |
|---|---|---|---|
data |
Array |
[] |
Array of data points with {x, y} required, optional {id, size, color, ...customData} |
onHover |
Function |
- | Callback when hovering over a dot: (item, event) => {} |
onLeave |
Function |
- | Callback when leaving a dot: (item, event) => {} |
onClick |
Function |
- | Callback when clicking a dot: (item, event) => {} |
onZoomStart |
Function |
- | Callback when zoom starts: (event) => {} |
onZoomEnd |
Function |
- | Callback when zoom ends: (event) => {} |
enableCollisionDetection |
Boolean |
true |
Enable D3 force simulation to prevent dot overlap |
zoomExtent |
Array |
[0.7, 10] |
Min/max zoom levels [min, max] |
margin |
Number |
0.1 |
Margin around data bounds as fraction (0.1 = 10% margin) |
dotStroke |
String |
"#111" |
Default stroke color for dots |
dotStrokeWidth |
Number |
0.2 |
Default stroke width for dots |
defaultColor |
String |
null |
Default color for dots without color property |
defaultSize |
Number |
2 |
Default size for dots without size property |
useImages |
Boolean |
false |
Enable image patterns inside dots (requires imageUrl or svgContent in data) |
imageProvider |
Function |
- | Function to provide image URLs: (id) => string | undefined |
hoverImageProvider |
Function |
- | Function to provide hover image URLs: (id) => string | undefined |
className |
String |
"" |
CSS class name for the SVG element |
style |
Object |
{} |
Inline styles for the SVG element |
Each data point should be an object with these properties:
{
id: string | number, // Required: Unique identifier
x: number, // Required: X coordinate
y: number, // Required: Y coordinate
size?: number, // Optional: Dot radius
color?: string, // Optional: Fill color (CSS color value)
imageUrl?: string, // Optional: URL to bitmap image (JPG, PNG, etc.)
svgContent?: string, // Optional: Raw SVG content for pattern
...customData // Optional: Any additional properties for your callbacks
}You can display images inside the circular dots using two approaches:
Use the imageUrl property to display bitmap images (JPG, PNG, etc.):
const data = [
{
id: 1,
x: 100, y: 150,
imageUrl: "/path/to/album-cover.jpg" // Local or remote image
},
{
id: 2,
x: 200, y: 100,
imageUrl: "https://example.com/photo.png" // Remote image
}
];Use the svgContent property to embed raw SVG:
import * as jdenticon from 'jdenticon';
const data = [
{
id: 1,
x: 100, y: 150,
svgContent: jdenticon.toSvg('user1', 64) // Generated identicon
},
{
id: 2,
x: 200, y: 100,
svgContent: '<svg xmlns="...">...</svg>' // Custom SVG
}
];To show images in dots, pass the useImages prop:
<DotVisualization
data={data}
useImages={true} // Enable image patterns
defaultSize={15} // Larger dots show images better
/>- Automatic scaling: Images automatically resize to match each dot's size (10px dot = 10px image, 50px dot = 50px image)
- Smart cropping: Images are centered and cropped to fill the entire circle (like CSS
background-size: cover) - Aspect ratio preserved: Images maintain their proportions while filling the circle completely
- Zoom responsive: Images scale smoothly with zoom level - no pixelation or distortion
- Circular masking: Images are automatically clipped to perfect circles
- Preserves all interactions: Hover, click, zoom, collision detection all work normally
- Fallback to colors: Dots without images use normal color fills
- Performance optimized: Uses SVG patterns for efficient rendering
const AlbumViz = () => {
const albums = [
{
id: 'album1',
x: 100, y: 100,
imageUrl: '/covers/dark-side-moon.jpg',
title: 'Dark Side of the Moon',
artist: 'Pink Floyd'
},
{
id: 'album2',
x: 200, y: 150,
imageUrl: '/covers/abbey-road.jpg',
title: 'Abbey Road',
artist: 'The Beatles'
}
];
return (
<DotVisualization
data={albums}
useImages={true}
defaultSize={20} // Larger dots for album covers
onHover={(album) => console.log(`${album.title} by ${album.artist}`)}
/>
);
};For better performance with large datasets or when images need to be loaded asynchronously, use the imageProvider and hoverImageProvider props instead of adding imageUrl to each data point.
The traditional approach of adding imageUrl to data objects causes performance issues:
- ❌ Images are re-fetched every time positions update
- ❌ Expensive async operations on every render
- ❌ SVG patterns are recreated unnecessarily
- ❌ Poor performance during animations and interactions
Image providers solve this by separating image loading from position updates:
- ✅ Images are loaded once and cached by the parent component
- ✅ Position updates become pure mathematical operations
- ✅ SVG patterns are created once and reused
- ✅ Smooth animations without blocking async calls
import { DotVisualization } from 'react-dot-visualization';
const MusicVisualization = () => {
const tracks = [
{ id: 'track1', x: 100, y: 150, title: 'Song One' },
{ id: 'track2', x: 200, y: 100, title: 'Song Two' }
];
// Cache images in parent component
const [artworkCache, setArtworkCache] = useState(new Map());
// Preload images when tracks change
useEffect(() => {
const loadArtwork = async () => {
const cache = new Map();
for (const track of tracks) {
try {
const imageUrl = await fetchArtworkForTrack(track.id);
cache.set(track.id, imageUrl);
} catch (error) {
// Handle missing artwork gracefully
cache.set(track.id, undefined);
}
}
setArtworkCache(cache);
};
loadArtwork();
}, [tracks]);
// Simple image provider function
const imageProvider = (id) => artworkCache.get(id);
return (
<DotVisualization
data={tracks}
useImages={true}
imageProvider={imageProvider}
defaultSize={20}
/>
);
};Use hoverImageProvider to show different images on hover (e.g., high-resolution versions):
const ImageVisualization = () => {
const [thumbnailCache, setThumbnailCache] = useState(new Map());
const [fullSizeCache, setFullSizeCache] = useState(new Map());
const imageProvider = (id) => thumbnailCache.get(id);
const hoverImageProvider = (id) => fullSizeCache.get(id);
return (
<DotVisualization
data={data}
useImages={true}
imageProvider={imageProvider}
hoverImageProvider={hoverImageProvider}
hoverSizeMultiplier={1.5}
/>
);
};// Fallback chain provider
const createFallbackProvider = (...providers) => (id) => {
for (const provider of providers) {
const result = provider(id);
if (result) return result;
}
return undefined;
};
// Category-based provider
const createCategoryProvider = (categoryMap, imageMap) => (id) => {
const category = categoryMap.get(id);
return imageMap.get(category);
};
// Composed provider example
const MyVisualization = () => {
const primaryProvider = (id) => primaryCache.get(id);
const fallbackProvider = (id) => `/placeholder/${id}.png`;
const imageProvider = createFallbackProvider(
primaryProvider,
fallbackProvider
);
return (
<DotVisualization
data={data}
useImages={true}
imageProvider={imageProvider}
/>
);
};Old approach (slower performance):
const data = [
{ id: 1, x: 100, y: 150, imageUrl: '/image1.jpg' },
{ id: 2, x: 200, y: 100, imageUrl: '/image2.jpg' }
];
<DotVisualization data={data} useImages={true} />New approach (optimized performance):
const data = [
{ id: 1, x: 100, y: 150 },
{ id: 2, x: 200, y: 100 }
];
const imageMap = new Map([
[1, '/image1.jpg'],
[2, '/image2.jpg']
]);
const imageProvider = (id) => imageMap.get(id);
<DotVisualization
data={data}
useImages={true}
imageProvider={imageProvider}
/>The component maintains backward compatibility - imageUrl properties still work, but imageProvider takes precedence when both are present.
import { DotVisualization } from 'react-dot-visualization';
const AdvancedExample = () => {
const [selectedItem, setSelectedItem] = useState(null);
// Generate data with custom properties
const data = Array.from({ length: 200 }, (_, i) => ({
id: i,
x: Math.random() * 1000,
y: Math.random() * 1000,
size: Math.random() * 8 + 2,
color: `hsl(${i * 137.508}deg, 70%, 50%)`, // Golden angle color distribution
category: ['A', 'B', 'C'][Math.floor(Math.random() * 3)],
value: Math.random() * 100
}));
return (
<DotVisualization
data={data}
onHover={(item) => console.log(`Hovering: ${item.category} - ${item.value}`)}
onClick={(item) => setSelectedItem(item)}
onZoomStart={() => setSelectedItem(null)} // Clear selection on zoom
enableCollisionDetection={true}
zoomExtent={[0.5, 20]}
margin={0.2}
dotStroke="#333"
dotStrokeWidth={1}
style={{
border: '2px solid #ddd',
borderRadius: '8px'
}}
/>
);
};- Modern browsers with SVG and ES6+ support
- Tested with React 18+
react(peer dependency)react-dom(peer dependency)d3- For zoom/pan behavior and force simulation