VRangeSlider.mjs 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. import { mergeProps as _mergeProps, createVNode as _createVNode, Fragment as _Fragment } from "vue";
  2. // Styles
  3. import "../VSlider/VSlider.css";
  4. // Components
  5. import { makeVInputProps, VInput } from "../VInput/VInput.mjs";
  6. import { VLabel } from "../VLabel/index.mjs";
  7. import { getOffset, makeSliderProps, useSlider, useSteps } from "../VSlider/slider.mjs";
  8. import { VSliderThumb } from "../VSlider/VSliderThumb.mjs";
  9. import { VSliderTrack } from "../VSlider/VSliderTrack.mjs"; // Composables
  10. import { makeFocusProps, useFocus } from "../../composables/focus.mjs";
  11. import { useRtl } from "../../composables/locale.mjs";
  12. import { useProxiedModel } from "../../composables/proxiedModel.mjs"; // Utilities
  13. import { computed, ref } from 'vue';
  14. import { genericComponent, propsFactory, useRender } from "../../util/index.mjs"; // Types
  15. export const makeVRangeSliderProps = propsFactory({
  16. ...makeFocusProps(),
  17. ...makeVInputProps(),
  18. ...makeSliderProps(),
  19. strict: Boolean,
  20. modelValue: {
  21. type: Array,
  22. default: () => [0, 0]
  23. }
  24. }, 'VRangeSlider');
  25. export const VRangeSlider = genericComponent()({
  26. name: 'VRangeSlider',
  27. props: makeVRangeSliderProps(),
  28. emits: {
  29. 'update:focused': value => true,
  30. 'update:modelValue': value => true,
  31. end: value => true,
  32. start: value => true
  33. },
  34. setup(props, _ref) {
  35. let {
  36. slots,
  37. emit
  38. } = _ref;
  39. const startThumbRef = ref();
  40. const stopThumbRef = ref();
  41. const inputRef = ref();
  42. const {
  43. rtlClasses
  44. } = useRtl();
  45. function getActiveThumb(e) {
  46. if (!startThumbRef.value || !stopThumbRef.value) return;
  47. const startOffset = getOffset(e, startThumbRef.value.$el, props.direction);
  48. const stopOffset = getOffset(e, stopThumbRef.value.$el, props.direction);
  49. const a = Math.abs(startOffset);
  50. const b = Math.abs(stopOffset);
  51. return a < b || a === b && startOffset < 0 ? startThumbRef.value.$el : stopThumbRef.value.$el;
  52. }
  53. const steps = useSteps(props);
  54. const model = useProxiedModel(props, 'modelValue', undefined, arr => {
  55. if (!arr?.length) return [0, 0];
  56. return arr.map(value => steps.roundValue(value));
  57. });
  58. const {
  59. activeThumbRef,
  60. hasLabels,
  61. max,
  62. min,
  63. mousePressed,
  64. onSliderMousedown,
  65. onSliderTouchstart,
  66. position,
  67. trackContainerRef,
  68. readonly
  69. } = useSlider({
  70. props,
  71. steps,
  72. onSliderStart: () => {
  73. emit('start', model.value);
  74. },
  75. onSliderEnd: _ref2 => {
  76. let {
  77. value
  78. } = _ref2;
  79. const newValue = activeThumbRef.value === startThumbRef.value?.$el ? [value, model.value[1]] : [model.value[0], value];
  80. if (!props.strict && newValue[0] < newValue[1]) {
  81. model.value = newValue;
  82. }
  83. emit('end', model.value);
  84. },
  85. onSliderMove: _ref3 => {
  86. let {
  87. value
  88. } = _ref3;
  89. const [start, stop] = model.value;
  90. if (!props.strict && start === stop && start !== min.value) {
  91. activeThumbRef.value = value > start ? stopThumbRef.value?.$el : startThumbRef.value?.$el;
  92. activeThumbRef.value?.focus();
  93. }
  94. if (activeThumbRef.value === startThumbRef.value?.$el) {
  95. model.value = [Math.min(value, stop), stop];
  96. } else {
  97. model.value = [start, Math.max(start, value)];
  98. }
  99. },
  100. getActiveThumb
  101. });
  102. const {
  103. isFocused,
  104. focus,
  105. blur
  106. } = useFocus(props);
  107. const trackStart = computed(() => position(model.value[0]));
  108. const trackStop = computed(() => position(model.value[1]));
  109. useRender(() => {
  110. const inputProps = VInput.filterProps(props);
  111. const hasPrepend = !!(props.label || slots.label || slots.prepend);
  112. return _createVNode(VInput, _mergeProps({
  113. "class": ['v-slider', 'v-range-slider', {
  114. 'v-slider--has-labels': !!slots['tick-label'] || hasLabels.value,
  115. 'v-slider--focused': isFocused.value,
  116. 'v-slider--pressed': mousePressed.value,
  117. 'v-slider--disabled': props.disabled
  118. }, rtlClasses.value, props.class],
  119. "style": props.style,
  120. "ref": inputRef
  121. }, inputProps, {
  122. "focused": isFocused.value
  123. }), {
  124. ...slots,
  125. prepend: hasPrepend ? slotProps => _createVNode(_Fragment, null, [slots.label?.(slotProps) ?? (props.label ? _createVNode(VLabel, {
  126. "class": "v-slider__label",
  127. "text": props.label
  128. }, null) : undefined), slots.prepend?.(slotProps)]) : undefined,
  129. default: _ref4 => {
  130. let {
  131. id,
  132. messagesId
  133. } = _ref4;
  134. return _createVNode("div", {
  135. "class": "v-slider__container",
  136. "onMousedown": !readonly.value ? onSliderMousedown : undefined,
  137. "onTouchstartPassive": !readonly.value ? onSliderTouchstart : undefined
  138. }, [_createVNode("input", {
  139. "id": `${id.value}_start`,
  140. "name": props.name || id.value,
  141. "disabled": !!props.disabled,
  142. "readonly": !!props.readonly,
  143. "tabindex": "-1",
  144. "value": model.value[0]
  145. }, null), _createVNode("input", {
  146. "id": `${id.value}_stop`,
  147. "name": props.name || id.value,
  148. "disabled": !!props.disabled,
  149. "readonly": !!props.readonly,
  150. "tabindex": "-1",
  151. "value": model.value[1]
  152. }, null), _createVNode(VSliderTrack, {
  153. "ref": trackContainerRef,
  154. "start": trackStart.value,
  155. "stop": trackStop.value
  156. }, {
  157. 'tick-label': slots['tick-label']
  158. }), _createVNode(VSliderThumb, {
  159. "ref": startThumbRef,
  160. "aria-describedby": messagesId.value,
  161. "focused": isFocused && activeThumbRef.value === startThumbRef.value?.$el,
  162. "modelValue": model.value[0],
  163. "onUpdate:modelValue": v => model.value = [v, model.value[1]],
  164. "onFocus": e => {
  165. focus();
  166. activeThumbRef.value = startThumbRef.value?.$el;
  167. // Make sure second thumb is focused if
  168. // the thumbs are on top of each other
  169. // and they are both at minimum value
  170. // but only if focused from outside.
  171. if (max.value !== min.value && model.value[0] === model.value[1] && model.value[1] === min.value && e.relatedTarget !== stopThumbRef.value?.$el) {
  172. startThumbRef.value?.$el.blur();
  173. stopThumbRef.value?.$el.focus();
  174. }
  175. },
  176. "onBlur": () => {
  177. blur();
  178. activeThumbRef.value = undefined;
  179. },
  180. "min": min.value,
  181. "max": model.value[1],
  182. "position": trackStart.value,
  183. "ripple": props.ripple
  184. }, {
  185. 'thumb-label': slots['thumb-label']
  186. }), _createVNode(VSliderThumb, {
  187. "ref": stopThumbRef,
  188. "aria-describedby": messagesId.value,
  189. "focused": isFocused && activeThumbRef.value === stopThumbRef.value?.$el,
  190. "modelValue": model.value[1],
  191. "onUpdate:modelValue": v => model.value = [model.value[0], v],
  192. "onFocus": e => {
  193. focus();
  194. activeThumbRef.value = stopThumbRef.value?.$el;
  195. // Make sure first thumb is focused if
  196. // the thumbs are on top of each other
  197. // and they are both at maximum value
  198. // but only if focused from outside.
  199. if (max.value !== min.value && model.value[0] === model.value[1] && model.value[0] === max.value && e.relatedTarget !== startThumbRef.value?.$el) {
  200. stopThumbRef.value?.$el.blur();
  201. startThumbRef.value?.$el.focus();
  202. }
  203. },
  204. "onBlur": () => {
  205. blur();
  206. activeThumbRef.value = undefined;
  207. },
  208. "min": model.value[0],
  209. "max": max.value,
  210. "position": trackStop.value,
  211. "ripple": props.ripple
  212. }, {
  213. 'thumb-label': slots['thumb-label']
  214. })]);
  215. }
  216. });
  217. });
  218. return {};
  219. }
  220. });
  221. //# sourceMappingURL=VRangeSlider.mjs.map