/** * Copyright (c) 2018-2019 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal * @author Alexander Rose */ import * as React from 'react' import { NumericInput } from './common'; import { noop } from '../../mol-util'; export class Slider extends React.Component<{ min: number, max: number, value: number, step?: number, onChange: (v: number) => void, disabled?: boolean, onEnter?: () => void }, { isChanging: boolean, current: number }> { state = { isChanging: false, current: 0 } static getDerivedStateFromProps(props: { value: number }, state: { isChanging: boolean, current: number }) { if (state.isChanging || props.value === state.current) return null; return { current: props.value }; } begin = () => { this.setState({ isChanging: true }); } end = (v: number) => { this.setState({ isChanging: false }); this.props.onChange(v); } updateCurrent = (current: number) => { this.setState({ current }); } updateManually = (v: number) => { this.setState({ isChanging: true }); let n = v; if (this.props.step === 1) n = Math.round(n); if (n < this.props.min) n = this.props.min; if (n > this.props.max) n = this.props.max; this.setState({ current: n, isChanging: true }); } onManualBlur = () => { this.setState({ isChanging: false }); this.props.onChange(this.state.current); } render() { let step = this.props.step; if (step === void 0) step = 1; return
; } } export class Slider2 extends React.Component<{ min: number, max: number, value: [number, number], step?: number, onChange: (v: [number, number]) => void, disabled?: boolean, onEnter?: () => void }, { isChanging: boolean, current: [number, number] }> { state = { isChanging: false, current: [0, 1] as [number, number] } static getDerivedStateFromProps(props: { value: [number, number] }, state: { isChanging: boolean, current: [number, number] }) { if (state.isChanging || (props.value[0] === state.current[0]) && (props.value[1] === state.current[1])) return null; return { current: props.value }; } begin = () => { this.setState({ isChanging: true }); } end = (v: [number, number]) => { this.setState({ isChanging: false }); this.props.onChange(v); } updateCurrent = (current: [number, number]) => { this.setState({ current }); } updateMax = (v: number) => { let n = v; if (this.props.step === 1) n = Math.round(n); if (n < this.state.current[0]) n = this.state.current[0] else if (n < this.props.min) n = this.props.min; if (n > this.props.max) n = this.props.max; this.props.onChange([this.state.current[0], n]); } updateMin = (v: number) => { let n = v; if (this.props.step === 1) n = Math.round(n); if (n < this.props.min) n = this.props.min; if (n > this.state.current[1]) n = this.state.current[1]; else if (n > this.props.max) n = this.props.max; this.props.onChange([n, this.state.current[1]]); } render() { let step = this.props.step; if (step === void 0) step = 1; return
; } } /** * The following code was adapted from react-components/slider library. * * The MIT License (MIT) * Copyright (c) 2015-present Alipay.com, https://www.alipay.com/ * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ function classNames(_classes: { [name: string]: boolean | number }) { let classes = []; let hasOwn = {}.hasOwnProperty; for (let i = 0; i < arguments.length; i++) { let arg = arguments[i]; if (!arg) continue; let argType = typeof arg; if (argType === 'string' || argType === 'number') { classes.push(arg); } else if (Array.isArray(arg)) { classes.push(classNames.apply(null, arg)); } else if (argType === 'object') { for (let key in arg) { if (hasOwn.call(arg, key) && arg[key]) { classes.push(key); } } } } return classes.join(' '); } function isNotTouchEvent(e: TouchEvent) { return e.touches.length > 1 || (e.type.toLowerCase() === 'touchend' && e.touches.length > 0); } function getTouchPosition(vertical: boolean, e: TouchEvent) { return vertical ? e.touches[0].clientY : e.touches[0].pageX; } function getMousePosition(vertical: boolean, e: MouseEvent) { return vertical ? e.clientY : e.pageX; } function getHandleCenterPosition(vertical: boolean, handle: HTMLElement) { const coords = handle.getBoundingClientRect(); return vertical ? coords.top + (coords.height * 0.5) : coords.left + (coords.width * 0.5); } function pauseEvent(e: MouseEvent | TouchEvent) { e.stopPropagation(); e.preventDefault(); } export class Handle extends React.Component, {}> { render() { const { className, tipFormatter, vertical, offset, value, index, } = this.props as HandleProps; const style = vertical ? { bottom: `${offset}%` } : { left: `${offset}%` }; return (
); } } export interface SliderBaseProps { min: number, max: number, step?: number, defaultValue?: number | number[], value?: number | number[], marks?: any, className?: string, prefixCls?: string, disabled?: boolean, onBeforeChange?: (value: number | number[]) => void, onChange?: (value: number | number[]) => void, onAfterChange?: (value: number | number[]) => void, handle?: JSX.Element, tipFormatter?: (value: number, index: number) => any, range?: boolean | number, vertical?: boolean, allowCross?: boolean, pushable?: boolean | number, } export interface SliderBaseState { handle: number | null, recent: number, bounds: number[] } export class SliderBase extends React.Component { private sliderElement = React.createRef(); private handleElements: React.Ref[] = []; constructor(props: SliderBaseProps) { super(props); const { range, min, max } = props; const initialValue = range ? Array.apply(null, Array(+range + 1)).map(() => min) : min; const defaultValue = ('defaultValue' in props ? props.defaultValue : initialValue); const value = (props.value !== undefined ? props.value : defaultValue); const bounds = (range ? value : [min, value]).map((v: number) => this.trimAlignValue(v)); let recent; if (range && bounds[0] === bounds[bounds.length - 1] && bounds[0] === max) { recent = 0; } else { recent = bounds.length - 1; } this.state = { handle: null, recent, bounds, }; } public static defaultProps: SliderBaseProps = { prefixCls: 'msp-slider-base', className: '', min: 0, max: 100, step: 1, marks: {}, handle: v} value={0} index={0} />, onBeforeChange: noop, onChange: noop, onAfterChange: noop, tipFormatter: (value, index) => value, disabled: false, range: false, vertical: false, allowCross: true, pushable: false, }; private dragOffset = 0; private startPosition = 0; private startValue = 0; private _getPointsCache: any = void 0; componentDidUpdate(prevProps: SliderBaseProps) { if (!('value' in this.props || 'min' in this.props || 'max' in this.props)) return; const { bounds } = this.state; if (prevProps.range) { const value = this.props.value || bounds; const nextBounds = (value as number[]).map((v: number) => this.trimAlignValue(v, this.props)); if (nextBounds.every((v: number, i: number) => v === bounds[i])) return; this.setState({ bounds: nextBounds } as SliderBaseState); if (bounds.some(v => this.isValueOutOfBounds(v, this.props))) { this.props.onChange!(nextBounds); } } else { const value = this.props.value !== undefined ? this.props.value : bounds[1]; const nextValue = this.trimAlignValue(value as number, this.props); if (nextValue === bounds[1] && bounds[0] === prevProps.min) return; this.setState({ bounds: [prevProps.min, nextValue] } as SliderBaseState); if (this.isValueOutOfBounds(bounds[1], this.props)) { this.props.onChange!(nextValue); } } } onChange(state: this['state']) { const props = this.props; const isNotControlled = !('value' in props); if (isNotControlled) { this.setState(state); } else if (state.handle !== undefined) { this.setState({ handle: state.handle } as SliderBaseState); } const data = { ...this.state, ...(state as any) }; const changedValue = props.range ? data.bounds : data.bounds[1]; props.onChange!(changedValue); } onMouseDown = (e: MouseEvent) => { if (e.button !== 0) { return; } let position = getMousePosition(this.props.vertical!, e); if (!this.isEventFromHandle(e)) { this.dragOffset = 0; } else { const handlePosition = getHandleCenterPosition(this.props.vertical!, e.target as HTMLElement); this.dragOffset = position - handlePosition; position = handlePosition; } this.onStart(position); this.addDocumentEvents('mouse'); pauseEvent(e); } onMouseMove(e: MouseEvent) { const position = getMousePosition(this.props.vertical!, e); this.onMove(e, position - this.dragOffset); } onMove(e: MouseEvent | TouchEvent, position: number) { pauseEvent(e); const props = this.props; const state = this.state; let diffPosition = position - this.startPosition; diffPosition = this.props.vertical ? -diffPosition : diffPosition; const diffValue = diffPosition / this.getSliderLength() * (props.max - props.min); const value = this.trimAlignValue(this.startValue + diffValue); const oldValue = state.bounds[state.handle!]; if (value === oldValue) return; const nextBounds = [...state.bounds]; nextBounds[state.handle!] = value; let nextHandle = state.handle!; if (props.pushable !== false) { const originalValue = state.bounds[nextHandle]; this.pushSurroundingHandles(nextBounds, nextHandle, originalValue); } else if (props.allowCross) { nextBounds.sort((a, b) => a - b); nextHandle = nextBounds.indexOf(value); } this.onChange({ handle: nextHandle, bounds: nextBounds, } as SliderBaseState); } onStart(position: number) { const props = this.props; props.onBeforeChange!(this.getValue()); const value = this.calcValueByPos(position); this.startValue = value; this.startPosition = position; const state = this.state; const { bounds } = state; let valueNeedChanging = 1; if (this.props.range) { let closestBound = 0; for (let i = 1; i < bounds.length - 1; ++i) { if (value > bounds[i]) { closestBound = i; } } if (Math.abs(bounds[closestBound + 1] - value) < Math.abs(bounds[closestBound] - value)) { closestBound = closestBound + 1; } valueNeedChanging = closestBound; const isAtTheSamePoint = (bounds[closestBound + 1] === bounds[closestBound]); if (isAtTheSamePoint) { valueNeedChanging = state.recent; } if (isAtTheSamePoint && (value !== bounds[closestBound + 1])) { valueNeedChanging = value < bounds[closestBound + 1] ? closestBound : closestBound + 1; } } this.setState({ handle: valueNeedChanging, recent: valueNeedChanging, } as SliderBaseState); const oldValue = state.bounds[valueNeedChanging]; if (value === oldValue) return; const nextBounds = [...state.bounds]; nextBounds[valueNeedChanging] = value; this.onChange({ bounds: nextBounds } as SliderBaseState); } onTouchMove = (e: TouchEvent) => { if (isNotTouchEvent(e)) { this.end('touch'); return; } const position = getTouchPosition(this.props.vertical!, e); this.onMove(e, position - this.dragOffset); } onTouchStart = (e: TouchEvent) => { if (isNotTouchEvent(e)) return; let position = getTouchPosition(this.props.vertical!, e); if (!this.isEventFromHandle(e)) { this.dragOffset = 0; } else { const handlePosition = getHandleCenterPosition(this.props.vertical!, e.target as HTMLElement); this.dragOffset = position - handlePosition; position = handlePosition; } this.onStart(position); this.addDocumentEvents('touch'); pauseEvent(e); } /** * Returns an array of possible slider points, taking into account both * `marks` and `step`. The result is cached. */ getPoints() { const { marks, step, min, max } = this.props; const cache = this._getPointsCache; if (!cache || cache.marks !== marks || cache.step !== step) { const pointsObject = { ...marks }; if (step !== null) { for (let point = min; point <= max; point += step!) { pointsObject[point] = point; } } const points = Object.keys(pointsObject).map(parseFloat); points.sort((a, b) => a - b); this._getPointsCache = { marks, step, points }; } return this._getPointsCache.points; } getPrecision(step: number) { const stepString = step.toString(); let precision = 0; if (stepString.indexOf('.') >= 0) { precision = stepString.length - stepString.indexOf('.') - 1; } return precision; } getSliderLength() { const slider = this.sliderElement.current; if (!slider) { return 0; } return this.props.vertical ? slider.clientHeight : slider.clientWidth; } getSliderStart() { const slider = this.sliderElement.current as HTMLElement; const rect = slider.getBoundingClientRect(); return this.props.vertical ? rect.top : rect.left; } getValue(): number { const { bounds } = this.state; return (this.props.range ? bounds : bounds[1]) as number; } private eventHandlers = { 'touchmove': (e: TouchEvent) => this.onTouchMove(e), 'touchend': (e: TouchEvent) => this.end('touch'), 'mousemove': (e: MouseEvent) => this.onMouseMove(e), 'mouseup': (e: MouseEvent) => this.end('mouse'), } addDocumentEvents(type: 'touch' | 'mouse') { if (type === 'touch') { document.addEventListener('touchmove', this.eventHandlers.touchmove); document.addEventListener('touchend', this.eventHandlers.touchend); } else if (type === 'mouse') { document.addEventListener('mousemove', this.eventHandlers.mousemove); document.addEventListener('mouseup', this.eventHandlers.mouseup); } } calcOffset = (value: number) => { const { min, max } = this.props; const ratio = (value - min) / (max - min); return ratio * 100; } calcValue(offset: number) { const { vertical, min, max } = this.props; const ratio = Math.abs(offset / this.getSliderLength()); const value = vertical ? (1 - ratio) * (max - min) + min : ratio * (max - min) + min; return value; } calcValueByPos(position: number) { const pixelOffset = position - this.getSliderStart(); const nextValue = this.trimAlignValue(this.calcValue(pixelOffset)); return nextValue; } end(type: 'mouse' | 'touch') { this.removeEvents(type); this.props.onAfterChange!(this.getValue()); this.setState({ handle: null } as SliderBaseState); } isEventFromHandle(e: MouseEvent | TouchEvent) { for (const h of this.handleElements as any) { if (h.current === e.target) return true; } return false; } isValueOutOfBounds(value: number, props: SliderBaseProps) { return value < props.min || value > props.max; } pushHandle(bounds: number[], handle: number, direction: number, amount: number) { const originalValue = bounds[handle]; let currentValue = bounds[handle]; while (direction * (currentValue - originalValue) < amount) { if (!this.pushHandleOnePoint(bounds, handle, direction)) { // can't push handle enough to create the needed `amount` gap, so we // revert its position to the original value bounds[handle] = originalValue; return false; } currentValue = bounds[handle]; } // the handle was pushed enough to create the needed `amount` gap return true; } pushHandleOnePoint(bounds: number[], handle: number, direction: number) { const points = this.getPoints(); const pointIndex = points.indexOf(bounds[handle]); const nextPointIndex = pointIndex + direction; if (nextPointIndex >= points.length || nextPointIndex < 0) { // reached the minimum or maximum available point, can't push anymore return false; } const nextHandle = handle + direction; const nextValue = points[nextPointIndex]; const { pushable: threshold } = this.props; const diffToNext = direction * (bounds[nextHandle] - nextValue); if (!this.pushHandle(bounds, nextHandle, direction, +threshold! - diffToNext)) { // couldn't push next handle, so we won't push this one either return false; } // push the handle bounds[handle] = nextValue; return true; } pushSurroundingHandles(bounds: number[], handle: number, originalValue: number) { const { pushable: threshold } = this.props; const value = bounds[handle]; let direction = 0; if (bounds[handle + 1] - value < threshold!) { direction = +1; } else if (value - bounds[handle - 1] < threshold!) { direction = -1; } if (direction === 0) { return; } const nextHandle = handle + direction; const diffToNext = direction * (bounds[nextHandle] - value); if (!this.pushHandle(bounds, nextHandle, direction, +threshold! - diffToNext)) { // revert to original value if pushing is impossible bounds[handle] = originalValue; } } removeEvents(type: 'touch' | 'mouse') { if (type === 'touch') { document.removeEventListener('touchmove', this.eventHandlers.touchmove); document.removeEventListener('touchend', this.eventHandlers.touchend); } else if (type === 'mouse') { document.removeEventListener('mousemove', this.eventHandlers.mousemove); document.removeEventListener('mouseup', this.eventHandlers.mouseup); } } trimAlignValue(v: number, props?: SliderBaseProps) { const { handle, bounds } = (this.state || {}) as this['state']; const { marks, step, min, max, allowCross } = { ...this.props, ...(props || {}) } as SliderBaseProps; let val = v; if (val <= min) { val = min; } if (val >= max) { val = max; } /* eslint-disable eqeqeq */ if (!allowCross && handle != null && handle > 0 && val <= bounds[handle - 1]) { val = bounds[handle - 1]; } if (!allowCross && handle != null && handle < bounds.length - 1 && val >= bounds[handle + 1]) { val = bounds[handle + 1]; } /* eslint-enable eqeqeq */ const points = Object.keys(marks).map(parseFloat); if (step !== null) { const closestStep = (Math.round((val - min) / step!) * step!) + min; points.push(closestStep); } const diffs = points.map((point) => Math.abs(val - point)); const closestPoint = points[diffs.indexOf(Math.min.apply(Math, diffs))]; return step !== null ? parseFloat(closestPoint.toFixed(this.getPrecision(step!))) : closestPoint; } render() { const { handle, bounds, } = this.state; const { className, prefixCls, disabled, vertical, range, step, marks, tipFormatter } = this.props; const customHandle = this.props.handle; const offsets = bounds.map(this.calcOffset); const handleClassName = `${prefixCls}-handle`; const handlesClassNames = bounds.map((v, i) => classNames({ [handleClassName]: true, [`${handleClassName}-${i + 1}`]: true, [`${handleClassName}-lower`]: i === 0, [`${handleClassName}-upper`]: i === bounds.length - 1, })); const isNoTip = (step === null) || (tipFormatter === null); const commonHandleProps = { prefixCls, noTip: isNoTip, tipFormatter, vertical, }; if (this.handleElements.length !== bounds.length) { this.handleElements = []; // = []; for (let i = 0; i < bounds.length; i++) this.handleElements.push(React.createRef()); } const handles = bounds.map((v, i) => React.cloneElement(customHandle!, { ...commonHandleProps, className: handlesClassNames[i], value: v, offset: offsets[i], dragging: handle === i, index: i, key: i, ref: this.handleElements[i] })); if (!range) { handles.shift(); } const sliderClassName = classNames({ [prefixCls!]: true, [`${prefixCls}-with-marks`]: Object.keys(marks).length, [`${prefixCls}-disabled`]: disabled!, [`${prefixCls}-vertical`]: this.props.vertical!, [className!]: !!className, }); return (
{handles}
); } } export interface HandleProps { className: string, vertical: boolean, offset: number, tipFormatter: (v: number, index: number) => any, value: number, index: number, }