Practice of Vue global custom instruction Modal drag

Practice of Vue global custom instruction Modal drag

background

The project I have been working on recently is a project built with Vue2, and the UI framework uses antdV. A lot of Modal dialog boxes are used in the project. Then suddenly one day the product said, "Why can't this dialog box be moved? It blocks my view of the information." There is no way, if there is a demand, we have to do it. This is the new generation of workers.

First, I went to the antdV official website to look at the configuration properties and methods of Modal to see if there are any properties or methods that can solve this problem directly, but I didn't find any. The second step was to start asking Baidu. There were some related blogs, but generally speaking, they were too diverse and difficult to understand. Then I found that there was an interesting way to use global custom specifications, and then I started to solve the problem myself.

Implementation ideas

First of all, we mainly want to complete the effect of moving the dialog box after pressing the left button when the mouse is placed on the head part of the Modal box.

Here we need to get the header element and listen to its onmousedown method, then during the onmousedown method, we calculate the height and width of the dialog box to set boundary limits to prevent it from moving off the screen, then listen to the document.onmousemove method to calculate the direction and distance of the mouse movement, and then adjust the position of the dialog box based on this.

Code Implementation

import Vue from 'vue'

// Pop-up window drag properties/**
 * @directive custom attributes * @todo pop-up drag attributes * @desc add v-drag using any html loaded inside the pop-up
 * @param .ant-modal-header The properties of the pop-up window header used for dragging* @param .ant-modal The properties to drag*/
Vue.directive('drag', (el, binding, vnode, oldVnode) => {
  // inserted (el, binding, vnode, oldVnode) {
  Vue.nextTick(() => {
    const isThemeModal = el.classList.contains('grid-theme')
    const dialogHeaderEl = isThemeModal ? el.querySelector('.ant-tabs-bar') : document.querySelector('.ant-modal-header')
    const dragDom = isThemeModal ? el.querySelector('.ant-modal') : document.querySelector('.ant-modal')
    //dialogHeaderEl.style.cursor = 'move';
    dialogHeaderEl.style.cssText += ';cursor:move;'
    // dragDom.style.cssText += ';top:0px;'

    // Get the original attributes ie dom element.currentStyle firefox google window.getComputedStyle(dom element, null);
    const sty = (function () {
      if (window.document.currentStyle) {
        return (dom, attr) => dom.currentStyle[attr]
      } else {
        return (dom, attr) => getComputedStyle(dom, false)[attr]
      }
    })()

    dialogHeaderEl.onmousedown = (e) => {
      // Mouse pressed, calculate the distance between the current element and the visible area const disX = e.clientX - dialogHeaderEl.offsetLeft
      const disY = e.clientY - dialogHeaderEl.offsetTop

      const screenWidth = document.body.clientWidth // Current width of body const screenHeight = document.documentElement.clientHeight // Visible area height (should be body height, but cannot be obtained in some environments)

      const dragDomWidth = dragDom.offsetWidth // Dialog box width const dragDomheight = dragDom.offsetHeight // Dialog box height const minDragDomLeft = dragDom.offsetLeft
      const maxDragDomLeft = screenWidth - dragDom.offsetLeft - dragDomWidth - (isThemeModal ? 10 : 0)

      const minDragDomTop = dragDom.offsetTop
      const maxDragDomTop = screenHeight - dragDom.offsetTop - dragDomheight - (isThemeModal ? 10 : 0)

      // Get the value with px regular expression to replace let styL = sty(dragDom, 'left')
      let styT = sty(dragDom, 'top')

      // Note that in IE, the first value obtained is the component's own 50% and the value is assigned to px after moving
      if (styL.includes('%')) {
        // eslint-disable-next-line no-useless-escape
        styL = +document.body.clientWidth * (+styL.replace(/\%/g, '') / 100)
        // eslint-disable-next-line no-useless-escape
        styT = +document.body.clientHeight * (+styT.replace(/\%/g, '') / 100)
      } else {
        styL = +styL.replace(/\px/g, '')
        styT = +styT.replace(/\px/g, '')
      };

      document.onmousemove = function (e) {
        // Calculate the moving distance through event delegation let left = e.clientX - disX
        let top = e.clientY - disY

        // Boundary processing if (-(left) > minDragDomLeft) {
          left = -(minDragDomLeft)
        } else if (left > maxDragDomLeft) {
          left = maxDragDomLeft
        }

        if (-(top) > minDragDomTop) {
          top = -(minDragDomTop)
        } else if (top > maxDragDomTop) {
          top = maxDragDomTop
        }
        // Move the current element dragDom.style.cssText += `;left:${left + styL}px;top:${top + styT}px;`
      }

      document.onmouseup = function (e) {
        document.onmousemove = null
        document.onmouseup = null
      }
    }
  })
})

Component call
A-modal component can be set to v-drag

...
<a-modal v-model="visible" title="Prompt" :maskClosable="false" @cancel="handleCancle" @ok="handleOk" v-drag>
  <p>Are you sure you want to delete the field?</p>
</a-modal>
...

Achieve results

This is the end of this article about the practice of Vue global custom directive Modal drag. For more relevant Vue Modal drag content, please search 123WORDPRESS.COM's previous articles or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • Vue custom directive details
  • A detailed guide to custom directives in Vue
  • Understand the usage of Vue2.x and Vue3.x custom instructions and the principle of hook functions
  • Basic usage of custom directives in Vue
  • Detailed explanation of Vue.js directive custom instructions
  • Vue custom instructions to achieve pop-up window drag four-side stretching and diagonal stretching effect
  • Detailed explanation of Vue custom instructions

<<:  Use of nginx custom variables and built-in predefined variables

>>:  Summary of common MySQL function examples [aggregate functions, strings, numbers, time and date processing, etc.]

Recommend

MySQL 8.0.12 winx64 detailed installation tutorial

This article shares the installation tutorial of ...

SQL IDENTITY_INSERT case study

Generally speaking, once a column in a data table...

Node script realizes automatic sign-in and lottery function

Table of contents 1. Introduction 2. Preparation ...

An example of how Vue implements four-level navigation and verification code

Effect: First create five vue interfaces 1.home.v...

How to position the header at the top using CSS sticky layout

Application scenarios: One of the new requirement...

Vite introduces the implementation of virtual files

Table of contents background Importing virtual fi...

JS practical object-oriented snake game example

Table of contents think 1. Greedy Snake Effect Pi...

js to write the carousel effect

This article shares the specific code of js to ac...

How to prevent hyperlink redirection using JavaScript (multiple ways of writing)

Through JavaScript, we can prevent hyperlinks fro...

Web page text design should be like smart girls wearing clothes

<br />"There are no ugly women in the w...

Detailed configuration of mysql8.x docker remote access

Table of contents Environmental conditions Errors...