mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
feat(Slider): add left-button、right-button slot (#8989)
This commit is contained in:
parent
a1d95e685c
commit
d72c7ddf48
@ -174,9 +174,11 @@ export default {
|
|||||||
|
|
||||||
### Slots
|
### Slots
|
||||||
|
|
||||||
| Name | Description |
|
| Name | Description | SlotProps |
|
||||||
| ------ | ------------- |
|
| --- | --- | --- |
|
||||||
| button | Custom button |
|
| button | Custom button | _{ value: number }_ |
|
||||||
|
| left-button `v3.1.3` | Custom left button in range mode | _{ value: number }_ |
|
||||||
|
| right-button `v3.1.3` | Custom right button in range mode | _{ value: number }_ |
|
||||||
|
|
||||||
### CSS Variables
|
### CSS Variables
|
||||||
|
|
||||||
|
@ -176,9 +176,11 @@ export default {
|
|||||||
|
|
||||||
### Slots
|
### Slots
|
||||||
|
|
||||||
| 名称 | 说明 |
|
| 名称 | 说明 | 参数 |
|
||||||
| ------ | -------------- |
|
| --- | --- | --- |
|
||||||
| button | 自定义滑动按钮 |
|
| button | 自定义滑块按钮 | _{ value: number }_ |
|
||||||
|
| left-button `v3.1.3` | 自定义左侧滑块按钮(双滑块模式下) | _{ value: number }_ |
|
||||||
|
| right-button `v3.1.3` | 自定义右侧滑块按钮 (双滑块模式下) | _{ value: number }_ |
|
||||||
|
|
||||||
### 样式变量
|
### 样式变量
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ import { useLinkField } from '../composables/use-link-field';
|
|||||||
|
|
||||||
const [name, bem] = createNamespace('slider');
|
const [name, bem] = createNamespace('slider');
|
||||||
|
|
||||||
type SliderValue = number | number[];
|
type SliderValue = number | [number, number];
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name,
|
name,
|
||||||
@ -53,7 +53,7 @@ export default defineComponent({
|
|||||||
emits: ['change', 'drag-end', 'drag-start', 'update:modelValue'],
|
emits: ['change', 'drag-end', 'drag-start', 'update:modelValue'],
|
||||||
|
|
||||||
setup(props, { emit, slots }) {
|
setup(props, { emit, slots }) {
|
||||||
let buttonIndex: number;
|
let buttonIndex: 0 | 1;
|
||||||
let startValue: SliderValue;
|
let startValue: SliderValue;
|
||||||
let currentValue: SliderValue;
|
let currentValue: SliderValue;
|
||||||
|
|
||||||
@ -71,8 +71,8 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
const isRange = (val: unknown): val is number[] =>
|
const isRange = (val: unknown): val is [number, number] =>
|
||||||
!!props.range && Array.isArray(val);
|
props.range && Array.isArray(val);
|
||||||
|
|
||||||
// 计算选中条的长度百分比
|
// 计算选中条的长度百分比
|
||||||
const calcMainAxis = () => {
|
const calcMainAxis = () => {
|
||||||
@ -117,7 +117,7 @@ export default defineComponent({
|
|||||||
JSON.stringify(newValue) === JSON.stringify(oldValue);
|
JSON.stringify(newValue) === JSON.stringify(oldValue);
|
||||||
|
|
||||||
// 处理两个滑块重叠之后的情况
|
// 处理两个滑块重叠之后的情况
|
||||||
const handleOverlap = (value: number[]) => {
|
const handleOverlap = (value: [number, number]) => {
|
||||||
if (value[0] > value[1]) {
|
if (value[0] > value[1]) {
|
||||||
return value.slice(0).reverse();
|
return value.slice(0).reverse();
|
||||||
}
|
}
|
||||||
@ -126,7 +126,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
const updateValue = (value: SliderValue, end?: boolean) => {
|
const updateValue = (value: SliderValue, end?: boolean) => {
|
||||||
if (isRange(value)) {
|
if (isRange(value)) {
|
||||||
value = handleOverlap(value).map(format);
|
value = handleOverlap(value).map(format) as [number, number];
|
||||||
} else {
|
} else {
|
||||||
value = format(value);
|
value = format(value);
|
||||||
}
|
}
|
||||||
@ -178,7 +178,7 @@ export default defineComponent({
|
|||||||
currentValue = props.modelValue;
|
currentValue = props.modelValue;
|
||||||
|
|
||||||
if (isRange(currentValue)) {
|
if (isRange(currentValue)) {
|
||||||
startValue = currentValue.map(format);
|
startValue = currentValue.map(format) as [number, number];
|
||||||
} else {
|
} else {
|
||||||
startValue = format(currentValue);
|
startValue = format(currentValue);
|
||||||
}
|
}
|
||||||
@ -205,7 +205,7 @@ export default defineComponent({
|
|||||||
const diff = (delta / total) * scope.value;
|
const diff = (delta / total) * scope.value;
|
||||||
|
|
||||||
if (isRange(startValue)) {
|
if (isRange(startValue)) {
|
||||||
(currentValue as number[])[buttonIndex] =
|
(currentValue as [number, number])[buttonIndex] =
|
||||||
startValue[buttonIndex] + diff;
|
startValue[buttonIndex] + diff;
|
||||||
} else {
|
} else {
|
||||||
currentValue = startValue + diff;
|
currentValue = startValue + diff;
|
||||||
@ -226,46 +226,59 @@ export default defineComponent({
|
|||||||
dragStatus.value = '';
|
dragStatus.value = '';
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderButton = (index?: number) => {
|
const getButtonClassName = (index?: 0 | 1) => {
|
||||||
const getClassName = () => {
|
|
||||||
if (typeof index === 'number') {
|
if (typeof index === 'number') {
|
||||||
const position = ['left', 'right'];
|
const position = ['left', 'right'];
|
||||||
return `button-wrapper-${position[index]}`;
|
return bem(`button-wrapper-${position[index]}`);
|
||||||
}
|
}
|
||||||
return `button-wrapper`;
|
return bem('button-wrapper');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const renderButtonContent = (value: number, index?: 0 | 1) => {
|
||||||
|
if (typeof index === 'number') {
|
||||||
|
const slot = slots[index === 0 ? 'left-button' : 'right-button'];
|
||||||
|
if (slot) {
|
||||||
|
return slot({ value });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (slots.button) {
|
||||||
|
return slots.button({ value });
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div class={bem('button')} style={getSizeStyle(props.buttonSize)} />
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderButton = (index?: 0 | 1) => {
|
||||||
const currentValue =
|
const currentValue =
|
||||||
typeof index === 'number'
|
typeof index === 'number'
|
||||||
? (props.modelValue as number[])[index]
|
? (props.modelValue as [number, number])[index]
|
||||||
: (props.modelValue as number);
|
: (props.modelValue as number);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
role="slider"
|
role="slider"
|
||||||
class={bem(getClassName())}
|
class={getButtonClassName(index)}
|
||||||
tabindex={props.disabled || props.readonly ? -1 : 0}
|
tabindex={props.disabled || props.readonly ? -1 : 0}
|
||||||
aria-valuemin={+props.min}
|
aria-valuemin={+props.min}
|
||||||
aria-valuenow={currentValue}
|
aria-valuenow={currentValue}
|
||||||
aria-valuemax={+props.max}
|
aria-valuemax={+props.max}
|
||||||
aria-orientation={props.vertical ? 'vertical' : 'horizontal'}
|
aria-orientation={props.vertical ? 'vertical' : 'horizontal'}
|
||||||
onTouchstart={(e) => {
|
onTouchstart={(event) => {
|
||||||
if (typeof index === 'number') {
|
if (typeof index === 'number') {
|
||||||
// save index of current button
|
// save index of current button
|
||||||
buttonIndex = index;
|
buttonIndex = index;
|
||||||
}
|
}
|
||||||
onTouchStart(e);
|
onTouchStart(event);
|
||||||
}}
|
}}
|
||||||
onTouchmove={onTouchMove}
|
onTouchmove={onTouchMove}
|
||||||
onTouchend={onTouchEnd}
|
onTouchend={onTouchEnd}
|
||||||
onTouchcancel={onTouchEnd}
|
onTouchcancel={onTouchEnd}
|
||||||
onClick={stopPropagation}
|
onClick={stopPropagation}
|
||||||
>
|
>
|
||||||
{slots.button ? (
|
{renderButtonContent(currentValue, index)}
|
||||||
slots.button()
|
|
||||||
) : (
|
|
||||||
<div class={bem('button')} style={getSizeStyle(props.buttonSize)} />
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
30
src/slider/test/__snapshots__/index.spec.ts.snap
Normal file
30
src/slider/test/__snapshots__/index.spec.ts.snap
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`should render left-button、right-button slot correctly 1`] = `
|
||||||
|
<div class="van-slider">
|
||||||
|
<div class="van-slider__bar"
|
||||||
|
style="width: 50%; left: 30%;"
|
||||||
|
>
|
||||||
|
<div role="slider"
|
||||||
|
class="van-slider__button-wrapper-left"
|
||||||
|
tabindex="0"
|
||||||
|
aria-valuemin="0"
|
||||||
|
aria-valuenow="30"
|
||||||
|
aria-valuemax="100"
|
||||||
|
aria-orientation="horizontal"
|
||||||
|
>
|
||||||
|
left-30
|
||||||
|
</div>
|
||||||
|
<div role="slider"
|
||||||
|
class="van-slider__button-wrapper-right"
|
||||||
|
tabindex="0"
|
||||||
|
aria-valuemin="0"
|
||||||
|
aria-valuenow="80"
|
||||||
|
aria-valuemax="100"
|
||||||
|
aria-orientation="horizontal"
|
||||||
|
>
|
||||||
|
right-80
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
@ -203,3 +203,19 @@ test('should format v-model with step correctly', async () => {
|
|||||||
await later();
|
await later();
|
||||||
expect(wrapper.emitted('update:modelValue')![0]).toEqual([31]);
|
expect(wrapper.emitted('update:modelValue')![0]).toEqual([31]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should render left-button、right-button slot correctly', async () => {
|
||||||
|
const wrapper = mount(Slider, {
|
||||||
|
props: {
|
||||||
|
range: true,
|
||||||
|
modelValue: [30, 80],
|
||||||
|
},
|
||||||
|
slots: {
|
||||||
|
'left-button': ({ value }) => `left-${value}`,
|
||||||
|
'right-button': ({ value }) => `right-${value}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
expect(wrapper.html()).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
Loading…
x
Reference in New Issue
Block a user