<html>
<head>
<meta charset="utf-8">
<title>Wheel Zoom & Drag</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="https://leeoniya.github.io/uPlot/dist/uPlot.min.css">
</head>
<body>
<script src="https://leeoniya.github.io/uPlot/dist/uPlot.iife.js"></script>
</body>
</html>
function wheelZoomPlugin(opts) {
let factor = opts.factor || 0.75;
let xMin, xMax, yMin, yMax, xRange, yRange;
function clamp(nRange, nMin, nMax, fRange, fMin, fMax) {
if (nRange > fRange) {
nMin = fMin;
nMax = fMax;
}
else if (nMin < fMin) {
nMin = fMin;
nMax = fMin + nRange;
}
else if (nMax > fMax) {
nMax = fMax;
nMin = fMax - nRange;
}
return [nMin, nMax];
}
return {
hooks: {
ready: u => {
xMin = u.scales.x.min;
xMax = u.scales.x.max;
yMin = u.scales.y.min;
yMax = u.scales.y.max;
xRange = xMax - xMin;
yRange = yMax - yMin;
let plot = u.root.querySelector(".u-over");
let rect = plot.getBoundingClientRect();
// wheel drag pan
plot.addEventListener("mousedown", e => {
if (e.button == 1) {
//plot.style.cursor = "move";
e.preventDefault();
let left0 = e.clientX;
let top0 = e.clientY;
let scXMin0 = u.scales.x.min;
let scXMax0 = u.scales.x.max;
let scYMin0 = u.scales.y.min;
let scYMax0 = u.scales.y.max;
let xUnitsPerPx = u.posToVal(1, 'x') - u.posToVal(0, 'x');
let yUnitsPerPx = u.posToVal(1, 'y') - u.posToVal(0, 'y');
function onmove(e) {
e.preventDefault();
let left1 = e.clientX;
let top1 = e.clientY;
let dx = xUnitsPerPx * (left1 - left0);
let dy = yUnitsPerPx * (top1 - top0);
u.batch(() => {
u.setScale('x', {
min: scXMin0 - dx,
max: scXMax0 - dx,
});
u.setScale("y", {
min: scYMin0 - dy,
max: scYMax0 - dy,
});
});
}
function onup(e) {
document.removeEventListener("mousemove", onmove);
document.removeEventListener("mouseup", onup);
}
document.addEventListener("mousemove", onmove);
document.addEventListener("mouseup", onup);
}
});
// wheel scroll zoom
plot.addEventListener("wheel", e => {
e.preventDefault();
let {left, top} = u.cursor;
let leftPct = left/rect.width;
let btmPct = 1 - top/rect.height;
let xVal = u.posToVal(left, "x");
let yVal = u.posToVal(top, "y");
let oxRange = u.scales.x.max - u.scales.x.min;
let oyRange = u.scales.y.max - u.scales.y.min;
let nxRange = e.deltaY < 0 ? oxRange * factor : oxRange / factor;
let nxMin = xVal - leftPct * nxRange;
let nxMax = nxMin + nxRange;
[nxMin, nxMax] = clamp(nxRange, nxMin, nxMax, xRange, xMin, xMax);
let nyRange = e.deltaY < 0 ? oyRange * factor : oyRange / factor;
let nyMin = yVal - btmPct * nyRange;
let nyMax = nyMin + nyRange;
[nyMin, nyMax] = clamp(nyRange, nyMin, nyMax, yRange, yMin, yMax);
u.batch(() => {
u.setScale("x", {
min: nxMin,
max: nxMax,
});
u.setScale("y", {
min: nyMin,
max: nyMax,
});
});
});
}
}
};
}
function makeChart() {
console.time('chart');
let opts = {
title: "Wheel Zoom & Drag",
width: 1000,
height: 500,
plugins: [
wheelZoomPlugin({factor: 0.75})
],
cursor: {
drag: {x: false, setScale: false}
},
scales: {
x: {
time: false,
},
y: {
auto: false,
min: -2,
max: 2
}
},
series: [
{},
{
label: "One",
stroke: "red",
},
/*{
label: "Two",
stroke: "blue",
},*/
]
};
const data = [
[ 1, 2, 3, 4, 5],
[40,43,60,65,71],
//[18,24,37,55,55],
];
for (var i = 0; i < 100000; i++) {
data[0][i] = i;
data[1][i] = Math.sin(i / 500);
//data[2][i] = Math.cos(i / 500);
}
//data[1][0] = 2;
//data[2][0] = -2;
let u = new uPlot(opts, data, document.body);
console.timeEnd('chart');
}
makeChart();
This Pen doesn't use any external CSS resources.
This Pen doesn't use any external JavaScript resources.