element-plus/packages/components/slider/src/useSlide.ts
opengraphica 16989d8187
feat(components): [slider] aria keyboard controls and attrs (#7389)
* 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
2022-04-27 21:38:47 +08:00

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,
}
}