vuetify
Version:
Vue Material Component Framework
202 lines (198 loc) • 7.71 kB
JavaScript
import { mergeProps as _mergeProps, createVNode as _createVNode, Fragment as _Fragment } from "vue";
// Styles
import "../VSlider/VSlider.css";
// Components
import { filterInputProps, makeVInputProps, VInput } from "../VInput/VInput.mjs";
import { getOffset, makeSliderProps, useSlider } from "../VSlider/slider.mjs";
import { VLabel } from "../VLabel/index.mjs";
import { VSliderThumb } from "../VSlider/VSliderThumb.mjs";
import { VSliderTrack } from "../VSlider/VSliderTrack.mjs"; // Composables
import { makeFocusProps, useFocus } from "../../composables/focus.mjs";
import { useProxiedModel } from "../../composables/proxiedModel.mjs"; // Utilities
import { computed, ref } from 'vue';
import { genericComponent, useRender } from "../../util/index.mjs"; // Types
export const VRangeSlider = genericComponent()({
name: 'VRangeSlider',
props: {
...makeFocusProps(),
...makeVInputProps(),
...makeSliderProps(),
strict: Boolean,
modelValue: {
type: Array,
default: () => [0, 0]
}
},
emits: {
'update:focused': value => true,
'update:modelValue': value => true
},
setup(props, _ref) {
let {
slots
} = _ref;
const startThumbRef = ref();
const stopThumbRef = ref();
const inputRef = ref();
function getActiveThumb(e) {
if (!startThumbRef.value || !stopThumbRef.value) return;
const startOffset = getOffset(e, startThumbRef.value.$el, props.direction);
const stopOffset = getOffset(e, stopThumbRef.value.$el, props.direction);
const a = Math.abs(startOffset);
const b = Math.abs(stopOffset);
return a < b || a === b && startOffset < 0 ? startThumbRef.value.$el : stopThumbRef.value.$el;
}
const {
activeThumbRef,
hasLabels,
max,
min,
mousePressed,
onSliderMousedown,
onSliderTouchstart,
position,
roundValue,
trackContainerRef
} = useSlider({
/* eslint-disable @typescript-eslint/no-use-before-define */
props,
handleSliderMouseUp: newValue => {
model.value = activeThumbRef.value === startThumbRef.value?.$el ? [newValue, model.value[1]] : [model.value[0], newValue];
},
handleMouseMove: newValue => {
const [start, stop] = model.value;
if (!props.strict && start === stop && start !== min.value) {
activeThumbRef.value = newValue > start ? stopThumbRef.value?.$el : startThumbRef.value?.$el;
activeThumbRef.value?.focus();
}
if (activeThumbRef.value === startThumbRef.value?.$el) {
model.value = [Math.min(newValue, stop), stop];
} else {
model.value = [start, Math.max(start, newValue)];
}
},
getActiveThumb
/* eslint-enable @typescript-eslint/no-use-before-define */
});
const model = useProxiedModel(props, 'modelValue', undefined, arr => {
// eslint-disable-next-line @typescript-eslint/prefer-optional-chain
if (!arr || !arr.length) return [0, 0];
return arr.map(value => roundValue(value));
});
const {
isFocused,
focus,
blur
} = useFocus(props);
const trackStart = computed(() => position(model.value[0]));
const trackStop = computed(() => position(model.value[1]));
useRender(() => {
const [inputProps, _] = filterInputProps(props);
const hasPrepend = !!(props.label || slots.label || slots.prepend);
return _createVNode(VInput, _mergeProps({
"class": ['v-slider', 'v-range-slider', {
'v-slider--has-labels': !!slots['tick-label'] || hasLabels.value,
'v-slider--focused': isFocused.value,
'v-slider--pressed': mousePressed.value,
'v-slider--disabled': props.disabled
}],
"ref": inputRef
}, inputProps, {
"focused": isFocused.value
}), {
...slots,
prepend: hasPrepend ? slotProps => _createVNode(_Fragment, null, [slots.label?.(slotProps) ?? props.label ? _createVNode(VLabel, {
"class": "v-slider__label",
"text": props.label
}, null) : undefined, slots.prepend?.(slotProps)]) : undefined,
default: _ref2 => {
let {
id,
messagesId
} = _ref2;
return _createVNode("div", {
"class": "v-slider__container",
"onMousedown": onSliderMousedown,
"onTouchstartPassive": onSliderTouchstart
}, [_createVNode("input", {
"id": `${id.value}_start`,
"name": props.name || id.value,
"disabled": props.disabled,
"readonly": props.readonly,
"tabindex": "-1",
"value": model.value[0]
}, null), _createVNode("input", {
"id": `${id.value}_stop`,
"name": props.name || id.value,
"disabled": props.disabled,
"readonly": props.readonly,
"tabindex": "-1",
"value": model.value[1]
}, null), _createVNode(VSliderTrack, {
"ref": trackContainerRef,
"start": trackStart.value,
"stop": trackStop.value
}, {
'tick-label': slots['tick-label']
}), _createVNode(VSliderThumb, {
"ref": startThumbRef,
"aria-describedby": messagesId.value,
"focused": isFocused && activeThumbRef.value === startThumbRef.value?.$el,
"modelValue": model.value[0],
"onUpdate:modelValue": v => model.value = [v, model.value[1]],
"onFocus": e => {
focus();
activeThumbRef.value = startThumbRef.value?.$el;
// Make sure second thumb is focused if
// the thumbs are on top of each other
// and they are both at minimum value
// but only if focused from outside.
if (model.value[0] === model.value[1] && model.value[1] === min.value && e.relatedTarget !== stopThumbRef.value?.$el) {
startThumbRef.value?.$el.blur();
stopThumbRef.value?.$el.focus();
}
},
"onBlur": () => {
blur();
activeThumbRef.value = undefined;
},
"min": min.value,
"max": model.value[1],
"position": trackStart.value
}, {
'thumb-label': slots['thumb-label']
}), _createVNode(VSliderThumb, {
"ref": stopThumbRef,
"aria-describedby": messagesId.value,
"focused": isFocused && activeThumbRef.value === stopThumbRef.value?.$el,
"modelValue": model.value[1],
"onUpdate:modelValue": v => model.value = [model.value[0], v],
"onFocus": e => {
focus();
activeThumbRef.value = stopThumbRef.value?.$el;
// Make sure first thumb is focused if
// the thumbs are on top of each other
// and they are both at maximum value
// but only if focused from outside.
if (model.value[0] === model.value[1] && model.value[0] === max.value && e.relatedTarget !== startThumbRef.value?.$el) {
stopThumbRef.value?.$el.blur();
startThumbRef.value?.$el.focus();
}
},
"onBlur": () => {
blur();
activeThumbRef.value = undefined;
},
"min": model.value[0],
"max": max.value,
"position": trackStop.value
}, {
'thumb-label': slots['thumb-label']
})]);
}
});
});
return {};
}
});
//# sourceMappingURL=VRangeSlider.mjs.map