mirror of
https://gitee.com/element-plus/element-plus.git
synced 2024-12-14 09:20:51 +08:00
16989d8187
* feat(components): [slider] aria keyboard controls and attrs fix #7350 * feat(components): [slider] fix lint error in slider types * feat(components): [slider] change start value for home/end unit test * feat(components): prevent scrolling on touch screen for runway * feat(components): [slider] type-o in locale prop * fix(components): [slider] PR comments r1 * fix(components): [slider] linting errors
211 lines
5.4 KiB
TypeScript
211 lines
5.4 KiB
TypeScript
import { computed, inject, nextTick, ref, shallowRef } from 'vue'
|
|
import {
|
|
CHANGE_EVENT,
|
|
INPUT_EVENT,
|
|
UPDATE_MODEL_EVENT,
|
|
} from '@element-plus/constants'
|
|
import { formContextKey, formItemContextKey } from '@element-plus/tokens'
|
|
import type { CSSProperties, ComponentInternalInstance, Ref } from 'vue'
|
|
import type {
|
|
ButtonRefs,
|
|
ISliderButton,
|
|
ISliderInitData,
|
|
ISliderProps,
|
|
} from './slider.type'
|
|
import type { FormContext, FormItemContext } from '@element-plus/tokens'
|
|
|
|
export const useSlide = (
|
|
props: ISliderProps,
|
|
initData: ISliderInitData,
|
|
emit: ComponentInternalInstance['emit']
|
|
) => {
|
|
const elForm = inject(formContextKey, {} as FormContext)
|
|
const elFormItem = inject(formItemContextKey, {} as FormItemContext)
|
|
|
|
const slider = shallowRef<HTMLElement>()
|
|
|
|
const firstButton = ref<ISliderButton>()
|
|
|
|
const secondButton = ref<ISliderButton>()
|
|
|
|
const buttonRefs: ButtonRefs = {
|
|
firstButton,
|
|
secondButton,
|
|
}
|
|
|
|
const sliderDisabled = computed(() => {
|
|
return props.disabled || elForm.disabled || false
|
|
})
|
|
|
|
const minValue = computed(() => {
|
|
return Math.min(initData.firstValue, initData.secondValue)
|
|
})
|
|
|
|
const maxValue = computed(() => {
|
|
return Math.max(initData.firstValue, initData.secondValue)
|
|
})
|
|
|
|
const barSize = computed(() => {
|
|
return props.range
|
|
? `${
|
|
(100 * (maxValue.value - minValue.value)) / (props.max - props.min)
|
|
}%`
|
|
: `${
|
|
(100 * (initData.firstValue - props.min)) / (props.max - props.min)
|
|
}%`
|
|
})
|
|
|
|
const barStart = computed(() => {
|
|
return props.range
|
|
? `${(100 * (minValue.value - props.min)) / (props.max - props.min)}%`
|
|
: '0%'
|
|
})
|
|
|
|
const runwayStyle = computed<CSSProperties>(() => {
|
|
return props.vertical ? { height: props.height } : {}
|
|
})
|
|
|
|
const barStyle = computed<CSSProperties>(() => {
|
|
return props.vertical
|
|
? {
|
|
height: barSize.value,
|
|
bottom: barStart.value,
|
|
}
|
|
: {
|
|
width: barSize.value,
|
|
left: barStart.value,
|
|
}
|
|
})
|
|
|
|
const resetSize = () => {
|
|
if (slider.value) {
|
|
initData.sliderSize =
|
|
slider.value[`client${props.vertical ? 'Height' : 'Width'}`]
|
|
}
|
|
}
|
|
|
|
const getButtonRefByPercent = (
|
|
percent: number
|
|
): Ref<ISliderButton | undefined> => {
|
|
const targetValue = props.min + (percent * (props.max - props.min)) / 100
|
|
if (!props.range) {
|
|
return firstButton
|
|
}
|
|
let buttonRefName: 'firstButton' | 'secondButton'
|
|
if (
|
|
Math.abs(minValue.value - targetValue) <
|
|
Math.abs(maxValue.value - targetValue)
|
|
) {
|
|
buttonRefName =
|
|
initData.firstValue < initData.secondValue
|
|
? 'firstButton'
|
|
: 'secondButton'
|
|
} else {
|
|
buttonRefName =
|
|
initData.firstValue > initData.secondValue
|
|
? 'firstButton'
|
|
: 'secondButton'
|
|
}
|
|
return buttonRefs[buttonRefName]
|
|
}
|
|
|
|
const setPosition = (percent: number): Ref<ISliderButton | undefined> => {
|
|
const buttonRef = getButtonRefByPercent(percent)
|
|
buttonRef.value!.setPosition(percent)
|
|
return buttonRef
|
|
}
|
|
|
|
const setFirstValue = (firstValue: number) => {
|
|
initData.firstValue = firstValue
|
|
_emit(props.range ? [minValue.value, maxValue.value] : firstValue)
|
|
}
|
|
|
|
const setSecondValue = (secondValue: number) => {
|
|
initData.secondValue = secondValue
|
|
|
|
if (props.range) {
|
|
_emit([minValue.value, maxValue.value])
|
|
}
|
|
}
|
|
|
|
const _emit = (val: number | number[]) => {
|
|
emit(UPDATE_MODEL_EVENT, val)
|
|
emit(INPUT_EVENT, val)
|
|
}
|
|
|
|
const emitChange = async () => {
|
|
await nextTick()
|
|
emit(
|
|
CHANGE_EVENT,
|
|
props.range ? [minValue.value, maxValue.value] : props.modelValue
|
|
)
|
|
}
|
|
|
|
const handleSliderPointerEvent = (
|
|
event: MouseEvent | TouchEvent
|
|
): Ref<ISliderButton | undefined> | undefined => {
|
|
if (sliderDisabled.value || initData.dragging) return
|
|
resetSize()
|
|
let newPercent = 0
|
|
if (props.vertical) {
|
|
const clientY =
|
|
(event as TouchEvent).touches?.item(0)?.clientY ??
|
|
(event as MouseEvent).clientY
|
|
const sliderOffsetBottom = slider.value!.getBoundingClientRect().bottom
|
|
newPercent = ((sliderOffsetBottom - clientY) / initData.sliderSize) * 100
|
|
} else {
|
|
const clientX =
|
|
(event as TouchEvent).touches?.item(0)?.clientX ??
|
|
(event as MouseEvent).clientX
|
|
const sliderOffsetLeft = slider.value!.getBoundingClientRect().left
|
|
newPercent = ((clientX - sliderOffsetLeft) / initData.sliderSize) * 100
|
|
}
|
|
if (newPercent < 0 || newPercent > 100) return
|
|
return setPosition(newPercent)
|
|
}
|
|
|
|
const onSliderWrapperPrevent = (event: TouchEvent) => {
|
|
if (
|
|
buttonRefs['firstButton'].value?.dragging ||
|
|
buttonRefs['secondButton'].value?.dragging
|
|
) {
|
|
event.preventDefault()
|
|
}
|
|
}
|
|
|
|
const onSliderDown = async (event: MouseEvent | TouchEvent) => {
|
|
const buttonRef = handleSliderPointerEvent(event)
|
|
if (buttonRef) {
|
|
await nextTick()
|
|
buttonRef.value!.onButtonDown(event)
|
|
}
|
|
}
|
|
|
|
const onSliderClick = (event: MouseEvent | TouchEvent) => {
|
|
const buttonRef = handleSliderPointerEvent(event)
|
|
if (buttonRef) {
|
|
emitChange()
|
|
}
|
|
}
|
|
|
|
return {
|
|
elFormItem,
|
|
slider,
|
|
firstButton,
|
|
secondButton,
|
|
sliderDisabled,
|
|
minValue,
|
|
maxValue,
|
|
runwayStyle,
|
|
barStyle,
|
|
resetSize,
|
|
setPosition,
|
|
emitChange,
|
|
onSliderWrapperPrevent,
|
|
onSliderClick,
|
|
onSliderDown,
|
|
setFirstValue,
|
|
setSecondValue,
|
|
}
|
|
}
|