There are many reasons for you to include drag and drop in your page, the easiest of which is to reorganize your data. For example, you might want the user to be able to reorganize a series of page elements, and placing an input or select component next to each element to represent their order is an alternative, making it possible for the group of elements to be dragged and dropped. Or maybe you want to have a navigation window on the website that can be moved by the user. These are all simple reasons to use the drag and drop function because you can do it!
The effect of dragging and dropping on your web page is not complicated. First, we know the location of the mouse, and then we need to understand when the user clicks an element so that we know we want to prepare to start dragging it, and finally we want to move the element.
Capture the mouse movement
In the first step, we need to obtain the coordinates of the mouse, and use a function and assign it to achieve this function:
Code
= mouseMove;
function mouseMove(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
}
function mouseCoords(ev) {
if( || ) {
return {x:, y:};
}
return {
x: + - ,
y: + -
};
}
First we need to explain the event object. No matter when you move, click the mouse, press the key, etc., an event will happen. In IE, this event is global, it is stored in, and for Firefox, and other browsers, this event will be passed to any function pointing to this page action. Therefore, we make the function pointing to the mouse move, and the function moving the mouse gets the event object.
In the above code, ev contains event objects in all browser environments. In Firefox, "||" will be ignored because it already contains events. In IE, the value of ev is so empty that it needs to be set to.
In this article, we need to capture the coordinates of the mouse multiple times, so we wrote a mouseCoords method, which has a parameter: event.
We want to discuss the differences between IE and other browsers again. Firefox and other browsers use and represent the location of the mouse relative to the document document. If you have a window of 500*500 and the mouse is in the middle of the window, the values of pageX and pageY will be 250. If you scroll down the window by 500 pixels, the value of pageY is 750.
In contrast, Microsoft's IE uses and represents the mouse's position relative to the window window, rather than the current document document. In the same example, if the mouse is placed in the middle of the 500*500 window, both clientX and clientY values will be 250. If you scroll down the page, clientY will still be 250 because it is measured relative to the window window, not the current document document. Therefore, in the mouse position, we should introduce the scrollLeft and scrollTop properties of the document body area. Finally, the document document in IE is not actually at the (0,0) position, there is a small (usually 2px) border around it, and the width of the border is included, so they need to be introduced in the mouse position.
Fortunately, now that we have the mouseCoords function, we no longer have to worry about getting the mouse position.
Capture mouse clicks
Next, we must know when the mouse is clicked and when it is released. If we skip this step, as long as your mouse moves through these elements, it will produce the effect of dragging them, which is annoying and contrary to human intuition.
Here, there are two functions that can help us: onmousedown and onmouseup. We've previously pointed to a function, so it seems logically that both the sum should be pointed to the function. If we let point to a function, the function will be executed because the mouse clicks on any element: text, image, table, etc. We just want specific elements in the page to have drag and drop functions, so we can do this by:
Code
= mouseUp;
var dragObject = null;
function makeClickable(object) {
= function() {
dragObject = this;
}
}
function mouseUp(ev) {
dragObject = null;
}
We now have a variable dragObject that contains any element you click on. When you release the mouse, dragObject is set to empty, so when dragObject is not empty, we need to drag it.
Move elements
We now know how to capture mouse movement and clicks. The next thing you need to do is move any element we want to drag. First, move an element exactly where we want it on the page, and the position value of the element style sheet must be absolute, which means you can set its or, measurements relative to the upper left corner of the page, because all our mouse movements are relative to the upper left corner of the page, which is usually the case.
Once we set ='absolute', we need to change the position of the top and left of the element to move it!
Code
= mouseMove;
= mouseUp;
var dragObject = null ;
var mouseOffset = null ;
function getMouseOffset(target, ev) {
ev = ev || ;
var docPos = getPosition(target);
var mousePos = mouseCoords(ev);
return {x: - , y: - } ;
}
function getPosition(e) {
var left = 0;
var top = 0;
while () {
left += ;
top += ;
e = ;
}
left += ;
top += ;
return {x:left, y:top} ;
}
function mouseMove(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
if (dragObject) {
= 'absolute';
= - ;
= - ;
return false ;
}
}
function mouseUp() {
dragObject = null ;
}
function makeDraggable(item) {
if (!item) return ;
= function (ev) {
dragObject = this ;
mouseOffset = getMouseOffset(this, ev);
return false ;
}
}
You will notice that these codes are based on our previous example (see the previous article), putting them together and you will be able to move the elements at will.
When we click on an element, another variable is stored, mouseOffset. mouseOffset simply contains the location information of our click elements. If we have an image of 20*20px and click on the middle of the image, mouseOffset should be {x:10, y:10}. If we click on the upper left corner of the image, mouseOffset should be {x:0, y:0}. We use it in the position information after the mouse moves. If we do not store this value, no matter which position you click on the element, the position of the element relative to the mouse will be the same.
The mouseOffset function uses another function getPosition. getPosition is intended to return the coordinate position of the element relative to the document. If we simply read or, what we get will be the position of the element relative to its parent element, not the document. In our script, all elements are relative to the document document, so this is needed.
To complete the work of getting the element relative to the document's position, getPosition starts with its own parent, loops to get its left and top values and accumulates, so we get the accumulated values of the element we want from the top and left of the document.
When we get this information and move the mouse, mouseMove starts running. First we need to ensure that the value is absolute. Then, we move the element to any place, and the mouse position will subtract the offset of the mouse we recorded from the element. When the mouse is released, dragObject will be set to null and the mouseMove function does nothing anymore.
Place elements
Our previous example has dealt with this problem by simply dragging an element and then dropping it. Then, there are often other purposes when we drop an element, we use dragging the element to the recycle bin as an example, or we might want the element to align with a specific area in the page.
Unfortunately we are getting into a relatively major problem here. Because the elements we are moving are always directly under our mouse, and it is impossible to trigger mouseover, mousedown, mouseup or other elements in the page. If you move an element to the Recycle Bin, your mouse will keep moving the element above, not the Recycle Bin.
So how should we deal with this problem? Here are several solutions. The purpose of mouseOffset mentioned above is to ensure that the element is always in the correct position below the mouse. If you ignore this and always make the element in the lower right corner of the mouse, your mouse will not be hidden by the element you are dragging, and we will not encounter any problems. But in fact, this is often not the case. For the sake of aesthetics, we usually have to keep the elements under the mouse.
Another option is to not move the element you are dragging, you can change the mouse style to tell the user that you are dragging an element until you place it somewhere. This solves our problem, but brings the same problem that the previous solution faces: aesthetics.
Our last solution does not affect the elements you are moving, nor does it affect the elements at the end point (such as the garbage bin). Unfortunately, this is more difficult than the previous two solutions. What we are going to do is get a set of targets we want to place. When the mouse is released, we manually check the current position of the mouse relative to each target to see if the mouse is released at the position of a target in this target. If so, we know that we have placed the element on our target.
Code
/*
All code from the previous example is needed with the exception
of the mouseUp function which is replaced below
*/
var dropTargets = [];
function addDropTarget(dropTarget) {
(dropTarget);
}
function mouseUp(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
for (var i = 0; i < ; i ++) {
var curTarget = dropTargets[i];
var targPos = getPosition(curTarget);
var targWidth = parseInt();
var targHeight = parseInt();
if (
( > ) &&
(mousePos. < ( + targWidth)) &&
( > ) &&
( < ( + targHeight))) {
// dragObject was dropped onto curTarget!
}
}
dragObject = null ;
}
In this example, when the mouse is released, we loop through each target that may place the element. If the mouse pointer is on the target, we have an event of placing the element. We judge the horizontal coordinate of the mouse is greater than the horizontal coordinate on the left of the target element (>) and less than the horizontal coordinate on the right of the target element (<(+ targWidth)). We make the same judgment on the Y coordinate. If all these values return true, then our mouse is within the scope of the target element.
Integrate all functions
Finally, we use all code snippets to create a complete drag-and-drop function script. The first thing we have to do is DOM operations. If you are not very familiar with this, you can read "JavaScript Primer on DOM Manipulation".
The following code creates containers and container groups so that each element can be dragged in these containers, which is done based on the second demo in this article. This code can be used to re-order elements, place the navigation window on the left or right of the page, or add other features you can think of.
We will use pseudo-code to explain step by step, leaving the real code to readers to view through comments.
1. When the document is loaded for the first time, we create a DIV tag called dragHelper. When we start moving an element, dragHelper will become a hidden element that can be moved around. The real element is not dragged, just use insertBefore and appendChild to move. We hide dragHelper at the beginning.
2. We create mouseDown and mouseUp functions. At first, all of these functions assume that the state of the mouse button is recorded, so that the iMouseDown variable is true when the mouse is pressed and false when it is not pressed.
3. We create a global variable DragDrops, and a function CreateDragContainer. DragDrops contains a set of interrelated containers. Any variables (representing containers) passed into CreateDragContainer are organized into a new collection, allowing elements to move freely between these containers. Through setAttribute, the CreateDragContainer function also binds elements in each container together.
4. Now our code knows the collection where each element is, now look at the mouseMove function. The mouseMove function first sets a variable target to represent the target element below the mouse. If this element is in the set (judged with getAttribute), continue the following operations:
4.1. First, when necessary, we run a simple script to change the class attributes of the target element, which creates a flip effect.
4.2. Then we check if the mouse clicks (because our code has run here) if the event occurs:
4.2.1. Set the variable curTarget to the current element.
4.2.2. Record the current position of the element in the document so that its value can be retrieved when needed.
4.2.3.Clone the current element into dragHelper, so that we can move the hidden backup of the element.
4.2.4. Because in dragHelper, we have a backup of the drag element, and this element will always be in the mouse. We must remove the dragObj attribute to let the code know that dragObj is no longer in the collection.
4.2.5. We quickly record the current position, width and height of each element in the collection. When the element starts to be dragged for the first time, we only need to do this once, otherwise we have to do it once every time the mouse moves, even hundreds of times in a second.
4.3. If the mouse does not click, either we have the same target element as before, or we do not have the target element, and in either case we will not do anything.
5. Now we check the curTarget variable. curTarget should contain only one object being dragged, so if it exists, it means we are dragging an element:
5.1. Move the hidden DIV to the mouse, and this element can be dragged just like the element created in the previous article.
5.2. Then we check whether the mouse exists in each container in the current collection.
5.2.1. If the mouse is in a container, we check each element in the container to see where the element we are dragging belongs.
5.2.2. Then we place the dragged element in front of another element in the container, or the last position of the container.
5.2.3. Finally, we make sure that the element is visible.
6. The rest is to capture the mouseUp event:
6.1. First, dragHelper needs to be hidden: it is no longer needed because we are not dragging anything.
6.2. If the dragged element is visible, it already exists in any container it belongs to, and all work has been completed.
6.3. If the dragged element is not visible, we put it back where it was originally located.
The effect of dragging and dropping on your web page is not complicated. First, we know the location of the mouse, and then we need to understand when the user clicks an element so that we know we want to prepare to start dragging it, and finally we want to move the element.
Capture the mouse movement
In the first step, we need to obtain the coordinates of the mouse, and use a function and assign it to achieve this function:
Code
Copy the codeThe code is as follows:
= mouseMove;
function mouseMove(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
}
function mouseCoords(ev) {
if( || ) {
return {x:, y:};
}
return {
x: + - ,
y: + -
};
}
First we need to explain the event object. No matter when you move, click the mouse, press the key, etc., an event will happen. In IE, this event is global, it is stored in, and for Firefox, and other browsers, this event will be passed to any function pointing to this page action. Therefore, we make the function pointing to the mouse move, and the function moving the mouse gets the event object.
In the above code, ev contains event objects in all browser environments. In Firefox, "||" will be ignored because it already contains events. In IE, the value of ev is so empty that it needs to be set to.
In this article, we need to capture the coordinates of the mouse multiple times, so we wrote a mouseCoords method, which has a parameter: event.
We want to discuss the differences between IE and other browsers again. Firefox and other browsers use and represent the location of the mouse relative to the document document. If you have a window of 500*500 and the mouse is in the middle of the window, the values of pageX and pageY will be 250. If you scroll down the window by 500 pixels, the value of pageY is 750.
In contrast, Microsoft's IE uses and represents the mouse's position relative to the window window, rather than the current document document. In the same example, if the mouse is placed in the middle of the 500*500 window, both clientX and clientY values will be 250. If you scroll down the page, clientY will still be 250 because it is measured relative to the window window, not the current document document. Therefore, in the mouse position, we should introduce the scrollLeft and scrollTop properties of the document body area. Finally, the document document in IE is not actually at the (0,0) position, there is a small (usually 2px) border around it, and the width of the border is included, so they need to be introduced in the mouse position.
Fortunately, now that we have the mouseCoords function, we no longer have to worry about getting the mouse position.
Capture mouse clicks
Next, we must know when the mouse is clicked and when it is released. If we skip this step, as long as your mouse moves through these elements, it will produce the effect of dragging them, which is annoying and contrary to human intuition.
Here, there are two functions that can help us: onmousedown and onmouseup. We've previously pointed to a function, so it seems logically that both the sum should be pointed to the function. If we let point to a function, the function will be executed because the mouse clicks on any element: text, image, table, etc. We just want specific elements in the page to have drag and drop functions, so we can do this by:
Code
Copy the codeThe code is as follows:
= mouseUp;
var dragObject = null;
function makeClickable(object) {
= function() {
dragObject = this;
}
}
function mouseUp(ev) {
dragObject = null;
}
We now have a variable dragObject that contains any element you click on. When you release the mouse, dragObject is set to empty, so when dragObject is not empty, we need to drag it.
Move elements
We now know how to capture mouse movement and clicks. The next thing you need to do is move any element we want to drag. First, move an element exactly where we want it on the page, and the position value of the element style sheet must be absolute, which means you can set its or, measurements relative to the upper left corner of the page, because all our mouse movements are relative to the upper left corner of the page, which is usually the case.
Once we set ='absolute', we need to change the position of the top and left of the element to move it!
Code
Copy the codeThe code is as follows:
= mouseMove;
= mouseUp;
var dragObject = null ;
var mouseOffset = null ;
function getMouseOffset(target, ev) {
ev = ev || ;
var docPos = getPosition(target);
var mousePos = mouseCoords(ev);
return {x: - , y: - } ;
}
function getPosition(e) {
var left = 0;
var top = 0;
while () {
left += ;
top += ;
e = ;
}
left += ;
top += ;
return {x:left, y:top} ;
}
function mouseMove(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
if (dragObject) {
= 'absolute';
= - ;
= - ;
return false ;
}
}
function mouseUp() {
dragObject = null ;
}
function makeDraggable(item) {
if (!item) return ;
= function (ev) {
dragObject = this ;
mouseOffset = getMouseOffset(this, ev);
return false ;
}
}
You will notice that these codes are based on our previous example (see the previous article), putting them together and you will be able to move the elements at will.
When we click on an element, another variable is stored, mouseOffset. mouseOffset simply contains the location information of our click elements. If we have an image of 20*20px and click on the middle of the image, mouseOffset should be {x:10, y:10}. If we click on the upper left corner of the image, mouseOffset should be {x:0, y:0}. We use it in the position information after the mouse moves. If we do not store this value, no matter which position you click on the element, the position of the element relative to the mouse will be the same.
The mouseOffset function uses another function getPosition. getPosition is intended to return the coordinate position of the element relative to the document. If we simply read or, what we get will be the position of the element relative to its parent element, not the document. In our script, all elements are relative to the document document, so this is needed.
To complete the work of getting the element relative to the document's position, getPosition starts with its own parent, loops to get its left and top values and accumulates, so we get the accumulated values of the element we want from the top and left of the document.
When we get this information and move the mouse, mouseMove starts running. First we need to ensure that the value is absolute. Then, we move the element to any place, and the mouse position will subtract the offset of the mouse we recorded from the element. When the mouse is released, dragObject will be set to null and the mouseMove function does nothing anymore.
Place elements
Our previous example has dealt with this problem by simply dragging an element and then dropping it. Then, there are often other purposes when we drop an element, we use dragging the element to the recycle bin as an example, or we might want the element to align with a specific area in the page.
Unfortunately we are getting into a relatively major problem here. Because the elements we are moving are always directly under our mouse, and it is impossible to trigger mouseover, mousedown, mouseup or other elements in the page. If you move an element to the Recycle Bin, your mouse will keep moving the element above, not the Recycle Bin.
So how should we deal with this problem? Here are several solutions. The purpose of mouseOffset mentioned above is to ensure that the element is always in the correct position below the mouse. If you ignore this and always make the element in the lower right corner of the mouse, your mouse will not be hidden by the element you are dragging, and we will not encounter any problems. But in fact, this is often not the case. For the sake of aesthetics, we usually have to keep the elements under the mouse.
Another option is to not move the element you are dragging, you can change the mouse style to tell the user that you are dragging an element until you place it somewhere. This solves our problem, but brings the same problem that the previous solution faces: aesthetics.
Our last solution does not affect the elements you are moving, nor does it affect the elements at the end point (such as the garbage bin). Unfortunately, this is more difficult than the previous two solutions. What we are going to do is get a set of targets we want to place. When the mouse is released, we manually check the current position of the mouse relative to each target to see if the mouse is released at the position of a target in this target. If so, we know that we have placed the element on our target.
Code
Copy the codeThe code is as follows:
/*
All code from the previous example is needed with the exception
of the mouseUp function which is replaced below
*/
var dropTargets = [];
function addDropTarget(dropTarget) {
(dropTarget);
}
function mouseUp(ev) {
ev = ev || ;
var mousePos = mouseCoords(ev);
for (var i = 0; i < ; i ++) {
var curTarget = dropTargets[i];
var targPos = getPosition(curTarget);
var targWidth = parseInt();
var targHeight = parseInt();
if (
( > ) &&
(mousePos. < ( + targWidth)) &&
( > ) &&
( < ( + targHeight))) {
// dragObject was dropped onto curTarget!
}
}
dragObject = null ;
}
In this example, when the mouse is released, we loop through each target that may place the element. If the mouse pointer is on the target, we have an event of placing the element. We judge the horizontal coordinate of the mouse is greater than the horizontal coordinate on the left of the target element (>) and less than the horizontal coordinate on the right of the target element (<(+ targWidth)). We make the same judgment on the Y coordinate. If all these values return true, then our mouse is within the scope of the target element.
Integrate all functions
Finally, we use all code snippets to create a complete drag-and-drop function script. The first thing we have to do is DOM operations. If you are not very familiar with this, you can read "JavaScript Primer on DOM Manipulation".
The following code creates containers and container groups so that each element can be dragged in these containers, which is done based on the second demo in this article. This code can be used to re-order elements, place the navigation window on the left or right of the page, or add other features you can think of.
We will use pseudo-code to explain step by step, leaving the real code to readers to view through comments.
1. When the document is loaded for the first time, we create a DIV tag called dragHelper. When we start moving an element, dragHelper will become a hidden element that can be moved around. The real element is not dragged, just use insertBefore and appendChild to move. We hide dragHelper at the beginning.
2. We create mouseDown and mouseUp functions. At first, all of these functions assume that the state of the mouse button is recorded, so that the iMouseDown variable is true when the mouse is pressed and false when it is not pressed.
3. We create a global variable DragDrops, and a function CreateDragContainer. DragDrops contains a set of interrelated containers. Any variables (representing containers) passed into CreateDragContainer are organized into a new collection, allowing elements to move freely between these containers. Through setAttribute, the CreateDragContainer function also binds elements in each container together.
4. Now our code knows the collection where each element is, now look at the mouseMove function. The mouseMove function first sets a variable target to represent the target element below the mouse. If this element is in the set (judged with getAttribute), continue the following operations:
4.1. First, when necessary, we run a simple script to change the class attributes of the target element, which creates a flip effect.
4.2. Then we check if the mouse clicks (because our code has run here) if the event occurs:
4.2.1. Set the variable curTarget to the current element.
4.2.2. Record the current position of the element in the document so that its value can be retrieved when needed.
4.2.3.Clone the current element into dragHelper, so that we can move the hidden backup of the element.
4.2.4. Because in dragHelper, we have a backup of the drag element, and this element will always be in the mouse. We must remove the dragObj attribute to let the code know that dragObj is no longer in the collection.
4.2.5. We quickly record the current position, width and height of each element in the collection. When the element starts to be dragged for the first time, we only need to do this once, otherwise we have to do it once every time the mouse moves, even hundreds of times in a second.
4.3. If the mouse does not click, either we have the same target element as before, or we do not have the target element, and in either case we will not do anything.
5. Now we check the curTarget variable. curTarget should contain only one object being dragged, so if it exists, it means we are dragging an element:
5.1. Move the hidden DIV to the mouse, and this element can be dragged just like the element created in the previous article.
5.2. Then we check whether the mouse exists in each container in the current collection.
5.2.1. If the mouse is in a container, we check each element in the container to see where the element we are dragging belongs.
5.2.2. Then we place the dragged element in front of another element in the container, or the last position of the container.
5.2.3. Finally, we make sure that the element is visible.
6. The rest is to capture the mouseUp event:
6.1. First, dragHelper needs to be hidden: it is no longer needed because we are not dragging anything.
6.2. If the dragged element is visible, it already exists in any container it belongs to, and all work has been completed.
6.3. If the dragged element is not visible, we put it back where it was originally located.