diff --git a/README.md b/README.md index 3f68fa4..06c6342 100644 --- a/README.md +++ b/README.md @@ -202,11 +202,25 @@ The _typenames_ is a string containing one or more _typename_ separated by white - `start` - after a new pointer becomes active (on mousedown or touchstart). - `drag` - after an active pointer moves (on mousemove or touchmove). - `end` - after an active pointer becomes inactive (on mouseup, touchend or touchcancel). +- `rangeChange` - after the range of the slider is changed +- `rangeStart` - after mousedown on the ticks area +- `rangeDrag` - after mousemove within the ticks area +- `rangeEnd` - after the range adjustment is inactive (mouseup) You might consider throttling `onchange` and `drag` events. For example using [`lodash.throttle`](https://lodash.com/docs/4.17.4#throttle). See [_dispatch_.on](https://github.com/d3/d3-dispatch#dispatch_on) for more. +# slider.dragToPan(value]) [<>](https://github.com/johnwalley/d3-simple-slider/blob/master/src/slider.js#L998 'Source') + +If _value_ is true, enables dragging on the ticks of the slider to change its range. +Dragging around the middle area for panning (moving both the minimum and the maximum). +Dragging near the left edge allows adjusting only the minimum, and dragging around near the right edge allows adjusting only the maximum. +The current slider knob—current value—is always kept within the range. +By default, panning is disabled. + + + ## 🤝 How to Contribute Please read the [contribution guidelines for this project](CONTRIBUTING.md) diff --git a/example/index.html b/example/index.html index 0937d9f..745d2a7 100644 --- a/example/index.html +++ b/example/index.html @@ -58,6 +58,11 @@

Transition

+

Drag to pan & zoom

+
+

+
+

Examples

New York Times

@@ -454,4 +459,28 @@

Color picker

}); d3.select('p#value-color-picker').text(`#${num2hex(rgb)}`); + + var dragToPanSlider = d3 + .sliderBottom() + .min(2) + .max(15) + .width(300) + .step(1) + .default(5) + .dragToPan(true) + .on('onchange', function (val) { + d3.select('p#value-drag-to-pan').text(val); + }); + + var dragToPanG = d3 + .select('#slider-drag-to-pan') + .append('svg') + .attr('width', 500) + .attr('height', 100) + .append('g') + .attr('transform', 'translate(30,30)'); + + dragToPanG.call(dragToPanSlider); + + d3.select('p#value-drag-to-pan').text(dragToPanSlider.value()); diff --git a/src/slider.js b/src/slider.js index c145555..7ace9ef 100644 --- a/src/slider.js +++ b/src/slider.js @@ -41,8 +41,22 @@ function slider(orientation, scale) { var ticks = null; var displayFormat = null; var fill = null; - - var listeners = dispatch('onchange', 'start', 'end', 'drag'); + var minTravelDrag = 5; + var dragStartPos = null; + var lastAnimation = null; + var dragModeFn = null; + var dragToPan = false; + + var listeners = dispatch( + 'onchange', + 'start', + 'end', + 'drag', + 'rangeChange', + 'rangeStart', + 'rangeDrag', + 'rangeEnd' + ); var selection = null; var identityClamped = null; @@ -108,6 +122,11 @@ function slider(orientation, scale) { .clamp(true); } + if (!step) { + step = + (domain[1] - domain[0]) / (ticks ? ticks : scale.ticks().length - 1); + } + identityClamped = scaleLinear() .range(scale.range()) .domain(scale.range()) @@ -129,6 +148,51 @@ function slider(orientation, scale) { .attr('transform', transformAcross(k * 7)) .attr('class', 'axis'); + var rangePanning = selection + .selectAll('.range-panning') + .data(dragToPan ? [null] : []); + + var rangePanningEnter = rangePanning + .enter() + .append('g') + .attr('class', 'range-panning') + .call( + dragToPan + ? drag() + .on('start', panZoomStart) + .on('drag', panZoomDrag) + .on('end', panZoomEnd) + : function () {} + ); + + rangePanningEnter + .append('rect') + .attr('x', function () { + return -SLIDER_END_PADDING; + }) + .attr('width', function () { + if ([top, bottom].indexOf(orientation) !== -1) { + return scale.range()[1] + 2 * SLIDER_END_PADDING; + } + + return 27; + }) + .attr('height', function () { + if ([top, bottom].indexOf(orientation) === -1) { + return scale.range()[0] + 2 * SLIDER_END_PADDING; + } + + return 27; + }) + .attr('y', k * 27 * 0.5) + .attr('fill', 'transparent') + .attr( + 'cursor', + orientation === top || orientation === bottom + ? 'ew-resize' + : 'ns-resize' + ) + var sliderSelection = selection.selectAll('.slider').data([null]); var sliderEnter = sliderSelection @@ -429,6 +493,147 @@ function slider(orientation, scale) { handleIndex = null; } + function inc(val) { + return val instanceof Date + ? new Date(val.getTime() + getStep()) + : val + getStep(); + } + + function subtract(val) { + return val instanceof Date + ? new Date(val.getTime() - getStep()) + : val - getStep(); + } + + // allow zoom in and out domain based on drag direction relative to drag start position + function rangeZoom(dragStartPos) { + /** + * drag start position = right + */ + var nextDomain; + if (dragStartPos.x > width * 0.5) { + nextDomain = dragStartPos.x < event.x ? inc(domain[1]) : subtract(domain[1]) + domain = [ + domain[0], + nextDomain < max(value) ? domain[1] : nextDomain, + ]; + } else if (dragStartPos.x <= width * 0.5) { + /** + * drag start position = left + */ + nextDomain = dragStartPos.x <= event.x ? inc(domain[0]) : subtract(domain[0]) + + domain = [ + nextDomain > min(value) ? domain[0] : nextDomain, + domain[1], + ]; + } + } + + function rangePan(dragStartPos) { + // drag direction = left + if (dragStartPos.x > event.x) { + domain = [ + domain[0] >= min(value) ? domain[0] : inc(domain[0]), + inc(domain[1]), + ]; + } else { + domain = [ + subtract(domain[0]), + domain[1] <= max(value) ? domain[1] : subtract(domain[1]), + ]; + } + + return domain; + } + + function getStep() { + // Need to validate further about step calc if no step given + return step; + } + + function panZoomStart() { + var zooming = event.x / width <= 0.33 || event.x / width >= 0.66; + + dragModeFn = zooming ? rangeZoom : rangePan; + dragStartPos = event; + + listeners.call('rangeStart', slider); + } + + function panZoomDrag() { + if (Math.abs(dragStartPos.x - event.x) < minTravelDrag) return; + + // prevent animation overlap + if (lastAnimation === null) { + lastAnimation = new Date(); + } + + listeners.call('rangeDrag', slider, event); + + // only after previous animation completed + if (new Date() - lastAnimation >= UPDATE_DURATION) { + dragModeFn.call(this, dragStartPos); + + scale.domain(domain); + + listeners.call('rangeChange', slider, domain); + + refreshAxis(); + + lastAnimation = null; + } + } + + function panZoomEnd() { + dragStartPos = null; + + listeners.call('rangeEnd', slider); + } + + function refreshAxis() { + selection + .select('.axis') + .transition() + .ease(easeQuadOut) + .duration(UPDATE_DURATION) + .call(function (sel) { + sel.call( + axisFunction(scale) + .ticks(ticks) + .tickFormat(tickFormat) + .tickValues(tickValues) + ); + + sel + .selectAll('text') + .attr('fill', '#aaa') + .attr(y, k * 20) + .attr( + 'dy', + orientation === top + ? '0em' + : orientation === bottom + ? '.71em' + : '.32em' + ) + .attr( + 'text-anchor', + orientation === right + ? 'start' + : orientation === left + ? 'end' + : 'middle' + ); + + updateHandle(value, true); + }) + .on('start', function () { + selection.selectAll('.axis line').attr('stroke', '#aaa'); + selection.select('.axis .domain').remove(); + }); + } + textSelection = selection.selectAll('.parameter-value text'); fillSelection = selection.select('.track-fill'); } @@ -790,6 +995,12 @@ function slider(orientation, scale) { return value === listeners ? slider : value; }; + slider.dragToPan = function (_) { + if (!arguments.length) return dragToPan; + dragToPan = _; + return slider; + }; + return slider; }