ant-design-vue/components/slider/index.tsx

206 lines
6.6 KiB
Vue
Raw Normal View History

2021-08-28 08:36:30 +00:00
import type { CSSProperties, VNodeTypes } from 'vue';
import { computed, ref } from 'vue';
2021-08-27 14:05:21 +00:00
import { defineComponent } from 'vue';
2019-01-12 03:33:27 +00:00
import BaseMixin from '../_util/BaseMixin';
import VcSlider from '../vc-slider/src/Slider';
import VcRange from '../vc-slider/src/Range';
import VcHandle from '../vc-slider/src/Handle';
2021-08-28 08:36:30 +00:00
import type { VueNode } from '../_util/type';
import { withInstall } from '../_util/type';
import type { PropType } from 'vue';
import type { TooltipPlacement } from '../tooltip/Tooltip';
2021-08-27 14:05:21 +00:00
import useConfigInject from '../_util/hooks/useConfigInject';
import SliderTooltip from './SliderTooltip';
import classNames from '../_util/classNames';
2018-04-01 11:23:56 +00:00
2020-10-13 14:59:24 +00:00
export type SliderValue = number | [number, number];
2021-08-27 14:05:21 +00:00
interface SliderMarks {
[key: number]:
| VueNode
| {
style: CSSProperties;
label: any;
};
}
2020-10-13 14:59:24 +00:00
interface HandleGeneratorInfo {
2021-08-27 14:05:21 +00:00
value?: number;
dragging?: boolean;
2020-10-13 14:59:24 +00:00
index: number;
2021-08-27 14:05:21 +00:00
}
interface SliderRange {
draggableTrack?: boolean;
2020-10-13 14:59:24 +00:00
}
export type HandleGeneratorFn = (config: {
tooltipPrefixCls?: string;
prefixCls?: string;
info: HandleGeneratorInfo;
}) => VNodeTypes;
2021-08-27 14:05:21 +00:00
type Value = [number, number] | number;
2020-10-13 14:59:24 +00:00
2021-08-27 14:05:21 +00:00
const defaultTipFormatter = (value: number) => (typeof value === 'number' ? value.toString() : '');
2021-08-28 08:36:30 +00:00
export const sliderProps = () => ({
2021-08-27 14:05:21 +00:00
prefixCls: String,
tooltipPrefixCls: String,
range: { type: [Boolean, Object] as PropType<boolean | SliderRange>, default: undefined },
reverse: { type: Boolean, default: undefined },
min: Number,
max: Number,
step: { type: [Number, Object] as PropType<null | number> },
marks: { type: Object as PropType<SliderMarks> },
dots: { type: Boolean, default: undefined },
value: { type: [Number, Array] as PropType<Value> },
defaultValue: { type: [Number, Array] as PropType<Value> },
included: { type: Boolean, default: undefined },
disabled: { type: Boolean, default: undefined },
vertical: { type: Boolean, default: undefined },
tipFormatter: {
type: Function as PropType<(value?: number) => any>,
default: defaultTipFormatter,
},
tooltipVisible: { type: Boolean, default: undefined },
tooltipPlacement: { type: String as PropType<TooltipPlacement> },
getTooltipPopupContainer: {
type: Function as PropType<(triggerNode: HTMLElement) => HTMLElement>,
},
autofocus: { type: Boolean, default: undefined },
onChange: { type: Function as PropType<(value: Value) => void> },
onAfterChange: { type: Function as PropType<(value: Value) => void> },
handleStyle: { type: [Object, Array] as PropType<CSSProperties[] | CSSProperties> },
trackStyle: { type: [Object, Array] as PropType<CSSProperties[] | CSSProperties> },
2019-01-12 03:33:27 +00:00
});
2018-04-01 11:23:56 +00:00
2021-08-27 14:05:21 +00:00
export type Visibles = { [index: number]: boolean };
2020-07-20 14:47:06 +00:00
2020-10-13 14:59:24 +00:00
const Slider = defineComponent({
2018-04-08 13:17:20 +00:00
name: 'ASlider',
2020-11-07 07:37:17 +00:00
mixins: [BaseMixin],
2020-07-20 14:47:06 +00:00
inheritAttrs: false,
2020-11-07 13:19:50 +00:00
props: {
2021-08-28 08:36:30 +00:00
...sliderProps(),
2020-11-07 13:19:50 +00:00
},
2021-08-27 14:05:21 +00:00
emits: ['update:value', 'change', 'afterChange'],
slots: ['mark'],
setup(props, { attrs, slots, emit, expose }) {
const { prefixCls, rootPrefixCls, direction, getPopupContainer, configProvider } =
useConfigInject('slider', props);
const sliderRef = ref();
const visibles = ref<Visibles>({});
const toggleTooltipVisible = (index: number, visible: boolean) => {
visibles.value[index] = visible;
2020-07-20 14:47:06 +00:00
};
2021-08-27 14:05:21 +00:00
const tooltipPlacement = computed(() => {
if (props.tooltipPlacement) {
return props.tooltipPlacement;
}
if (!props.vertical) {
return 'top';
}
return direction.value === 'rtl' ? 'left' : 'right';
});
const focus = () => {
sliderRef.value?.focus();
2019-01-12 03:33:27 +00:00
};
2021-08-27 14:05:21 +00:00
const blur = () => {
sliderRef.value?.blur();
};
const handleChange = (val: SliderValue) => {
emit('update:value', val);
emit('change', val);
};
expose({
focus,
blur,
});
const handleWithTooltip: HandleGeneratorFn = ({
tooltipPrefixCls,
info: { value, dragging, index, ...restProps },
}) => {
const { tipFormatter, tooltipVisible, getTooltipPopupContainer } = props;
const isTipFormatter = tipFormatter ? visibles.value[index] || dragging : false;
2019-01-12 03:33:27 +00:00
const visible = tooltipVisible || (tooltipVisible === undefined && isTipFormatter);
2018-04-01 11:23:56 +00:00
return (
2021-08-27 14:05:21 +00:00
<SliderTooltip
prefixCls={tooltipPrefixCls}
title={tipFormatter ? tipFormatter(value) : ''}
visible={visible}
placement={tooltipPlacement.value}
transitionName={`${rootPrefixCls.value}-zoom-down`}
key={index}
overlayClassName={`${prefixCls.value}-tooltip`}
getPopupContainer={getTooltipPopupContainer || getPopupContainer.value}
>
<VcHandle
{...restProps}
value={value}
onMouseenter={() => toggleTooltipVisible(index, true)}
onMouseleave={() => toggleTooltipVisible(index, false)}
/>
</SliderTooltip>
);
};
return () => {
const { tooltipPrefixCls: customizeTooltipPrefixCls, range, ...restProps } = props;
const tooltipPrefixCls = configProvider.getPrefixCls('tooltip', customizeTooltipPrefixCls);
const cls = classNames(attrs.class, {
[`${prefixCls.value}-rtl`]: direction.value === 'rtl',
});
// make reverse default on rtl direction
if (direction.value === 'rtl' && !restProps.vertical) {
restProps.reverse = !restProps.reverse;
}
// extrack draggableTrack from range={{ ... }}
let draggableTrack: boolean | undefined;
if (typeof range === 'object') {
draggableTrack = range.draggableTrack;
}
if (range) {
return (
<VcRange
{...restProps}
step={restProps.step!}
draggableTrack={draggableTrack}
class={cls}
ref={ref}
handle={(info: HandleGeneratorInfo) =>
handleWithTooltip({
tooltipPrefixCls,
prefixCls: prefixCls.value,
info,
})
}
prefixCls={prefixCls.value}
onChange={handleChange}
v-slots={{ mark: slots.mark }}
/>
);
}
return (
<VcSlider
{...restProps}
step={restProps.step!}
class={cls}
ref={ref}
handle={(info: HandleGeneratorInfo) =>
handleWithTooltip({
tooltipPrefixCls,
prefixCls: prefixCls.value,
info,
})
}
prefixCls={prefixCls.value}
onChange={handleChange}
v-slots={{ mark: slots.mark }}
/>
2019-01-12 03:33:27 +00:00
);
};
2018-04-01 11:23:56 +00:00
},
2020-10-13 14:59:24 +00:00
});
2020-11-01 07:03:33 +00:00
export default withInstall(Slider);