index.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. <template>
  2. <div
  3. class="el-slider"
  4. :class="{ 'is-vertical': vertical, 'el-slider--with-input': showInput }"
  5. role="slider"
  6. :aria-valuemin="min"
  7. :aria-valuemax="max"
  8. :aria-orientation="vertical ? 'vertical' : 'horizontal'"
  9. :aria-disabled="sliderDisabled">
  10. <el-input-number
  11. v-model="firstValue"
  12. v-if="showInput && !range"
  13. class="el-slider__input"
  14. ref="input"
  15. @change="emitChange"
  16. :step="step"
  17. :disabled="sliderDisabled"
  18. :controls="showInputControls"
  19. :min="min"
  20. :max="max"
  21. :debounce="debounce"
  22. :size="inputSize">
  23. </el-input-number>
  24. <div
  25. class="el-slider__runway"
  26. :class="{ 'show-input': showInput, disabled: sliderDisabled }"
  27. :style="runwayStyle"
  28. @click="onSliderClick"
  29. ref="slider">
  30. <div
  31. class="el-slider__bar"
  32. :style="barStyle"></div>
  33. <slider-button
  34. :vertical="vertical"
  35. :marks="marks"
  36. v-model="firstValue"
  37. :tooltip-class="tooltipClass"
  38. ref="button1">
  39. </slider-button>
  40. <slider-button
  41. :vertical="vertical"
  42. :marks="marks"
  43. v-model="secondValue"
  44. :tooltip-class="tooltipClass"
  45. ref="button2"
  46. v-if="range">
  47. </slider-button>
  48. <div
  49. class="el-slider__stop"
  50. v-for="(item, key) in stops"
  51. :key="key"
  52. :style="getStopStyle(item)"
  53. v-show="showStops"></div>
  54. <template v-if="markList.length > 0">
  55. <div>
  56. <div
  57. v-for="(item, key) in markList"
  58. :style="getStopStyle(item.position)"
  59. @click.stop="onMarkClick(key)"
  60. class="el-slider__stop el-slider__marks-stop"
  61. :key="key"></div>
  62. </div>
  63. <div class="el-slider__marks">
  64. <slider-marker
  65. :mark="item.mark"
  66. v-for="(item, key) in markList"
  67. @click="onMarkClick(key)"
  68. :key="key"
  69. :style="getStopStyle(item.position)">
  70. </slider-marker>
  71. </div>
  72. </template>
  73. </div>
  74. </div>
  75. </template>
  76. <script type="text/babel">
  77. import ElInputNumber from 'element-ui/packages/input-number'
  78. import Emitter from 'element-ui/src/mixins/emitter'
  79. import SliderButton from './button.vue'
  80. import SliderMarker from './marker.vue'
  81. export default {
  82. name: 'ElSlider',
  83. components: {
  84. ElInputNumber,
  85. SliderButton,
  86. SliderMarker,
  87. },
  88. mixins: [Emitter],
  89. inject: {
  90. elForm: {
  91. default: '',
  92. },
  93. },
  94. props: {
  95. min: {
  96. type: Number,
  97. default: 0,
  98. },
  99. max: {
  100. type: Number,
  101. default: 100,
  102. },
  103. step: {
  104. type: Number,
  105. default: 1,
  106. },
  107. value: {
  108. // string 是数据框被清空时的值. 理论上不会出现, 这里只是用来规避控制台报错
  109. type: [Number, Array, String],
  110. default: 0,
  111. },
  112. showInput: {
  113. type: Boolean,
  114. default: false,
  115. },
  116. showInputControls: {
  117. type: Boolean,
  118. default: true,
  119. },
  120. inputSize: {
  121. type: String,
  122. default: 'small',
  123. },
  124. showStops: {
  125. type: Boolean,
  126. default: false,
  127. },
  128. showTooltip: {
  129. type: Boolean,
  130. default: true,
  131. },
  132. formatTooltip: Function,
  133. disabled: {
  134. type: Boolean,
  135. default: false,
  136. },
  137. range: {
  138. type: Boolean,
  139. default: false,
  140. },
  141. vertical: {
  142. type: Boolean,
  143. default: false,
  144. },
  145. height: {
  146. type: String,
  147. },
  148. debounce: {
  149. type: Number,
  150. default: 300,
  151. },
  152. label: {
  153. type: String,
  154. },
  155. tooltipClass: String,
  156. marks: Object,
  157. },
  158. data() {
  159. return {
  160. firstValue: null,
  161. secondValue: null,
  162. oldValue: null,
  163. dragging: false,
  164. sliderSize: 1,
  165. }
  166. },
  167. computed: {
  168. stops() {
  169. if (!this.showStops || this.min > this.max) return []
  170. if (this.step === 0) {
  171. process.env.NODE_ENV !== 'production' &&
  172. console.warn('[Element Warn][Slider]step should not be 0.')
  173. return []
  174. }
  175. const stopCount = (this.max - this.min) / this.step
  176. const stepWidth = (100 * this.step) / (this.max - this.min)
  177. const result = []
  178. for (let i = 1; i < stopCount; i++) {
  179. result.push(i * stepWidth)
  180. }
  181. if (this.range) {
  182. return result.filter(step => {
  183. return (
  184. step < (100 * (this.minValue - this.min)) / (this.max - this.min) ||
  185. step > (100 * (this.maxValue - this.min)) / (this.max - this.min)
  186. )
  187. })
  188. } else {
  189. return result.filter(
  190. step =>
  191. step > (100 * (this.firstValue - this.min)) / (this.max - this.min)
  192. )
  193. }
  194. },
  195. markList() {
  196. if (!this.marks) {
  197. return []
  198. }
  199. const marksKeys = Object.keys(this.marks)
  200. return marksKeys
  201. .map(parseFloat)
  202. .sort((a, b) => a - b)
  203. .filter(point => point <= this.max && point >= this.min)
  204. .map((point, index) => ({
  205. point,
  206. // 改造使得mark在滑动区域内均分
  207. position:
  208. (100 * index * (this.max - this.min)) /
  209. (marksKeys.length - 1) /
  210. (this.max - this.min),
  211. // position: ((point - this.min) * 100) / (this.max - this.min),
  212. mark: this.marks[point],
  213. }))
  214. },
  215. minValue() {
  216. return Math.min(this.firstValue, this.secondValue)
  217. },
  218. maxValue() {
  219. return Math.max(this.firstValue, this.secondValue)
  220. },
  221. barSize() {
  222. if (this.marks) {
  223. const marks = Object.keys(this.marks).map(Number)
  224. // 找出目标值位于marks的那个档位, 即 即将到达哪一个'档'
  225. const index = marks.findIndex(i => this.firstValue < i)
  226. // console.log(index, 'index')
  227. if (this.range) {
  228. // todo 范围值的计算. 这个还没改. 不知道怎么算.
  229. return `${
  230. (100 * (this.maxValue - this.minValue)) / (this.max - this.min)
  231. }%`
  232. }
  233. if (index === -1) return '100%'
  234. // 经过后一个档位之后占总进度条的比例 + (当前值超出后一个档位的差值 / 前一个档位与后一个档位的差值) / (进度被marks划分成的档位数量)
  235. // 上述公式要除档位数量是因为, 当前值与档位差值占单个档位区间的百分比 !== 单个区间值占总进度条的百分比
  236. const value =
  237. (index - 1) / (marks.length - 1) +
  238. (this.firstValue - marks[index - 1]) /
  239. (marks[index] - marks[index - 1]) /
  240. (marks.length - 1)
  241. // console.log('value', value)
  242. return value > 1 ? '100%' : `${value * 100}%`
  243. } else {
  244. return this.range
  245. ? `${
  246. (100 * (this.maxValue - this.minValue)) / (this.max - this.min)
  247. }%`
  248. : `${(100 * (this.firstValue - this.min)) / (this.max - this.min)}%`
  249. }
  250. },
  251. barStart() {
  252. return this.range
  253. ? `${(100 * (this.minValue - this.min)) / (this.max - this.min)}%`
  254. : '0%'
  255. },
  256. precision() {
  257. const precisions = [this.min, this.max, this.step].map(item => {
  258. const decimal = ('' + item).split('.')[1]
  259. return decimal ? decimal.length : 0
  260. })
  261. return Math.max.apply(null, precisions)
  262. },
  263. runwayStyle() {
  264. return this.vertical ? { height: this.height } : {}
  265. },
  266. barStyle() {
  267. return this.vertical
  268. ? {
  269. height: this.barSize,
  270. bottom: this.barStart,
  271. }
  272. : {
  273. width: this.barSize,
  274. left: this.barStart,
  275. }
  276. },
  277. sliderDisabled() {
  278. return this.disabled || (this.elForm || {}).disabled
  279. },
  280. },
  281. watch: {
  282. value(val, oldVal) {
  283. if (
  284. this.dragging ||
  285. (Array.isArray(val) &&
  286. Array.isArray(oldVal) &&
  287. val.every((item, index) => item === oldVal[index]))
  288. ) {
  289. return
  290. }
  291. this.setValues()
  292. },
  293. dragging(val) {
  294. if (!val) {
  295. this.setValues()
  296. }
  297. },
  298. firstValue(val) {
  299. if (this.range) {
  300. this.$emit('input', [this.minValue, this.maxValue])
  301. } else {
  302. this.$emit('input', val)
  303. }
  304. },
  305. secondValue() {
  306. if (this.range) {
  307. this.$emit('input', [this.minValue, this.maxValue])
  308. }
  309. },
  310. min() {
  311. this.setValues()
  312. },
  313. max() {
  314. this.setValues()
  315. },
  316. },
  317. mounted() {
  318. let valuetext
  319. if (this.range) {
  320. if (Array.isArray(this.value)) {
  321. this.firstValue = Math.max(this.min, this.value[0])
  322. this.secondValue = Math.min(this.max, this.value[1])
  323. } else {
  324. this.firstValue = this.min
  325. this.secondValue = this.max
  326. }
  327. this.oldValue = [this.firstValue, this.secondValue]
  328. valuetext = `${this.firstValue}-${this.secondValue}`
  329. } else {
  330. if (typeof this.value !== 'number' || isNaN(this.value)) {
  331. this.firstValue = this.min
  332. } else {
  333. this.firstValue = Math.min(this.max, Math.max(this.min, this.value))
  334. }
  335. this.oldValue = this.firstValue
  336. valuetext = this.firstValue
  337. }
  338. this.$el.setAttribute('aria-valuetext', valuetext)
  339. // label screen reader
  340. this.$el.setAttribute(
  341. 'aria-label',
  342. this.label ? this.label : `slider between ${this.min} and ${this.max}`
  343. )
  344. this.resetSize()
  345. window.addEventListener('resize', this.resetSize)
  346. },
  347. beforeDestroy() {
  348. window.removeEventListener('resize', this.resetSize)
  349. },
  350. methods: {
  351. valueChanged() {
  352. if (this.range) {
  353. return ![this.minValue, this.maxValue].every(
  354. (item, index) => item === this.oldValue[index]
  355. )
  356. } else {
  357. return this.value !== this.oldValue
  358. }
  359. },
  360. setValues() {
  361. if (this.min > this.max) {
  362. console.error(
  363. '[Element Error][Slider]min should not be greater than max.'
  364. )
  365. return
  366. }
  367. const val = this.value
  368. if (this.range && Array.isArray(val)) {
  369. if (val[1] < this.min) {
  370. this.$emit('input', [this.min, this.min])
  371. } else if (val[0] > this.max) {
  372. this.$emit('input', [this.max, this.max])
  373. } else if (val[0] < this.min) {
  374. this.$emit('input', [this.min, val[1]])
  375. } else if (val[1] > this.max) {
  376. this.$emit('input', [val[0], this.max])
  377. } else {
  378. this.firstValue = val[0]
  379. this.secondValue = val[1]
  380. if (this.valueChanged()) {
  381. this.dispatch('ElFormItem', 'el.form.change', [
  382. this.minValue,
  383. this.maxValue,
  384. ])
  385. this.oldValue = val.slice()
  386. }
  387. }
  388. } else if (!this.range && typeof val === 'number' && !isNaN(val)) {
  389. if (val < this.min) {
  390. this.$emit('input', this.min)
  391. } else if (val > this.max) {
  392. this.$emit('input', this.max)
  393. } else {
  394. this.firstValue = val
  395. if (this.valueChanged()) {
  396. this.dispatch('ElFormItem', 'el.form.change', val)
  397. this.oldValue = val
  398. }
  399. }
  400. }
  401. },
  402. setPosition(percent) {
  403. const targetValue = this.min + (percent * (this.max - this.min)) / 100
  404. if (!this.range) {
  405. this.$refs.button1.setPosition(percent)
  406. return
  407. }
  408. let button
  409. if (
  410. Math.abs(this.minValue - targetValue) <
  411. Math.abs(this.maxValue - targetValue)
  412. ) {
  413. button = this.firstValue < this.secondValue ? 'button1' : 'button2'
  414. } else {
  415. button = this.firstValue > this.secondValue ? 'button1' : 'button2'
  416. }
  417. this.$refs[button].setPosition(percent)
  418. },
  419. onSliderClick(event) {
  420. if (this.sliderDisabled || this.dragging) return
  421. this.resetSize()
  422. if (this.vertical) {
  423. const sliderOffsetBottom =
  424. this.$refs.slider.getBoundingClientRect().bottom
  425. this.setPosition(
  426. ((sliderOffsetBottom - event.clientY) / this.sliderSize) * 100
  427. )
  428. } else {
  429. const sliderOffsetLeft = this.$refs.slider.getBoundingClientRect().left
  430. this.setPosition(
  431. ((event.clientX - sliderOffsetLeft) / this.sliderSize) * 100
  432. )
  433. }
  434. this.emitChange()
  435. },
  436. onMarkClick(index) {
  437. if (typeof index !== 'number') return
  438. // 这个判断和resetSize从onSliderClick抄过来的.
  439. // 因为直接点击mark的 标志文案 得到的不是准确的数(宽度导致的位置不同), 所以魔改了一下, 点击mark按固定比例算.
  440. if (this.sliderDisabled || this.dragging) return
  441. this.resetSize()
  442. const marks = Object.keys(this.marks).map(Number)
  443. // 计算当前点击位于marks的第几项, 算百分比
  444. this.setPosition((100 * index) / (marks.length - 1))
  445. this.emitChange()
  446. },
  447. resetSize() {
  448. if (this.$refs.slider) {
  449. this.sliderSize =
  450. this.$refs.slider[`client${this.vertical ? 'Height' : 'Width'}`]
  451. }
  452. },
  453. emitChange() {
  454. this.$nextTick(() => {
  455. this.$emit(
  456. 'change',
  457. this.range ? [this.minValue, this.maxValue] : this.value
  458. )
  459. })
  460. },
  461. getStopStyle(position) {
  462. return this.vertical
  463. ? { bottom: position + '%' }
  464. : { left: position + '%' }
  465. },
  466. },
  467. }
  468. </script>