123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224 |
- import { mergeProps as _mergeProps, createVNode as _createVNode, Fragment as _Fragment } from "vue";
- // Styles
- import "../VSlider/VSlider.css";
- // Components
- import { makeVInputProps, VInput } from "../VInput/VInput.mjs";
- import { VLabel } from "../VLabel/index.mjs";
- import { getOffset, makeSliderProps, useSlider, useSteps } from "../VSlider/slider.mjs";
- import { VSliderThumb } from "../VSlider/VSliderThumb.mjs";
- import { VSliderTrack } from "../VSlider/VSliderTrack.mjs"; // Composables
- import { makeFocusProps, useFocus } from "../../composables/focus.mjs";
- import { useRtl } from "../../composables/locale.mjs";
- import { useProxiedModel } from "../../composables/proxiedModel.mjs"; // Utilities
- import { computed, ref } from 'vue';
- import { genericComponent, propsFactory, useRender } from "../../util/index.mjs"; // Types
- export const makeVRangeSliderProps = propsFactory({
- ...makeFocusProps(),
- ...makeVInputProps(),
- ...makeSliderProps(),
- strict: Boolean,
- modelValue: {
- type: Array,
- default: () => [0, 0]
- }
- }, 'VRangeSlider');
- export const VRangeSlider = genericComponent()({
- name: 'VRangeSlider',
- props: makeVRangeSliderProps(),
- emits: {
- 'update:focused': value => true,
- 'update:modelValue': value => true,
- end: value => true,
- start: value => true
- },
- setup(props, _ref) {
- let {
- slots,
- emit
- } = _ref;
- const startThumbRef = ref();
- const stopThumbRef = ref();
- const inputRef = ref();
- const {
- rtlClasses
- } = useRtl();
- 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 steps = useSteps(props);
- const model = useProxiedModel(props, 'modelValue', undefined, arr => {
- if (!arr?.length) return [0, 0];
- return arr.map(value => steps.roundValue(value));
- });
- const {
- activeThumbRef,
- hasLabels,
- max,
- min,
- mousePressed,
- onSliderMousedown,
- onSliderTouchstart,
- position,
- trackContainerRef,
- readonly
- } = useSlider({
- props,
- steps,
- onSliderStart: () => {
- emit('start', model.value);
- },
- onSliderEnd: _ref2 => {
- let {
- value
- } = _ref2;
- const newValue = activeThumbRef.value === startThumbRef.value?.$el ? [value, model.value[1]] : [model.value[0], value];
- if (!props.strict && newValue[0] < newValue[1]) {
- model.value = newValue;
- }
- emit('end', model.value);
- },
- onSliderMove: _ref3 => {
- let {
- value
- } = _ref3;
- const [start, stop] = model.value;
- if (!props.strict && start === stop && start !== min.value) {
- activeThumbRef.value = value > start ? stopThumbRef.value?.$el : startThumbRef.value?.$el;
- activeThumbRef.value?.focus();
- }
- if (activeThumbRef.value === startThumbRef.value?.$el) {
- model.value = [Math.min(value, stop), stop];
- } else {
- model.value = [start, Math.max(start, value)];
- }
- },
- getActiveThumb
- });
- const {
- isFocused,
- focus,
- blur
- } = useFocus(props);
- const trackStart = computed(() => position(model.value[0]));
- const trackStop = computed(() => position(model.value[1]));
- useRender(() => {
- const inputProps = VInput.filterProps(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
- }, rtlClasses.value, props.class],
- "style": props.style,
- "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: _ref4 => {
- let {
- id,
- messagesId
- } = _ref4;
- return _createVNode("div", {
- "class": "v-slider__container",
- "onMousedown": !readonly.value ? onSliderMousedown : undefined,
- "onTouchstartPassive": !readonly.value ? onSliderTouchstart : undefined
- }, [_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 (max.value !== min.value && 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,
- "ripple": props.ripple
- }, {
- '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 (max.value !== min.value && 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,
- "ripple": props.ripple
- }, {
- 'thumb-label': slots['thumb-label']
- })]);
- }
- });
- });
- return {};
- }
- });
- //# sourceMappingURL=VRangeSlider.mjs.map
|