preface

Some time ago, I did operation activities to build a platform, one of the main functions: the editing page is divided into left – component area and right – preview area, the content of the component area can be freely placed in the preview area.

As shown below:

There are some similar implementations in the community, but they are used in much the same way, all with drag-and-drop capabilities. We will often use drag in daily development, such as drag sorting, drag upload and so on. Of course, there are a lot of drag packages, such as React-DND, Vue’s own drag and drop capabilities.

However, our preview area uses the iframe approach, and community-friendly libraries generally do not support drag and drop across iframes. Here we chose to use native drag and dropAPI

There are two main functions that need to be implemented:

1. Check drag inside and outside the IFrame.

2. Data-driven display of iframe internal components.

We simply generate page functionality:

// Build the edit page
//drag.jsx
import React, { useState, useEffect } from 'react';
import Drag from './drag.js';

require('./styles.less');

//iframe hooks
const useIframeLoad = () = > {
  const [iframeState, setIframeState] = useState(false);
  const [windowState, setWindowState] = useState( document.readyState === "complete");

  const iframeLoad = () = > {
    const iframeEle = document.getElementById("my-iframe");
    iframeEle && setIframeState(iframeEle.contentDocument.readyState === "complete");
    if(! iframeState && iframeEle) { iframeEle.onload =() = > {
        setIframeState(true); }; }}; useEffect(() = > {
    if(! windowState) { setIframeState(false);
      window.addEventListener('load'.() = > {
        setWindowState(true); iframeLoad(); })}else{ iframeLoad(); }} []);return iframeState;
}

