JavaScript implements draggable tree
As Web applications are increasingly used, we increasingly need to design more efficient methods for interaction with web pages. One of them is to use JavaScript to implement a draggable tree (Drag & Drop Tree). This article will introduce how to use JavaScript to create a draggable tree, detailing the implementation process and related technical details.
1. Goals to be achieved
The draggable tree described in this article refers to a structure on the web page, which contains nodes of the tree structure, and we can drag and drop them. Reorganize their hierarchical relationships. To implement such a tree, the following two key aspects need to be completed.
- Implementing the tree structure
First we need to create nodes for the tree structure in the page and determine the levels and associations between the nodes. These contents can be represented using JSON. For example, we can store the structure of the tree in the following JSON format:
{ name: "节点A", children: [{ name: "子节点A1", children: [] }, { name: "子节点A2", children: [{ name: "子节点A2-1", children: [] }] }] }
When rendered as a tree structure, it should look like this:
- 节点A |- 子节点A1 |- 子节点A2 |- 子节点A2-1
- Implementing drag and drop functionality
Making nodes drag-and-drop requires the use of some JavaScript techniques. Regarding the drag and drop API, generally speaking, it involves three categories:
- Dragable elements
- Place target elements
- Drag data
Using these APIs, we can easily implement the drag-and-drop function of nodes.
2. Technical details
After understanding our goals, let’s now discuss the implementation details in detail. The following are the steps to implement:
- Build a tree structure
We need to create node elements first and add them to HTML, usually using ul and li element levels. accomplish. For each node, a li element is required, and more li elements must be nested in the child node ul. In order to associate the tree structure with JSON data, you can use the data-* attributes to store the JSON data in the corresponding li element.
<ul id="tree"> <li data-name="节点A"> <div class="node">节点A</div> <ul> <li data-name="子节点A1"> <div class="node">子节点A1</div> </li> <li data-name="子节点A2"> <div class="node">子节点A2</div> <ul> <li data-name="子节点A2-1"> <div class="node">子节点A2-1</div> </li> </ul> </li> </ul> </li> </ul>
- Add drag events to nodes
We need to add drag events for each node, including mousedown, dragstart, dragover, dragleave, drop and dragend. Among them, mousedown and dragstart events need to be processed before dragging starts, and subsequent processing is dragover, dragleave, drop and dragend respectively. The handling functions of these drag and drop events can be completed through event listeners.
// 获取所有节点并添加事件监听器 const nodes = document.querySelectorAll('.node'); nodes.forEach((node) => { node.addEventListener('mousedown', onMouseDown); node.addEventListener('dragstart', onDragStart); node.addEventListener('dragover', onDragOver); node.addEventListener('dragleave', onDragLeave); node.addEventListener('drop', onDrop); node.addEventListener('dragend', onDragEnd); });
- Implementing the processing function of the drag event
The processing function of the drag event is somewhat complicated, and each step of the operation needs to be carefully designed. The following are instructions for each step:
- mousedown: Record the element where the drag starts and set isDragged to true.
- dragstart: Set the data transmission type and the data to be transmitted. In addition, it is necessary to determine whether the drag operation can be performed based on the status of isDragged. To set the data transmission type, you can use the setData() method.
function onDragStart(event) { if (!isDragged) { draggedItem = event.currentTarget.parentNode; event.dataTransfer.setData('text/plain', event.currentTarget.dataset.name); isDragged = true; } }
- dragover: Prevent the default event and add the isOver attribute to the current element. This attribute indicates that the current element is placed above other elements and can be placed. Default events can be prevented through the event.preventDefault() method.
function onDragOver(event) { event.preventDefault(); if (!event.currentTarget.dataset.isOver) { event.currentTarget.parentNode.classList.add('over'); event.currentTarget.dataset.isOver = true; } }
- dragleave: Remove the over attribute of the current element, indicating that there is no mouse hovering over the element.
function onDragLeave(event) { if (event.currentTarget.dataset.isOver) { event.currentTarget.parentNode.classList.remove('over'); event.currentTarget.dataset.isOver = false; } }
- drop: Determine whether the placement operation can be performed based on whether the current element has the over attribute. If it doesn't work, exit directly; if it works, perform a placement operation and adjust the tree structure.
function onDrop(event) { event.preventDefault(); if (event.currentTarget.dataset.isOver) { const droppedItem = event.currentTarget.parentNode; const parent = draggedItem.parentNode; parent.removeChild(draggedItem); event.currentTarget.parentNode.insertBefore(draggedItem, droppedItem.nextSibling); } }
- dragend: implements the termination event of drag operation. In this event, reset the value of isDragged and remove all over attributes.
function onDragEnd(event) { event.currentTarget.parentNode.classList.remove('over'); event.currentTarget.dataset.isOver = false; isDragged = false; }
3. Complete code
Finally, we integrate the above Javascript code to display a complete draggable tree. You can use this code directly, copy it into a text editor, save it as an html file and run it in the browser.
可拖动的树 <ul id="tree"> <li data-name="节点A"> <div class="node">节点A</div> <ul> <li data-name="子节点A1"> <div class="node">子节点A1</div> </li> <li data-name="子节点A2"> <div class="node">子节点A2</div> <ul> <li data-name="子节点A2-1"> <div class="node">子节点A2-1</div> </li> </ul> </li> </ul> </li> </ul> <script> let draggedItem = null; let isDragged = false; const nodes = document.querySelectorAll('.node'); nodes.forEach((node) => { node.addEventListener('mousedown', onMouseDown); node.addEventListener('dragstart', onDragStart); node.addEventListener('dragover', onDragOver); node.addEventListener('dragleave', onDragLeave); node.addEventListener('drop', onDrop); node.addEventListener('dragend', onDragEnd); }); function onMouseDown(event) { event.preventDefault(); } function onDragStart(event) { if (!isDragged) { draggedItem = event.currentTarget.parentNode; event.dataTransfer.setData('text/plain', event.currentTarget.dataset.name); isDragged = true; } } function onDragOver(event) { event.preventDefault(); if (!event.currentTarget.dataset.isOver) { event.currentTarget.parentNode.classList.add('over'); event.currentTarget.dataset.isOver = true; } } function onDragLeave(event) { if (event.currentTarget.dataset.isOver) { event.currentTarget.parentNode.classList.remove('over'); event.currentTarget.dataset.isOver = false; } } function onDrop(event) { event.preventDefault(); if (event.currentTarget.dataset.isOver) { const droppedItem = event.currentTarget.parentNode; const parent = draggedItem.parentNode; parent.removeChild(draggedItem); event.currentTarget.parentNode.insertBefore(draggedItem, droppedItem.nextSibling); } } function onDragEnd(event) { event.currentTarget.parentNode.classList.remove('over'); event.currentTarget.dataset.isOver = false; isDragged = false; } </script>
Through the above code implementation, we successfully created a draggable tree structure. In the web page, users can easily adjust the tree structure by dragging and dropping. At the same time, we also introduced in detail various technical details during the implementation process. This is undoubtedly a very useful practical case for developers who are learning JavaScript.
The above is the detailed content of JavaScript implements draggable tree. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics



React combines JSX and HTML to improve user experience. 1) JSX embeds HTML to make development more intuitive. 2) The virtual DOM mechanism optimizes performance and reduces DOM operations. 3) Component-based management UI to improve maintainability. 4) State management and event processing enhance interactivity.

Article discusses connecting React components to Redux store using connect(), explaining mapStateToProps, mapDispatchToProps, and performance impacts.

The article discusses defining routes in React Router using the <Route> component, covering props like path, component, render, children, exact, and nested routing.

Vue 2's reactivity system struggles with direct array index setting, length modification, and object property addition/deletion. Developers can use Vue's mutation methods and Vue.set() to ensure reactivity.

Redux reducers are pure functions that update the application's state based on actions, ensuring predictability and immutability.

The article discusses Redux actions, their structure, and dispatching methods, including asynchronous actions using Redux Thunk. It emphasizes best practices for managing action types to maintain scalable and maintainable applications.

TypeScript enhances React development by providing type safety, improving code quality, and offering better IDE support, thus reducing errors and improving maintainability.

React components can be defined by functions or classes, encapsulating UI logic and accepting input data through props. 1) Define components: Use functions or classes to return React elements. 2) Rendering component: React calls render method or executes function component. 3) Multiplexing components: pass data through props to build a complex UI. The lifecycle approach of components allows logic to be executed at different stages, improving development efficiency and code maintainability.
