ant-design-vue/components/vc-slider/src/Slider.jsx

198 lines
5.1 KiB
Vue

import PropTypes from '../../_util/vue-types';
import BaseMixin from '../../_util/BaseMixin';
import { hasProp } from '../../_util/props-util';
import Track from './common/Track';
import createSlider from './common/createSlider';
import * as utils from './utils';
const Slider = {
name: 'Slider',
inheritAttrs: false,
mixins: [BaseMixin],
props: {
defaultValue: PropTypes.number,
value: PropTypes.number,
disabled: PropTypes.bool,
autofocus: PropTypes.bool,
tabindex: PropTypes.number,
reverse: PropTypes.bool,
min: PropTypes.number,
max: PropTypes.number,
},
data() {
const defaultValue = this.defaultValue !== undefined ? this.defaultValue : this.min;
const value = this.value !== undefined ? this.value : defaultValue;
return {
sValue: this.trimAlignValue(value),
dragging: false,
};
},
watch: {
value: {
handler(val) {
this.setChangeValue(val);
},
deep: true,
},
min() {
const { sValue } = this;
this.setChangeValue(sValue);
},
max() {
const { sValue } = this;
this.setChangeValue(sValue);
},
},
methods: {
setChangeValue(value) {
const newValue = value !== undefined ? value : this.sValue;
const nextValue = this.trimAlignValue(newValue, this.$props);
if (nextValue === this.sValue) return;
this.setState({ sValue: nextValue });
if (utils.isValueOutOfRange(newValue, this.$props)) {
this.__emit('change', nextValue);
}
},
onChange(state) {
const isNotControlled = !hasProp(this, 'value');
const nextState = state.sValue > this.max ? { ...state, sValue: this.max } : state;
if (isNotControlled) {
this.setState(nextState);
}
const changedValue = nextState.sValue;
this.__emit('change', changedValue);
},
onStart(position) {
this.setState({ dragging: true });
const { sValue } = this;
this.__emit('beforeChange', sValue);
const value = this.calcValueByPos(position);
this.startValue = value;
this.startPosition = position;
if (value === sValue) return;
this.prevMovedHandleIndex = 0;
this.onChange({ sValue: value });
},
onEnd(force) {
const { dragging } = this;
this.removeDocumentEvents();
if (dragging || force) {
this.__emit('afterChange', this.sValue);
}
this.setState({ dragging: false });
},
onMove(e, position) {
utils.pauseEvent(e);
const { sValue } = this;
const value = this.calcValueByPos(position);
if (value === sValue) return;
this.onChange({ sValue: value });
},
onKeyboard(e) {
const { reverse, vertical } = this.$props;
const valueMutator = utils.getKeyboardValueMutator(e, vertical, reverse);
if (valueMutator) {
utils.pauseEvent(e);
const { sValue } = this;
const mutatedValue = valueMutator(sValue, this.$props);
const value = this.trimAlignValue(mutatedValue);
if (value === sValue) return;
this.onChange({ sValue: value });
this.__emit('afterChange', value);
this.onEnd();
}
},
getLowerBound() {
return this.min;
},
getUpperBound() {
return this.sValue;
},
trimAlignValue(v, nextProps = {}) {
if (v === null) {
return null;
}
const mergedProps = { ...this.$props, ...nextProps };
const val = utils.ensureValueInRange(v, mergedProps);
return utils.ensureValuePrecision(val, mergedProps);
},
getTrack({ prefixCls, reverse, vertical, included, offset, minimumTrackStyle, _trackStyle }) {
return (
<Track
class={`${prefixCls}-track`}
vertical={vertical}
included={included}
offset={0}
reverse={reverse}
length={offset}
style={{
...minimumTrackStyle,
..._trackStyle,
}}
/>
);
},
renderSlider() {
const {
prefixCls,
vertical,
included,
disabled,
minimumTrackStyle,
trackStyle,
handleStyle,
tabindex,
min,
max,
reverse,
handle,
defaultHandle,
} = this;
const handleGenerator = handle || defaultHandle;
const { sValue, dragging } = this;
const offset = this.calcOffset(sValue);
const handles = handleGenerator({
class: `${prefixCls}-handle`,
prefixCls,
vertical,
offset,
value: sValue,
dragging,
disabled,
min,
max,
reverse,
index: 0,
tabindex,
style: handleStyle[0] || handleStyle,
ref: h => this.saveHandle(0, h),
onFocus: this.onFocus,
onBlur: this.onBlur,
});
const _trackStyle = trackStyle[0] || trackStyle;
return {
tracks: this.getTrack({
prefixCls,
reverse,
vertical,
included,
offset,
minimumTrackStyle,
_trackStyle,
}),
handles,
};
},
},
};
export default createSlider(Slider);