export default() = > {const init = () = > {
    Drag.init({
      dragEle: document.getElementById('drag-box'),
      dropEle: document.getElementById('my-iframe').contentDocument.getElementById('drop-box')
    })
  }

  useIframeLoad() && init();

  return <>
   <! -- Component area -->
    <div id="drag-box">
      <div className="drag-item">Drag the elements</div>
      <div className="drag-item">Drag the elements</div>
      <div className="drag-item">Drag the elements</div>
    </div>
    <! -- Preview area -->
    <div className="drop-content">
      <iframe id="my-iframe" src="#/iframe" style={{ width: "100% ",height: "480px", border: "none}} "/ >
    </div>
  </>
}
Copy the code

Preview area iframe page:

//iframe.jsx
import React from 'react';

require('./styles.less');

export default() = > {return <div id="drop-box">
    <div className="item">Element 1</div>
    <div className="item">Element 2</div>
    <div className="item">Elements 3</div>
  </div>
}
Copy the code

At this point, the simple setup edit layout is complete. Next, let’s look at the drag part:

Across the iframe drag and drop

We can start by looking at what native events are available

Native events

drag // This event is triggered when you drag an element or text selection.
dragstart // This event is emitted when the user starts dragging an element or selection text
dragend This event is emitted when the drag operation ends (by releasing the mouse button or pressing the exit key)

dragover // This event is emitted when the dragged element moves within the release zone
dragenter This event is emitted when the screen space is occupied by dragging elements into the free area
dragleave // This event is emitted when the dragged element leaves the release zone without dropping it
dragexit This event is emitted when an element is no longer the immediate selection target for a drag operation
drop // This event is emitted when the dragged element is dropped in the release area
Copy the code

Native Drag and drop drag

Based on the requirements, break down the key drag processes:

  • Initialization element sets the drag element and the target node
  • Register events Register drag events for drag elements and target node elements
  • The placeholder node is generated during the dragging process of listening to events, and the placeholder node is deleted after dragging

The incomplete code is as follows:

//drag.js
class Drag {
  params = {}

  init = (params) = >{... };// Initialize the drag element
  initDrag = dragEle= > {
    if(dragEle.childNodes.length) {
      const { length } = dragEle.childNodes;
      let i = 0
      while (i< length) {
        this.setDrag(dragEle.childNodes[i]);
        i += 1; }}else {
      this.setDrag(dragEle); }}// Initialize the release zone
  initDrop = dropEle= > {
    if (dropEle.childNodes.length) {
      const { length } = dropEle.childNodes;
      let i = 0;
      while (i < length) {
        this.setDrop(dropEle.childNodes[i]);
        i += 1; }}else {
      this.setDrop(dropEle); }}// Drag the element to register the event
  setDrag = el= > {
    el.setAttribute("draggable"."true");
    el.ondragstart = this.dragStartEvent;
    el.ondrag = this.dragEvent;
    el.ondragend = this.dragEndEvent;
  };

  // Release zone registration event
  setDrop = el= > {
    el.ondrop = this.dropEvent;
    el.ondragenter = this.dragEnterEvent;
    el.ondragover = this.dragOverEvent;
    el.ondragleave = this.dragLeaveEvent; }...// Create placeholder elements
  createElePlaceholder = (() = > {
    let ele = null;
    return () = > {
      if(! ele) { ele =document.createElement("div");
        ele.setAttribute("id"."drag-ele-placeholder");
        ele.innerHTML = `
      
; } returnele; }; }) ();// Remove the placeholder element removePlaceholderEle = () = > { const iframe = this.getIframe(); const removeEle = iframe.contentDocument.getElementById("drag-ele-placeholder"); const { dropEle } = this.params; if(this.isHasPlaceholderEle()) { dropEle.removeChild(removeEle) }; } /****** Event handling ******/ dragEndEvent = ev= > { this.removePlaceholderEle() console.log('End of drag'); console.log('Remove placeholder element'); }; // Insert placeholder elements dragEnterEvent = ev= > { ev.preventDefault(); const insertEle = this.createElePlaceholder(); ev.target.before(insertEle); console.log('Into the placement zone'); console.log('Insert placeholder element'); }; // Remove placeholder elements dragLeaveEvent = ev= > { ev.preventDefault(); this.removePlaceholderEle() console.log('Leave storage area'); console.log('Remove placeholder element'); }; dropEvent = ev= > { ev.preventDefault(); console.log('Release mouse over placement area'); }}export default new Drag(); Copy the code

After preliminary completion, the results are as follows:

There are some problems here:

  1. The page blinks during insertion
  2. The entry event is triggered only when the mouse position enters the release area
  3. The addition of the first element cannot be implemented

Problem analysis

  • When dragged into the preview area, the node dragenter event in the preview area is triggered. Each time a placeholder element is inserted on the current node, the node’s position changes, triggering the node dragleave event and removing the placeholder element. This process keeps repeating, causing constant flickering.
  • The above 2,3 problems are due to the drag/drop API limitations

Since the current way can not really perfect the function, decided to abandon the dragover, Dragenter, dragleave events

Reorganize the function points that need to be optimized:

  1. When the drag element touches the edge of the iframe, it enters the release zone
  2. Drag to insert elements above and elements below

Precise coordinate calculations are used to handle entry to release zones and insertions above and below elements

Make some changes to drag.js:

class Drag {
  params = {}

  / / declare
  mouseOffsetBottom = 0;
  mouseOffsetRight = 0;

  init = (params) = >{... };// Initialize the drag element
  initDrag = dragEle= >{... }// Initialize the release zone
  initDrop = dropEle= >{... }// Drag the element to register the event
  setDrag = el= >{... };// Release zone registration event
  setDrop = el= >{... }// Get the location of iframe
  getIframeOffset = () = > {
    const iframeEle = this.getIframe();
    return iframeEle
      ? this.getRealOffset(iframeEle)
      : { offsetLeft: 0.offsetTop: 0 };
  };

  // Recursively calculates the element's offset from its parent
  getRealOffset = (el, parentName) = > {
    let left = el.offsetLeft;
    let top = el.offsetTop;
    if(el.offsetParent && el.offsetParent.tagName ! == parentName) {const p = this.getRealOffset(el.offsetParent, parentName);
      left += p.offsetLeft;
      top += p.offsetTop;
    }
    return { offsetLeft: left, offsetTop: top };
  }

  // Get the element position
  getElOffset = el= > {
    const { offsetTop: iframeTop } = this.getIframeOffset();
    const { offsetTop: targetOffsetTop } = this.getRealOffset(el);
    return {
      midLine: el.clientHeight / 2 + targetOffsetTop + iframeTop,
      topLine: targetOffsetTop + iframeTop,
      bottomLine: el.clientHeight + targetOffsetTop + iframeTop
    };
  };

  // Release the position of the element inside the zone
  getDropOffset = () = > {
    const result = [];
    const { dropEle } = this.params;
    const el = dropEle.childNodes;

    let i = 0;
    while (i < el.length) {
      const midLine = this.getElOffset(el[i]);
      result.push(midLine);
      i += 1;
    }
    return result;
  };

  // Position comparison
  locationCompare = (ev) = > {
    let inside = false;
    const { dropEle } = this.params;
    console.log(ev.clientX);
    // Drag the position of the element
    const sourceRight = ev.clientX + this.mouseOffsetRight;
    const sourceLeft = sourceRight - ev.currentTarget.clientWidth;

    const { offsetLeft: iframeLeft } = this.getIframeOffset();
    const { offsetLeft: targetLeft } = this.getRealOffset(dropEle);

    /* The location of the release area */
    const targetOffsetLeft = iframeLeft + targetLeft;
    const targetOffsetRight = targetOffsetLeft + dropEle.clientWidth;

    if (sourceRight > targetOffsetLeft && sourceLeft < targetOffsetRight) {
      // Drag to the release area
      inside = true;
    } else {
      // Outside the release zone
      inside = false;
    }
    return inside;

  }

  // Insert placeholder elements
  insertPlaceholderEle = (sourceMidLine) = > {
    const dropOffset = this.getDropOffset(); // The location property of the release area
    const insertEl = this.createElePlaceholder();
    const { dropEle } = this.params;
    const dropEleChild = dropEle.childNodes;
    if (dropOffset.length) {
      dropOffset.map((item, i) = > {
        const Ele = dropEleChild[i];
        // Insert placeholder elements before the element
        if (sourceMidLine > item.topLine && sourceMidLine < item.midLine) {
          Ele.before(insertEl);
        }
        // Insert placeholder elements after the element
        if (sourceMidLine < item.bottomLine && sourceMidLine > item.midLine) {
          this.index = i + 1;
          Ele.after(insertEl);
        }
        // Appends a placeholder element
        if (sourceMidLine > dropOffset[dropOffset.length - 1].bottomLine) {
          dropEle.append(insertEl);
        }
        return item;
      });
    }
    // Insert the first placeholder element (when there is no component inside the iframe)
    if (!dropEleChild.length) {
      dropEle.append(insertEl);
    }
  }

  /****** Event handling ******/
  dragStartEvent = ev= > {
    // console.log(' Start dragging ');
    // Get the mouse distance below the drag element
    this.mouseOffsetBottom = ev.currentTarget.clientHeight - ev.offsetY;
    // Get the mouse distance to the right of the drag element
    this.mouseOffsetRight = ev.currentTarget.clientWidth - ev.offsetX;
  };

  dragEvent = ev= > {
    // Gets the distance between the line in the drag element and the top of the screen
    const sourceMidLine =
      ev.clientY + this.mouseOffsetBottom - ev.currentTarget.clientHeight / 2;
    if(this.locationCompare(ev)) {
      this.insertPlaceholderEle(sourceMidLine)
      console.log('Inside release zone')}else {
      this.removePlaceholderEle()
      console.log('Outside the release zone')}}; }export default new Drag();
Copy the code

The result is as follows:

At this point, the problem of non-stop flickering has been solved, as well as precise coordinate calculation, to achieve the insertion of elements up and down.

But there are still some problems:

  • As you can clearly see in the demo, you can insert placeholder elements when dragging elements to the right of the iframe, but when the mouse position enters the IFrame, the element is deleted again

What is the reason for this?

If we look at the printed mouse coordinates, we can see that when the mouse position enters the IFrame, ev.clientX changes to 0. Therefore, when the mouse position enters the IFrame, the iframe is used as the window. This causes the mouse position to mutate to 0, which causes the computed position to be biased so that the drag element is considered outside the release zone, so the placeholder element is removed.

How to solve this problem?

Several schemes came up:

  1. One is to listen for changes in coordinates and then recalculate positions to further compare positions.
  2. Zoom in the IFrame and the screen is greater than or equal to the screen size, dragging from the start to make the inside of the iframe.

Scheme analysis:

  • In the first scheme, the critical condition that the monitoring coordinate changes to 0 is unreliable, because the drag event is triggered every 50ms. Depending on how fast you move the mouse, the clientX obtained when the mouse enters iframe is inconsistent, so the first scheme is not feasible.
  • The second scheme, iframe amplification, is theoretically possible, so let’s try it. The main thing is to change the layout.

The code is as follows:

.drop-content {
  position: absolute;
  width: 100vw; //iframeZoom in as big as the windowheight: 100%;
}

#drop-box {
  width: 375px; //iframeThe inner element sets the widthmargin: 100px auto;

  .item{... }}Copy the code

The demo looks like thisAs you can see in the demo,Overrides the component area on the left. This is due to the high Z-index in the right view area.

Optimization scheme

There are two options

  • The element layout is shifted so that the DOM element in the right view area is placed in front of the component area.
  • Change the Z-Index to lower the z-index in the right view area
Plan 1

The core code

//drag.jsx
// Swap the two elements
<>
  <div className="drop-content">
    <iframe id="my-iframe" src="#/iframe" style={{ width: "100% ",height: "480px", border: "none}} "/ >
  </div>
  <div id="drag-box">
    <div className="drag-item">Drag the elements</div>
    <div className="drag-item">Drag the elements</div>
    <div className="drag-item">Drag the elements</div>
  </div>
</>
Copy the code

The effect after implementation

And as you can see, it solves the drag problem perfectly. But the layout was changed.

Scheme 2

The core code

.drop-content {
  position: absolute;
  z-index: -1; / / makeiframethez-indexA little lowwidth: 100vw; //iframeZoom in as big as the windowheight: 100%;
}

#drop-box {
  width: 375px; //iframeThe inner element sets the widthmargin: 100px auto;

  .item {
    width: 100%;
    height: 50px;
    background-color: # 875; }}Copy the code

The effect after implementationAs you can see in the demo, the drag and drop problem is solved perfectly, but the iframe element click event is not triggered.

On second thought, since z-index can solve clientX mutation problem, can we do it without enlarging iframe? This will not affect the event trigger, so let’s try it.

The core code

//drag.js
// Start dragging
dragStartEvent = ev= > {
  document.getElementsByClassName("drop-content") [0].style.zIndex =
    "1";
};

// End of drag
dragEndEvent = ev= > {
  ev.preventDefault();
  document.getElementsByClassName("drop-content") [0].style.zIndex = "0";
};

Copy the code

The demo looks like thisGood, because this also solves the drag problem perfectly without changing the dom position.

The rolling process

Will there be a problem when there are too many elements in the view area and a scroll bar appears on the page? Let’s try to write the height of the iframe a little higher

 <iframe id="my-iframe" src="#/iframe" style={{ width: "100% ",height: "880px", border: "none}} "/ >
Copy the code

The demo looks like this

As you can see in the demo, the page has a scrollbar, the view area is scrolling up, the top of the iframe is rolling up to the top of the screen, and when we drag the element to insert, we’re going to insert it out of place, is that calculation wrong again?

If you look closely at the code, when the top of the iframe rolls into the top of the screen, it will calculate a negative number, causing the calculation to be biased and the insertion placeholder to be misplaced.

// Recursively calculates the element's offset from its parent
getRealOffset = (el, parentName) = > {
  let left = el.offsetLeft;
  let top = el.offsetTop;
  if(el.offsetParent && el.offsetParent.tagName ! == parentName) {const p = this.getRealOffset(el.offsetParent, parentName);
    left += p.offsetLeft;
    top += p.offsetTop;
  }
  return { offsetLeft: left, offsetTop: top };
}
Copy the code

Optimized calculation scheme

The core code

// Calculates the element's offset from its parent
getRealOffset = (el, parentName) = > {
  const { left, top } = el.getBoundingClientRect();
  return { offsetLeft: left, offsetTop: top };
}
Copy the code

Use the getBoundingClientRect method to get the location of the specific window

Demonstrate the followingThis optimization, can be a perfect solution to the drag of some problems, the above two solutions are line.

Across the iframe communication

How do I get the data inside the iframe to update the rendering in real time after the drag element is inserted?

Here’s the idea:

  • Iframe mounts an update method
  • Within the drag-complete callback, call UPDATE, passing in data
  • Triggers rendering of elements inside an iframe
  1. Maintains a component’s data store, getStore, and setStore methods
//store.js
class Store {
  state = {
    list: []
  }
  getStore = () = > this.state
  setStore = (data) = > {
    this.state = { ... this.state, ... data } } }export default new Store()
Copy the code
  1. Data processing for component insertion, including, add and INSERT operations, and methods for updating iframe synchronously
// update.js
import Store from './store';

const add = (params) = > {
  const { list } = Store.getStore()
  Store.setStore({ list: [...list, params.data]})
};

const insert = (params) = > {
  const { list } = Store.getStore()
  const { index } = params;
  list.splice(index, 0, params.data)
  Store.setStore({ list: [...list] })
};

const update = {
  add,
  insert
}

// Update the iframe internal data method
const iframeUpdate = (params) = > {
  document.getElementById("my-iframe") &&
    document.getElementById("my-iframe").contentWindow &&
    document.getElementById("my-iframe").contentWindow.update &&
    document.getElementById("my-iframe").contentWindow.update(params);
}

export default (params) => {
  const{ type, ... argv } = params;if(! type)return Promise.reject()
  return new Promise(r= > r())
    .then(() = > update[type](argv))
    .then(() = > {
      const { list } = Store.getStore()
      iframeUpdate(list)
    })
}
Copy the code
  1. When you drag, the type of operation on the element and the position of the element to be inserted are passed through the callback function
//drag.js
class Drag {
  params = {}

  mouseOffsetBottom = 0;
  mouseOffsetRight = 0;

  index = 0; // Insert the index of the element
  type = 'add'; // Operation type

  init = (params) = >{... }; .// Calculates the element's offset from its parent
  getRealOffset = (el, parentName) = > {
    const { left, top } = el.getBoundingClientRect();
    return { offsetLeft: left, offsetTop: top };
  }

  // Get the element position
  getElOffset = el= > {
    const { offsetTop: iframeTop } = this.getIframeOffset();
    const { offsetTop: targetOffsetTop } = this.getRealOffset(el);
    return {
      midLine: el.clientHeight / 2 + targetOffsetTop + iframeTop,
      topLine: targetOffsetTop + iframeTop,
      bottomLine: el.clientHeight + targetOffsetTop + iframeTop
    };
  };

  // Release the position of the element inside the zone
  getDropOffset = () = > {
    const result = [];
    const { dropEle } = this.params;
    const el = dropEle.childNodes;

    let i = 0;
    while (i < el.length) {
      const midLine = this.getElOffset(el[i]);
      result.push(midLine);
      i += 1;
    }
    returnresult; }; .// Insert placeholder elements
  insertPlaceholderEle = (sourceMidLine) = > {
    const dropOffset = this.getDropOffset(); // The location property of the release area
    const insertEl = this.createElePlaceholder();
    const { dropEle } = this.params;
    const dropEleChild = dropEle.childNodes;
    if (dropOffset.length) {
      dropOffset.map((item, i) = > {
        const Ele = dropEleChild[i];
        // Insert placeholder elements before the element
        if (sourceMidLine > item.topLine && sourceMidLine < item.midLine) {
          Ele.before(insertEl);
          this.index = i;
          this.type = 'insert'
        }
        // Insert placeholder elements after the element
        if (sourceMidLine < item.bottomLine && sourceMidLine > item.midLine) {
          this.index = i + 1;
          Ele.after(insertEl);
          this.type = 'insert'
        }
        // Appends a placeholder element
        if (sourceMidLine > dropOffset[dropOffset.length - 1].bottomLine) {
          dropEle.append(insertEl);
          this.type = 'add'
        }
        return item;
      });
    }
    // Insert the first placeholder element (when there is no component inside the iframe)
    if(! dropEleChild.length) {this.type = 'add'dropEle.append(insertEl); }}/****** Event handling ******/
  // Start dragging
  dragStartEvent = ev= > {
    document.getElementsByClassName("drop-content") [0].style.zIndex =
      "1";
    // Get the mouse distance below the drag element
    this.mouseOffsetBottom = ev.currentTarget.clientHeight - ev.offsetY;
    // Get the mouse distance to the right of the drag element
    this.mouseOffsetRight = ev.currentTarget.clientWidth - ev.offsetX;
  };

  dragEvent = ev= > {
    // Gets the distance between the line in the drag element and the top of the screen
    const sourceMidLine =
      ev.clientY + this.mouseOffsetBottom - ev.currentTarget.clientHeight / 2;
    if(this.locationCompare(ev)) {
      this.insertPlaceholderEle(sourceMidLine)
      // console.log(' Free zone inside ')
    } else {
      this.removePlaceholderEle()
      // console.log(' outside of free zone ')}};// End of drag
  dragEndEvent = ev= > {
    ev.preventDefault();
    document.getElementsByClassName("drop-content") [0].style.zIndex = "0";
    const { callback } = this.params;
    this.locationCompare(ev) &&
      callback &&
      callback({
        type: this.type,
        index: this.index
      });
  };
}

export default new Drag();
Copy the code
  1. Update is called after the drag to update the data source
//drag.jsx
import React, { useState, useEffect } from 'react';
import Drag from './drag';
import update from '@/store/update';

require('./styles.less');

//iframe hooks
const useIframeLoad = () = >{...//iframe load state hooks
  return iframeState;
}

export default() = > {const callback = params= >{ update({ ... params,data: { name: new Date().getTime() } })
  }

  const init = () = > {
    Drag.init({
      dragEle: document.getElementById('drag-box'),
      dropEle: document.getElementById('my-iframe').contentDocument.getElementById('drop-box'),
      callback
    })
  }

  useIframeLoad() && init();
  return <>.</>
}
Copy the code
  1. The iframe internal update method is called, which triggers data updates and component rendering.
//iframe.jsx
import React, { useState } from 'react';

require('./styles.less');

export default() = > {const [list, setList] = useState([]);

  // Mount the update method, cross-iframe data transfer, update
  window.update = params= > {
    setList(params);
  }

  return <div id="drop-box">
    {
      list.map((item) =>
        <div className="item" key={item.name} onClick={()= >Alert (' click event ')}> element {item.name}</div>)}</div>
}
Copy the code

The demo looks like this

Finally, drag and drop and communication across iframes are implemented.

conclusion

The operation page building drag communication function is completed in the continuous upgrade. It involves the following points:

  • The element enters the view area and determines the x coordinates of the left side of the iframe from the screen < the x coordinates of the right side of the dragged element from the screen < the x coordinates of the right side of the iframe from the screen.
  • < iframe the y coordinates of the inner line of the element being dragged from the screen to the front of the insert, and the y coordinates of the inner line of the element being dragged from the screen to the back of the insert.
  • Z-index resolution of clientX coordinate mutation problem.
  • Scroll position problem getBoundingClientRect resolved.

I hope this article is helpful to you. Welcome to share with us.

Progress a little bit every day, quality change from paying attention to the big round FE start.