custom-el-slider-button.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. exports.ids = [8];
  2. exports.modules = {
  3. /***/ 158:
  4. /***/ (function(module, __webpack_exports__, __webpack_require__) {
  5. "use strict";
  6. // ESM COMPAT FLAG
  7. __webpack_require__.r(__webpack_exports__);
  8. // CONCATENATED MODULE: ./node_modules/babel-loader/lib??ref--2-0!./node_modules/vue-loader/lib/loaders/templateLoader.js??ref--6!./node_modules/@nuxt/components/dist/loader.js??ref--0-0!./node_modules/vue-loader/lib??vue-loader-options!./components/custom-el-slider/button.vue?vue&type=template&id=09a28971
  9. var render = function render() {
  10. var _vm = this,
  11. _c = _vm._self._c;
  12. return _c('div', {
  13. ref: "button",
  14. staticClass: "el-slider__button-wrapper",
  15. class: {
  16. hover: _vm.hovering,
  17. dragging: _vm.dragging
  18. },
  19. style: _vm.wrapperStyle,
  20. attrs: {
  21. "tabindex": "0"
  22. },
  23. on: {
  24. "mouseenter": _vm.handleMouseEnter,
  25. "mouseleave": _vm.handleMouseLeave,
  26. "mousedown": _vm.onButtonDown,
  27. "touchstart": _vm.onButtonDown,
  28. "focus": _vm.handleMouseEnter,
  29. "blur": _vm.handleMouseLeave,
  30. "keydown": [function ($event) {
  31. if (!$event.type.indexOf('key') && _vm._k($event.keyCode, "left", 37, $event.key, ["Left", "ArrowLeft"])) return null;
  32. if ('button' in $event && $event.button !== 0) return null;
  33. return _vm.onLeftKeyDown.apply(null, arguments);
  34. }, function ($event) {
  35. if (!$event.type.indexOf('key') && _vm._k($event.keyCode, "right", 39, $event.key, ["Right", "ArrowRight"])) return null;
  36. if ('button' in $event && $event.button !== 2) return null;
  37. return _vm.onRightKeyDown.apply(null, arguments);
  38. }, function ($event) {
  39. if (!$event.type.indexOf('key') && _vm._k($event.keyCode, "down", 40, $event.key, ["Down", "ArrowDown"])) return null;
  40. $event.preventDefault();
  41. return _vm.onLeftKeyDown.apply(null, arguments);
  42. }, function ($event) {
  43. if (!$event.type.indexOf('key') && _vm._k($event.keyCode, "up", 38, $event.key, ["Up", "ArrowUp"])) return null;
  44. $event.preventDefault();
  45. return _vm.onRightKeyDown.apply(null, arguments);
  46. }]
  47. }
  48. }, [_c('el-tooltip', {
  49. ref: "tooltip",
  50. attrs: {
  51. "placement": "top",
  52. "popper-class": _vm.tooltipClass,
  53. "disabled": !_vm.showTooltip
  54. }
  55. }, [_c('span', {
  56. attrs: {
  57. "slot": "content"
  58. },
  59. slot: "content"
  60. }, [_vm._v(_vm._s(_vm.formatValue))]), _vm._v(" "), _c('div', {
  61. staticClass: "el-slider__button",
  62. class: {
  63. hover: _vm.hovering,
  64. dragging: _vm.dragging
  65. }
  66. })])], 1);
  67. };
  68. var staticRenderFns = [];
  69. // CONCATENATED MODULE: ./components/custom-el-slider/button.vue?vue&type=template&id=09a28971
  70. // CONCATENATED MODULE: ./node_modules/babel-loader/lib??ref--2-0!./node_modules/@nuxt/components/dist/loader.js??ref--0-0!./node_modules/vue-loader/lib??vue-loader-options!./components/custom-el-slider/button.vue?vue&type=script&lang=js
  71. // import ElTooltip from 'element-ui/packages/tooltip'
  72. /* harmony default export */ var buttonvue_type_script_lang_js = ({
  73. name: 'ElSliderButton',
  74. components: {
  75. // ElTooltip,
  76. },
  77. props: {
  78. value: {
  79. type: Number,
  80. default: 0
  81. },
  82. vertical: {
  83. type: Boolean,
  84. default: false
  85. },
  86. tooltipClass: String,
  87. marks: Object
  88. },
  89. data() {
  90. return {
  91. hovering: false,
  92. dragging: false,
  93. isClick: false,
  94. startX: 0,
  95. currentX: 0,
  96. startY: 0,
  97. currentY: 0,
  98. startPosition: 0,
  99. newPosition: null,
  100. oldValue: this.value
  101. };
  102. },
  103. computed: {
  104. disabled() {
  105. return this.$parent.sliderDisabled;
  106. },
  107. max() {
  108. return this.$parent.max;
  109. },
  110. min() {
  111. return this.$parent.min;
  112. },
  113. step() {
  114. return this.$parent.step;
  115. },
  116. showTooltip() {
  117. return this.$parent.showTooltip;
  118. },
  119. precision() {
  120. return this.$parent.precision;
  121. },
  122. currentPosition() {
  123. if (this.marks) {
  124. // 同父组件index中barSize的计算逻辑. 直接从那边抄过来的. 不要在这里改, 在index改完再抄过来.
  125. // 注意父组件的 this.firstValue, 这里是 this.value
  126. const marks = Object.keys(this.marks).map(Number);
  127. // 找出目标值位于marks的那个档位, 即 即将到达哪一个'档'
  128. const index = marks.findIndex(i => this.value < i);
  129. if (index === -1) return '100%';
  130. // 经过后一个档位之后占总进度条的比例 + (当前值超出后一个档位的差值 / 前一个档位与后一个档位的差值) / (进度被marks划分成的档位数量)
  131. // 上述公式要除档位数量是因为, 当前值与档位差值占单个档位区间的百分比 !== 单个区间值占总进度条的百分比
  132. const value = (index - 1) / (marks.length - 1) + (this.value - marks[index - 1]) / (marks[index] - marks[index - 1]) / (marks.length - 1);
  133. return value > 1 ? '100%' : `${value * 100}%`;
  134. } else {
  135. // 这里是原本的逻辑. 没有传marks的.
  136. return `${(this.value - this.min) / (this.max - this.min) * 100}%`;
  137. }
  138. },
  139. enableFormat() {
  140. return this.$parent.formatTooltip instanceof Function;
  141. },
  142. formatValue() {
  143. return this.enableFormat && this.$parent.formatTooltip(this.value) || this.value;
  144. },
  145. wrapperStyle() {
  146. return this.vertical ? {
  147. bottom: this.currentPosition
  148. } : {
  149. left: this.currentPosition
  150. };
  151. }
  152. },
  153. watch: {
  154. dragging(val) {
  155. this.$parent.dragging = val;
  156. }
  157. },
  158. methods: {
  159. displayTooltip() {
  160. this.$refs.tooltip && (this.$refs.tooltip.showPopper = true);
  161. },
  162. hideTooltip() {
  163. this.$refs.tooltip && (this.$refs.tooltip.showPopper = false);
  164. },
  165. handleMouseEnter() {
  166. this.hovering = true;
  167. this.displayTooltip();
  168. },
  169. handleMouseLeave() {
  170. this.hovering = false;
  171. this.hideTooltip();
  172. },
  173. onButtonDown(event) {
  174. if (this.disabled) return;
  175. event.preventDefault();
  176. this.onDragStart(event);
  177. window.addEventListener('mousemove', this.onDragging);
  178. window.addEventListener('touchmove', this.onDragging);
  179. window.addEventListener('mouseup', this.onDragEnd);
  180. window.addEventListener('touchend', this.onDragEnd);
  181. window.addEventListener('contextmenu', this.onDragEnd);
  182. },
  183. onLeftKeyDown() {
  184. if (this.disabled) return;
  185. this.newPosition = parseFloat(this.currentPosition) - this.step / (this.max - this.min) * 100;
  186. this.setPosition(this.newPosition);
  187. this.$parent.emitChange();
  188. },
  189. onRightKeyDown() {
  190. if (this.disabled) return;
  191. this.newPosition = parseFloat(this.currentPosition) + this.step / (this.max - this.min) * 100;
  192. this.setPosition(this.newPosition);
  193. this.$parent.emitChange();
  194. },
  195. onDragStart(event) {
  196. this.dragging = true;
  197. this.isClick = true;
  198. if (event.type === 'touchstart') {
  199. event.clientY = event.touches[0].clientY;
  200. event.clientX = event.touches[0].clientX;
  201. }
  202. if (this.vertical) {
  203. this.startY = event.clientY;
  204. } else {
  205. this.startX = event.clientX;
  206. }
  207. this.startPosition = parseFloat(this.currentPosition);
  208. this.newPosition = this.startPosition;
  209. },
  210. onDragging(event) {
  211. if (this.dragging) {
  212. this.isClick = false;
  213. this.displayTooltip();
  214. this.$parent.resetSize();
  215. let diff = 0;
  216. if (event.type === 'touchmove') {
  217. event.clientY = event.touches[0].clientY;
  218. event.clientX = event.touches[0].clientX;
  219. }
  220. if (this.vertical) {
  221. this.currentY = event.clientY;
  222. diff = (this.startY - this.currentY) / this.$parent.sliderSize * 100;
  223. } else {
  224. this.currentX = event.clientX;
  225. diff = (this.currentX - this.startX) / this.$parent.sliderSize * 100;
  226. }
  227. this.newPosition = this.startPosition + diff;
  228. this.setPosition(this.newPosition);
  229. }
  230. },
  231. onDragEnd() {
  232. if (this.dragging) {
  233. /*
  234. * 防止在 mouseup 后立即触发 click,导致滑块有几率产生一小段位移
  235. * 不使用 preventDefault 是因为 mouseup 和 click 没有注册在同一个 DOM 上
  236. */
  237. setTimeout(() => {
  238. this.dragging = false;
  239. this.hideTooltip();
  240. if (!this.isClick) {
  241. this.setPosition(this.newPosition);
  242. this.$parent.emitChange();
  243. }
  244. }, 0);
  245. window.removeEventListener('mousemove', this.onDragging);
  246. window.removeEventListener('touchmove', this.onDragging);
  247. window.removeEventListener('mouseup', this.onDragEnd);
  248. window.removeEventListener('touchend', this.onDragEnd);
  249. window.removeEventListener('contextmenu', this.onDragEnd);
  250. }
  251. },
  252. setPosition(newPosition) {
  253. if (newPosition === null || isNaN(newPosition)) return;
  254. if (newPosition < 0) {
  255. newPosition = 0;
  256. } else if (newPosition > 100) {
  257. newPosition = 100;
  258. }
  259. const lengthPerStep = 100 / ((this.max - this.min) / this.step);
  260. const steps = Math.round(newPosition / lengthPerStep);
  261. let value = steps * lengthPerStep * (this.max - this.min) * 0.01 + this.min;
  262. if (this.marks) {
  263. // 处理传递marks时的逻辑. 没有传marks的保持原样.
  264. // 有marks时无视原计算逻辑. 重新赋值
  265. const marks = Object.keys(this.marks).map((current, index, arr) => {
  266. return {
  267. value: Number(current),
  268. perc: 100 * index / (arr.length - 1)
  269. };
  270. });
  271. // 找出目标值位于marks的那个档位, 即 即将到达哪一个'档'
  272. if (newPosition === 100) {
  273. value = marks[marks.length - 1].value;
  274. } else {
  275. const index = marks.findIndex(item => item.perc > newPosition);
  276. // 后一个区间的代表值 + ((区间数量 * 超出后一个区间的百分比值 / 100) * 所处区间的值差)
  277. if (index === -1) {
  278. // 在最后一个区间
  279. value = marks[marks.length - 2].value + (marks.length - 1) * ((newPosition - marks[marks.length - 2].perc) / 100 * (marks[marks.length - 1].value - marks[marks.length - 2].value));
  280. } else {
  281. value = marks[index - 1].value + (marks.length - 1) * ((newPosition - marks[index - 1].perc) / 100 * (marks[index].value - marks[index - 1].value));
  282. }
  283. }
  284. }
  285. value = parseFloat(value.toFixed(this.precision));
  286. this.$emit('input', value);
  287. this.$nextTick(() => {
  288. this.displayTooltip();
  289. this.$refs.tooltip && this.$refs.tooltip.updatePopper();
  290. });
  291. if (!this.dragging && this.value !== this.oldValue) {
  292. this.oldValue = this.value;
  293. }
  294. }
  295. }
  296. });
  297. // CONCATENATED MODULE: ./components/custom-el-slider/button.vue?vue&type=script&lang=js
  298. /* harmony default export */ var custom_el_slider_buttonvue_type_script_lang_js = (buttonvue_type_script_lang_js);
  299. // EXTERNAL MODULE: ./node_modules/vue-loader/lib/runtime/componentNormalizer.js
  300. var componentNormalizer = __webpack_require__(2);
  301. // CONCATENATED MODULE: ./components/custom-el-slider/button.vue
  302. /* normalize component */
  303. var component = Object(componentNormalizer["a" /* default */])(
  304. custom_el_slider_buttonvue_type_script_lang_js,
  305. render,
  306. staticRenderFns,
  307. false,
  308. null,
  309. null,
  310. "5e8cb690"
  311. )
  312. /* harmony default export */ var custom_el_slider_button = __webpack_exports__["default"] = (component.exports);
  313. /***/ })
  314. };;
  315. //# sourceMappingURL=custom-el-slider-button.js.map