123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475 |
- <template>
- <div
- class="el-slider"
- :class="{ 'is-vertical': vertical, 'el-slider--with-input': showInput }"
- role="slider"
- :aria-valuemin="min"
- :aria-valuemax="max"
- :aria-orientation="vertical ? 'vertical' : 'horizontal'"
- :aria-disabled="sliderDisabled">
- <el-input-number
- v-model="firstValue"
- v-if="showInput && !range"
- class="el-slider__input"
- ref="input"
- @change="emitChange"
- :step="step"
- :disabled="sliderDisabled"
- :controls="showInputControls"
- :min="min"
- :max="max"
- :debounce="debounce"
- :size="inputSize">
- </el-input-number>
- <div
- class="el-slider__runway"
- :class="{ 'show-input': showInput, disabled: sliderDisabled }"
- :style="runwayStyle"
- @click="onSliderClick"
- ref="slider">
- <div
- class="el-slider__bar"
- :style="barStyle"></div>
- <slider-button
- :vertical="vertical"
- :marks="marks"
- v-model="firstValue"
- :tooltip-class="tooltipClass"
- ref="button1">
- </slider-button>
- <slider-button
- :vertical="vertical"
- :marks="marks"
- v-model="secondValue"
- :tooltip-class="tooltipClass"
- ref="button2"
- v-if="range">
- </slider-button>
- <div
- class="el-slider__stop"
- v-for="(item, key) in stops"
- :key="key"
- :style="getStopStyle(item)"
- v-show="showStops"></div>
- <template v-if="markList.length > 0">
- <div>
- <div
- v-for="(item, key) in markList"
- :style="getStopStyle(item.position)"
- @click.stop="onMarkClick(key)"
- class="el-slider__stop el-slider__marks-stop"
- :key="key"></div>
- </div>
- <div class="el-slider__marks">
- <slider-marker
- :mark="item.mark"
- v-for="(item, key) in markList"
- @click="onMarkClick(key)"
- :key="key"
- :style="getStopStyle(item.position)">
- </slider-marker>
- </div>
- </template>
- </div>
- </div>
- </template>
- <script type="text/babel">
- import ElInputNumber from 'element-ui/packages/input-number'
- import Emitter from 'element-ui/src/mixins/emitter'
- import SliderButton from './button.vue'
- import SliderMarker from './marker.vue'
- export default {
- name: 'ElSlider',
- components: {
- ElInputNumber,
- SliderButton,
- SliderMarker,
- },
- mixins: [Emitter],
- inject: {
- elForm: {
- default: '',
- },
- },
- props: {
- min: {
- type: Number,
- default: 0,
- },
- max: {
- type: Number,
- default: 100,
- },
- step: {
- type: Number,
- default: 1,
- },
- value: {
- // string 是数据框被清空时的值. 理论上不会出现, 这里只是用来规避控制台报错
- type: [Number, Array, String],
- default: 0,
- },
- showInput: {
- type: Boolean,
- default: false,
- },
- showInputControls: {
- type: Boolean,
- default: true,
- },
- inputSize: {
- type: String,
- default: 'small',
- },
- showStops: {
- type: Boolean,
- default: false,
- },
- showTooltip: {
- type: Boolean,
- default: true,
- },
- formatTooltip: Function,
- disabled: {
- type: Boolean,
- default: false,
- },
- range: {
- type: Boolean,
- default: false,
- },
- vertical: {
- type: Boolean,
- default: false,
- },
- height: {
- type: String,
- },
- debounce: {
- type: Number,
- default: 300,
- },
- label: {
- type: String,
- },
- tooltipClass: String,
- marks: Object,
- },
- data() {
- return {
- firstValue: null,
- secondValue: null,
- oldValue: null,
- dragging: false,
- sliderSize: 1,
- }
- },
- computed: {
- stops() {
- if (!this.showStops || this.min > this.max) return []
- if (this.step === 0) {
- process.env.NODE_ENV !== 'production' &&
- console.warn('[Element Warn][Slider]step should not be 0.')
- return []
- }
- const stopCount = (this.max - this.min) / this.step
- const stepWidth = (100 * this.step) / (this.max - this.min)
- const result = []
- for (let i = 1; i < stopCount; i++) {
- result.push(i * stepWidth)
- }
- if (this.range) {
- return result.filter(step => {
- return (
- step < (100 * (this.minValue - this.min)) / (this.max - this.min) ||
- step > (100 * (this.maxValue - this.min)) / (this.max - this.min)
- )
- })
- } else {
- return result.filter(
- step =>
- step > (100 * (this.firstValue - this.min)) / (this.max - this.min)
- )
- }
- },
- markList() {
- if (!this.marks) {
- return []
- }
- const marksKeys = Object.keys(this.marks)
- return marksKeys
- .map(parseFloat)
- .sort((a, b) => a - b)
- .filter(point => point <= this.max && point >= this.min)
- .map((point, index) => ({
- point,
- // 改造使得mark在滑动区域内均分
- position:
- (100 * index * (this.max - this.min)) /
- (marksKeys.length - 1) /
- (this.max - this.min),
- // position: ((point - this.min) * 100) / (this.max - this.min),
- mark: this.marks[point],
- }))
- },
- minValue() {
- return Math.min(this.firstValue, this.secondValue)
- },
- maxValue() {
- return Math.max(this.firstValue, this.secondValue)
- },
- barSize() {
- if (this.marks) {
- const marks = Object.keys(this.marks).map(Number)
- // 找出目标值位于marks的那个档位, 即 即将到达哪一个'档'
- const index = marks.findIndex(i => this.firstValue < i)
- // console.log(index, 'index')
- if (this.range) {
- // todo 范围值的计算. 这个还没改. 不知道怎么算.
- return `${
- (100 * (this.maxValue - this.minValue)) / (this.max - this.min)
- }%`
- }
- if (index === -1) return '100%'
- // 经过后一个档位之后占总进度条的比例 + (当前值超出后一个档位的差值 / 前一个档位与后一个档位的差值) / (进度被marks划分成的档位数量)
- // 上述公式要除档位数量是因为, 当前值与档位差值占单个档位区间的百分比 !== 单个区间值占总进度条的百分比
- const value =
- (index - 1) / (marks.length - 1) +
- (this.firstValue - marks[index - 1]) /
- (marks[index] - marks[index - 1]) /
- (marks.length - 1)
- // console.log('value', value)
- return value > 1 ? '100%' : `${value * 100}%`
- } else {
- return this.range
- ? `${
- (100 * (this.maxValue - this.minValue)) / (this.max - this.min)
- }%`
- : `${(100 * (this.firstValue - this.min)) / (this.max - this.min)}%`
- }
- },
- barStart() {
- return this.range
- ? `${(100 * (this.minValue - this.min)) / (this.max - this.min)}%`
- : '0%'
- },
- precision() {
- const precisions = [this.min, this.max, this.step].map(item => {
- const decimal = ('' + item).split('.')[1]
- return decimal ? decimal.length : 0
- })
- return Math.max.apply(null, precisions)
- },
- runwayStyle() {
- return this.vertical ? { height: this.height } : {}
- },
- barStyle() {
- return this.vertical
- ? {
- height: this.barSize,
- bottom: this.barStart,
- }
- : {
- width: this.barSize,
- left: this.barStart,
- }
- },
- sliderDisabled() {
- return this.disabled || (this.elForm || {}).disabled
- },
- },
- watch: {
- value(val, oldVal) {
- if (
- this.dragging ||
- (Array.isArray(val) &&
- Array.isArray(oldVal) &&
- val.every((item, index) => item === oldVal[index]))
- ) {
- return
- }
- this.setValues()
- },
- dragging(val) {
- if (!val) {
- this.setValues()
- }
- },
- firstValue(val) {
- if (this.range) {
- this.$emit('input', [this.minValue, this.maxValue])
- } else {
- this.$emit('input', val)
- }
- },
- secondValue() {
- if (this.range) {
- this.$emit('input', [this.minValue, this.maxValue])
- }
- },
- min() {
- this.setValues()
- },
- max() {
- this.setValues()
- },
- },
- mounted() {
- let valuetext
- if (this.range) {
- if (Array.isArray(this.value)) {
- this.firstValue = Math.max(this.min, this.value[0])
- this.secondValue = Math.min(this.max, this.value[1])
- } else {
- this.firstValue = this.min
- this.secondValue = this.max
- }
- this.oldValue = [this.firstValue, this.secondValue]
- valuetext = `${this.firstValue}-${this.secondValue}`
- } else {
- if (typeof this.value !== 'number' || isNaN(this.value)) {
- this.firstValue = this.min
- } else {
- this.firstValue = Math.min(this.max, Math.max(this.min, this.value))
- }
- this.oldValue = this.firstValue
- valuetext = this.firstValue
- }
- this.$el.setAttribute('aria-valuetext', valuetext)
- // label screen reader
- this.$el.setAttribute(
- 'aria-label',
- this.label ? this.label : `slider between ${this.min} and ${this.max}`
- )
- this.resetSize()
- window.addEventListener('resize', this.resetSize)
- },
- beforeDestroy() {
- window.removeEventListener('resize', this.resetSize)
- },
- methods: {
- valueChanged() {
- if (this.range) {
- return ![this.minValue, this.maxValue].every(
- (item, index) => item === this.oldValue[index]
- )
- } else {
- return this.value !== this.oldValue
- }
- },
- setValues() {
- if (this.min > this.max) {
- console.error(
- '[Element Error][Slider]min should not be greater than max.'
- )
- return
- }
- const val = this.value
- if (this.range && Array.isArray(val)) {
- if (val[1] < this.min) {
- this.$emit('input', [this.min, this.min])
- } else if (val[0] > this.max) {
- this.$emit('input', [this.max, this.max])
- } else if (val[0] < this.min) {
- this.$emit('input', [this.min, val[1]])
- } else if (val[1] > this.max) {
- this.$emit('input', [val[0], this.max])
- } else {
- this.firstValue = val[0]
- this.secondValue = val[1]
- if (this.valueChanged()) {
- this.dispatch('ElFormItem', 'el.form.change', [
- this.minValue,
- this.maxValue,
- ])
- this.oldValue = val.slice()
- }
- }
- } else if (!this.range && typeof val === 'number' && !isNaN(val)) {
- if (val < this.min) {
- this.$emit('input', this.min)
- } else if (val > this.max) {
- this.$emit('input', this.max)
- } else {
- this.firstValue = val
- if (this.valueChanged()) {
- this.dispatch('ElFormItem', 'el.form.change', val)
- this.oldValue = val
- }
- }
- }
- },
- setPosition(percent) {
- const targetValue = this.min + (percent * (this.max - this.min)) / 100
- if (!this.range) {
- this.$refs.button1.setPosition(percent)
- return
- }
- let button
- if (
- Math.abs(this.minValue - targetValue) <
- Math.abs(this.maxValue - targetValue)
- ) {
- button = this.firstValue < this.secondValue ? 'button1' : 'button2'
- } else {
- button = this.firstValue > this.secondValue ? 'button1' : 'button2'
- }
- this.$refs[button].setPosition(percent)
- },
- onSliderClick(event) {
- if (this.sliderDisabled || this.dragging) return
- this.resetSize()
- if (this.vertical) {
- const sliderOffsetBottom =
- this.$refs.slider.getBoundingClientRect().bottom
- this.setPosition(
- ((sliderOffsetBottom - event.clientY) / this.sliderSize) * 100
- )
- } else {
- const sliderOffsetLeft = this.$refs.slider.getBoundingClientRect().left
- this.setPosition(
- ((event.clientX - sliderOffsetLeft) / this.sliderSize) * 100
- )
- }
- this.emitChange()
- },
- onMarkClick(index) {
- if (typeof index !== 'number') return
- // 这个判断和resetSize从onSliderClick抄过来的.
- // 因为直接点击mark的 标志文案 得到的不是准确的数(宽度导致的位置不同), 所以魔改了一下, 点击mark按固定比例算.
- if (this.sliderDisabled || this.dragging) return
- this.resetSize()
- const marks = Object.keys(this.marks).map(Number)
- // 计算当前点击位于marks的第几项, 算百分比
- this.setPosition((100 * index) / (marks.length - 1))
- this.emitChange()
- },
- resetSize() {
- if (this.$refs.slider) {
- this.sliderSize =
- this.$refs.slider[`client${this.vertical ? 'Height' : 'Width'}`]
- }
- },
- emitChange() {
- this.$nextTick(() => {
- this.$emit(
- 'change',
- this.range ? [this.minValue, this.maxValue] : this.value
- )
- })
- },
- getStopStyle(position) {
- return this.vertical
- ? { bottom: position + '%' }
- : { left: position + '%' }
- },
- },
- }
- </script>
|