UNPKG

@patternfly/react-core

Version:

This library provides a set of common React components for use with the PatternFly reference implementation.

239 lines • 12.7 kB
import { __rest } from "tslib"; import * as React from 'react'; import { useState } from 'react'; import styles from '@patternfly/react-styles/css/components/Slider/slider'; import { css } from '@patternfly/react-styles'; import { SliderStep } from './SliderStep'; import { InputGroup, InputGroupText } from '../InputGroup'; import { TextInput } from '../TextInput'; import { Tooltip } from '../Tooltip'; const getPercentage = (current, max) => (100 * current) / max; export const Slider = (_a) => { var { className, value = 0, customSteps, areCustomStepsContinuous = false, isDisabled = false, isInputVisible = false, inputValue = 0, inputLabel, inputAriaLabel = 'Slider value input', thumbAriaLabel = 'Value', hasTooltipOverThumb = false, inputPosition = 'right', onChange, leftActions, rightActions, step = 1, min = 0, max = 100, showTicks = false, showBoundaries = true } = _a, props = __rest(_a, ["className", "value", "customSteps", "areCustomStepsContinuous", "isDisabled", "isInputVisible", "inputValue", "inputLabel", "inputAriaLabel", "thumbAriaLabel", "hasTooltipOverThumb", "inputPosition", "onChange", "leftActions", "rightActions", "step", "min", "max", "showTicks", "showBoundaries"]); const sliderRailRef = React.useRef(); const thumbRef = React.useRef(); const [localValue, setValue] = useState(value); const [localInputValue, setLocalInputValue] = useState(inputValue); React.useEffect(() => { setValue(value); }, [value]); React.useEffect(() => { setLocalInputValue(inputValue); }, [inputValue]); let diff = 0; let snapValue; // calculate style value percentage const stylePercent = ((localValue - min) * 100) / (max - min); const style = { '--pf-c-slider--value': `${stylePercent}%` }; const widthChars = React.useMemo(() => localInputValue.toString().length, [localInputValue]); const inputStyle = { '--pf-c-slider__value--c-form-control--width-chars': widthChars }; const onChangeHandler = (value) => { setLocalInputValue(Number(value)); }; const handleKeyPressOnInput = (event) => { if (event.key === 'Enter') { event.preventDefault(); if (onChange) { onChange(localValue, localInputValue, setLocalInputValue); } } }; const onInputFocus = (e) => { e.stopPropagation(); }; const onThumbClick = () => { thumbRef.current.focus(); }; const onBlur = () => { if (onChange) { onChange(localValue, localInputValue, setLocalInputValue); } }; const findAriaTextValue = () => { if (!areCustomStepsContinuous && customSteps) { const matchingStep = customSteps.find(stepObj => stepObj.value === localValue); if (matchingStep) { return matchingStep.label; } } // For continuous steps default to showing 2 decimals in tooltip // Consider making it configurable via a property return Number(Number(localValue).toFixed(2)).toString(); }; const handleThumbDragEnd = () => { document.removeEventListener('mousemove', callbackThumbMove); document.removeEventListener('mouseup', callbackThumbUp); document.removeEventListener('touchmove', callbackThumbMove); document.removeEventListener('touchend', callbackThumbUp); document.removeEventListener('touchcancel', callbackThumbUp); }; const handleMouseDown = (e) => { e.stopPropagation(); e.preventDefault(); diff = e.clientX - thumbRef.current.getBoundingClientRect().left; document.addEventListener('mousemove', callbackThumbMove); document.addEventListener('mouseup', callbackThumbUp); }; const handleTouchStart = (e) => { e.stopPropagation(); diff = e.touches[0].clientX - thumbRef.current.getBoundingClientRect().left; document.addEventListener('touchmove', callbackThumbMove, { passive: false }); document.addEventListener('touchend', callbackThumbUp); document.addEventListener('touchcancel', callbackThumbUp); }; const onSliderRailClick = (e) => { handleThumbMove(e); if (snapValue && !areCustomStepsContinuous) { thumbRef.current.style.setProperty('--pf-c-slider--value', `${snapValue}%`); setValue(snapValue); if (onChange) { onChange(snapValue); } } }; const handleThumbMove = (e) => { if (e.type === 'touchmove') { e.preventDefault(); e.stopImmediatePropagation(); } const clientPosition = e.touches && e.touches.length ? e.touches[0].clientX : e.clientX; let newPosition = clientPosition - diff - sliderRailRef.current.getBoundingClientRect().left; const end = sliderRailRef.current.offsetWidth - thumbRef.current.offsetWidth; const start = 0; if (newPosition < start) { newPosition = 0; } if (newPosition > end) { newPosition = end; } const newPercentage = getPercentage(newPosition, end); thumbRef.current.style.setProperty('--pf-c-slider--value', `${newPercentage}%`); // convert percentage to value const newValue = Math.round(((newPercentage * (max - min)) / 100 + min) * 100) / 100; setValue(newValue); if (!customSteps) { // snap to new value if not custom steps snapValue = Math.round((Math.round((newValue - min) / step) * step + min) * 100) / 100; thumbRef.current.style.setProperty('--pf-c-slider--value', `${snapValue}%`); setValue(snapValue); } /* If custom steps are discrete, snap to closest step value */ if (!areCustomStepsContinuous && customSteps) { let percentage = newPercentage; if (customSteps[customSteps.length - 1].value !== 100) { const numSteps = customSteps.length - 1; const percentagePerStep = 100 / numSteps; percentage = newPercentage / percentagePerStep; } const stepIndex = customSteps.findIndex(stepObj => stepObj.value >= percentage); if (customSteps[stepIndex].value === percentage) { snapValue = customSteps[stepIndex].value; } else { const midpoint = (customSteps[stepIndex].value + customSteps[stepIndex - 1].value) / 2; if (midpoint > percentage) { snapValue = customSteps[stepIndex - 1].value; } else { snapValue = customSteps[stepIndex].value; } } setValue(snapValue); } // Call onchange callback if (onChange) { if (snapValue !== undefined) { onChange(snapValue); } else { onChange(newValue); } } }; const callbackThumbMove = React.useCallback(handleThumbMove, [min, max, customSteps, onChange]); const callbackThumbUp = React.useCallback(handleThumbDragEnd, [min, max, customSteps, onChange]); const handleThumbKeys = (e) => { const key = e.key; if (key !== 'ArrowLeft' && key !== 'ArrowRight') { return; } e.preventDefault(); let newValue = localValue; if (!areCustomStepsContinuous && customSteps) { const stepIndex = customSteps.findIndex(stepObj => stepObj.value === localValue); if (key === 'ArrowRight') { if (stepIndex + 1 < customSteps.length) { { newValue = customSteps[stepIndex + 1].value; } } } else if (key === 'ArrowLeft') { if (stepIndex - 1 >= 0) { newValue = customSteps[stepIndex - 1].value; } } } else { if (key === 'ArrowRight') { newValue = localValue + step <= max ? localValue + step : max; } else if (key === 'ArrowLeft') { newValue = localValue - step >= min ? localValue - step : min; } } if (newValue !== localValue) { thumbRef.current.style.setProperty('--pf-c-slider--value', `${newValue}%`); setValue(newValue); if (onChange) { onChange(newValue); } } }; const displayInput = () => { const textInput = (React.createElement(TextInput, { className: css(styles.formControl), isDisabled: isDisabled, type: "number", value: localInputValue, "aria-label": inputAriaLabel, onKeyDown: handleKeyPressOnInput, onChange: onChangeHandler, onClick: onInputFocus, onFocus: onInputFocus, onBlur: onBlur })); if (inputLabel) { return (React.createElement(InputGroup, null, textInput, React.createElement(InputGroupText, { variant: "plain" }, " ", inputLabel))); } else { return textInput; } }; const getStepValue = (val, min, max) => ((val - min) * 100) / (max - min); const buildSteps = () => { const builtSteps = []; for (let i = min; i <= max; i = i + step) { const stepValue = getStepValue(i, min, max); // If we boundaries but not ticks just generate the needed steps // so that we don't pullute them DOM with empty divs if (!showTicks && showBoundaries && i !== min && i !== max) { continue; } builtSteps.push(React.createElement(SliderStep, { key: i, value: stepValue, label: i.toString(), isTickHidden: !showTicks, isLabelHidden: (i === min || i === max) && showBoundaries ? false : true, isActive: i <= localValue })); } return builtSteps; }; const thumbComponent = (React.createElement("div", { className: css(styles.sliderThumb), ref: thumbRef, tabIndex: isDisabled ? -1 : 0, role: "slider", "aria-valuemin": customSteps ? customSteps[0].value : min, "aria-valuemax": customSteps ? customSteps[customSteps.length - 1].value : max, "aria-valuenow": localValue, "aria-valuetext": findAriaTextValue(), "aria-label": thumbAriaLabel, "aria-disabled": isDisabled, onMouseDown: !isDisabled ? handleMouseDown : null, onTouchStart: !isDisabled ? handleTouchStart : null, onKeyDown: !isDisabled ? handleThumbKeys : null, onClick: !isDisabled ? onThumbClick : null })); return (React.createElement("div", Object.assign({ className: css(styles.slider, className, isDisabled && styles.modifiers.disabled), style: Object.assign(Object.assign({}, style), inputStyle) }, props), leftActions && React.createElement("div", { className: css(styles.sliderActions) }, leftActions), React.createElement("div", { className: css(styles.sliderMain) }, React.createElement("div", { className: css(styles.sliderRail), ref: sliderRailRef, onClick: !isDisabled ? onSliderRailClick : null }, React.createElement("div", { className: css(styles.sliderRailTrack) })), customSteps && (React.createElement("div", { className: css(styles.sliderSteps), "aria-hidden": "true" }, customSteps.map(stepObj => { const minValue = customSteps[0].value; const maxValue = customSteps[customSteps.length - 1].value; const stepValue = getStepValue(stepObj.value, minValue, maxValue); return (React.createElement(SliderStep, { key: stepObj.value, value: stepValue, label: stepObj.label, isLabelHidden: stepObj.isLabelHidden, isActive: stepObj.value <= localValue })); }))), !customSteps && (showTicks || showBoundaries) && (React.createElement("div", { className: css(styles.sliderSteps), "aria-hidden": "true" }, buildSteps())), hasTooltipOverThumb ? (React.createElement(Tooltip, { entryDelay: 0, content: findAriaTextValue() }, thumbComponent)) : (thumbComponent), isInputVisible && inputPosition === 'aboveThumb' && (React.createElement("div", { className: css(styles.sliderValue, styles.modifiers.floating) }, displayInput()))), isInputVisible && inputPosition === 'right' && React.createElement("div", { className: css(styles.sliderValue) }, displayInput()), rightActions && React.createElement("div", { className: css(styles.sliderActions) }, rightActions))); }; Slider.displayName = 'Slider'; //# sourceMappingURL=Slider.js.map