dragModal.js 3.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. import Vue from 'vue'
  2. // 弹窗拖拽属性
  3. /**
  4. * @directive 自定义属性
  5. * @todo 弹窗拖拽属性
  6. * @desc 使用在弹窗内部任意加载的html添加v-drag
  7. * @param .ant-modal-header 弹窗头部用来拖动的属性
  8. * @param .ant-modal 拖动的属性
  9. */
  10. Vue.directive('drag', (el, binding, vnode, oldVnode) => {
  11. // inserted (el, binding, vnode, oldVnode) {
  12. Vue.nextTick(() => {
  13. const isThemeModal = el.classList.contains('grid-theme')
  14. const dialogHeaderEl = isThemeModal ? el.querySelector('.ant-tabs-bar') : document.querySelector('.ant-modal-header')
  15. const dragDom = isThemeModal ? el.querySelector('.ant-modal') : document.querySelector('.ant-modal')
  16. // dialogHeaderEl.style.cursor = 'move';
  17. dialogHeaderEl.style.cssText += ';cursor:move;'
  18. // dragDom.style.cssText += ';top:0px;'
  19. // 获取原有属性 ie dom元素.currentStyle 火狐谷歌 window.getComputedStyle(dom元素, null);
  20. const sty = (function () {
  21. if (window.document.currentStyle) {
  22. return (dom, attr) => dom.currentStyle[attr]
  23. } else {
  24. return (dom, attr) => getComputedStyle(dom, false)[attr]
  25. }
  26. })()
  27. dialogHeaderEl.onmousedown = (e) => {
  28. // 鼠标按下,计算当前元素距离可视区的距离
  29. const disX = e.clientX - dialogHeaderEl.offsetLeft
  30. const disY = e.clientY - dialogHeaderEl.offsetTop
  31. const screenWidth = document.body.clientWidth // body当前宽度
  32. const screenHeight = document.documentElement.clientHeight // 可见区域高度(应为body高度,可某些环境下无法获取)
  33. const dragDomWidth = dragDom.offsetWidth // 对话框宽度
  34. const dragDomheight = dragDom.offsetHeight // 对话框高度
  35. const minDragDomLeft = dragDom.offsetLeft
  36. const maxDragDomLeft = screenWidth - dragDom.offsetLeft - dragDomWidth - (isThemeModal ? 10 : 0)
  37. const minDragDomTop = dragDom.offsetTop
  38. const maxDragDomTop = screenHeight - dragDom.offsetTop - dragDomheight - (isThemeModal ? 10 : 0)
  39. // 获取到的值带px 正则匹配替换
  40. let styL = sty(dragDom, 'left')
  41. let styT = sty(dragDom, 'top')
  42. // 注意在ie中 第一次获取到的值为组件自带50% 移动之后赋值为px
  43. if (styL.includes('%')) {
  44. // eslint-disable-next-line no-useless-escape
  45. styL = +document.body.clientWidth * (+styL.replace(/\%/g, '') / 100)
  46. // eslint-disable-next-line no-useless-escape
  47. styT = +document.body.clientHeight * (+styT.replace(/\%/g, '') / 100)
  48. } else {
  49. styL = +styL.replace(/\px/g, '')
  50. styT = +styT.replace(/\px/g, '')
  51. };
  52. document.onmousemove = function (e) {
  53. // 通过事件委托,计算移动的距离
  54. let left = e.clientX - disX
  55. let top = e.clientY - disY
  56. // 边界处理
  57. if (-(left) > minDragDomLeft) {
  58. left = -(minDragDomLeft)
  59. } else if (left > maxDragDomLeft) {
  60. left = maxDragDomLeft
  61. }
  62. if (-(top) > minDragDomTop) {
  63. top = -(minDragDomTop)
  64. } else if (top > maxDragDomTop) {
  65. top = maxDragDomTop
  66. }
  67. // 移动当前元素
  68. dragDom.style.cssText += `;left:${left + styL}px;top:${top + styT}px;`
  69. }
  70. document.onmouseup = function (e) {
  71. document.onmousemove = null
  72. document.onmouseup = null
  73. }
  74. }
  75. })
  76. })