Drag and Drop#
TL;DR
Wunderbaum implements drag and drop according to the native HTML protocol. This allows interaction between different tree instances, browsers, and even native applcations.
Wunderbaum supports drag and drop of nodes within the tree and between trees.
It is also possible to drag nodes from the tree to other elements on the page or
vice versa. Even cross-window drag and drop is supported.
The implementation is purely based on the native
HTML Drag and Drop API.
Note that there is no automatic modification of nodes. Instead, the
drop
event is fired on the target tree and it is up to the application to
modify the tree accordingly.
Drag and Drop Events#
The following events are fired on the tree during drag and drop.
Info
Note that the dragStart
, drag
, and dragEnd
events are fired on the
tree that contains the dragged node (i.e. the source node).
The other events are fired on the tree that contains the drop target.
The events are named after the corresponding HTML Drag and Drop events. However, the event handlers are passed an object with the following properties:
e = {
type: "dnd.EVENTNAME",
node: // the source or target node, depending on the event type
event: // <the original HTML DragEvent>
}
These are events are sent in a typical drag and drop operation:
-
dragStart(e)
: Fired when a drag operation is started.
This event handler MUST be implemented by the application in order to enable dragging in general.
The handler can returnfalse
to prevent dragging the source node.
The handler can set thee.event.dataTransfer.effectAllowed
property in order to adjust copy/move/link behavior.
The handler can set thee.event.dataTransfer.dropEffect
property in order to adjust copy/move/link behavior. -
drag(e)
: Fired repeadedly during a drag operation.
We will hardly ever have to implement this handler. -
dragEnter(e)
: Fired when a dragged item enters a drop target.
This event handler MUST be implemented by the application in order to enable dropping in general.
The handler can returnfalse
to prevent the drop operation or return a set of drop regions to indicate which drop regions are allowed.
The handler can set thee.event.dataTransfer.dropEffect
property in order to adjust copy/move/link behavior. -
dragOver(e)
: Fired continuously when a dragged item is moved over a drop target.
We will hardly ever have to implement this handler.
The handler can set thee.event.dataTransfer.dropEffect
property in order to adjust copy/move/link behavior. -
dragLeave(e)
: Fired when a dragged node leaves a drop target.
We will hardly ever have to implement this handler. -
drop(e)
: Fired when a dragged node is dropped on a drop target.
This is the most important event handler. It is responsible for modifying the tree according to the drop operation.
e = {
type: "dnd.drop",
node: // the target node
event: // <the original HTML DragEvent>
region: // 'before', 'after', 'over'
suggestedDropMode: // 'before', 'after', 'appendChild'
// (compatible with node.moveTo() and .appendChild())
suggestedDropEffect: // 'copy', 'move', 'link'
sourceNode: // the source node if available
sourceNodeData: // the serialized data of the source node if any
}
Foreign source data can be retreived from the e.event.dataTransfer
object.
dragEnd(e)
: Fired when a drag operation is ended.
We will hardly ever have to implement this handler.
Related Tree Options#
See also
See also the API Documentation for DnD options and the live demo.
Examples#
Basic Drag and Drop#
Allow sorting of plain nodes:
const tree = new Wunderbaum({
// --- Common Options ---
...
dnd: {
dragStart: (e) => {
if (e.node.type === "folder") {
return false; // do not allow dragging folders
}
return true;
},
dragEnter: (e) => {
if (e.node.type === "folder") {
return "over";
}
return ["before", "after"];
},
drop: (e) => {
console.log(
`Drop ${e.sourceNode} => ${e.suggestedDropEffect} ${e.suggestedDropMode} ${e.node}`, e
);
e.sourceNode.moveTo(e.node, e.suggestedDropMode)
},
},
});
Basic Drag and Drop (move)#
const tree = new Wunderbaum({
// --- Common Options ---
...
dnd: {
effectAllowed: "all",
dropEffectDefault: "move",
guessDropEffect: true,
dragStart: (e) => {
// if (e.node.type === "folder") {
// return false;
// }
return true;
},
dragEnter: (e) => {
// console.log(`DragEnter ${e.event.dataTransfer.dropEffect} ${e.node}`, e);
// We can only drop 'over' a folder, so the source node becomes a child.
// We can drop 'before' or 'after' a non-folder, so the source node becomes a sibling.
if (e.node.type === "folder") {
// e.event.dataTransfer.dropEffect = "link";
return "over";
}
return ["before", "after"];
},
drag: (e) => {
// e.tree.log(e.type, e);
},
drop: (e) => {
console.log(
`Drop ${e.sourceNode} => ${e.suggestedDropEffect} ${e.suggestedDropMode} ${e.node}`,
e
);
switch (e.suggestedDropEffect) {
case "copy":
e.node.addNode(
{ title: `Copy of ${e.sourceNodeData.title}` },
e.suggestedDropMode
);
break;
case "link":
e.node.addNode(
{ title: `Link to ${e.sourceNodeData.title}` },
e.suggestedDropMode
);
break;
default:
e.sourceNode.moveTo(e.node, e.suggestedDropMode);
}
},
},
Related Methods#
util.foo()
Related CSS Rules#
div.wb-row.wb-drag-source {
/* The dragged node */
}
div.wb-row.wb-drop-target {
/* The current target node while dragging */
}
div.wb-row.wb-drop-target.wb-drop-before .wb-node .wb-icon::after {
/* Drop marker */
}
Code Hacks#