先说下思路: 从0搭建的话,需要先考虑下面这些问题:
HTML结构:首先在HTML中创建一个容器,用于包含整个树控件的结构。可以使用div 和备注不同的class名来表示树的层级结构。每个列表项中可以包含一个展开/折叠的图标、一个节点名称等元素。
CSS样式:添加样式来美化树控件。可以使用CSS选择器来选择各个元素,并设置宽度、高度、颜色、边框等属性。还可以添加js函数来添加class来设置展开/折叠图标的样式。
JavaScript逻辑:使用JavaScript编写树控件的逻辑。需要实现以下功能:
树的数据结构:定义一个代表树的数据结构,可以使用对象、数组等来表示节点之间的层级关系。初始化及渲染:在页面加载时,通过JavaScript将树的数据结构生成树的DOM结构,并进行初始化渲染。可以根据需要设置初始展开状态、选中状态等。
添加事件监听器:为树的各个交互元素(展开/折叠图标)添加事件监听器,以便响应用户的操作,并更新相关的数据和DOM元素
根据需要进行样式和功能的优化:依据实际需求,对树控件进行样式和功能上的优化,如添加动画效果、优化性能等
示例图片:
现在开始上代码:
css部分:
.tree-img{
width: 14px;
}
.tree-ul{
padding-left: 30px;
margin: 0;
}
.tree-li-info a{
display: block;
margin-right: 4px;
}
.img-a{
width: 14px;
height: 14px;
background-image: url();
background-repeat: no-repeat;
background-color: transparent;
background-size: cover; /* 缩放背景图片以覆盖容器 */
}
.tree-li-info{
width: 100%;
display: inline-flex;
align-items: center;
color: #555;
line-height: 22px;
font-size: 14px;
}
.tree-li-info:hover{
background-color: #eee;
}
.no-expand-img{
transform:rotate(-90deg);
}
.expand-img{
transform:rotate(0deg)
}
.no-expand{
display: none;
}
.expand{
display: block;
}
.selected{
background-color: #eee;
}
html部分
<div class="tree" id="tree">
<div id="div"></div>
</div>
js部分
const list=[{'name':'李四','child':[{'name':'王五'},{'name':'张三','child':[{'name':'王五'},{'name':'张三','child':[{'name':'王五'},{'name':'张三'}]}]}]},{'name':'王五','child':[{'name':'王五'},{'name':'张三'}]}]
let div=document.getElementById('div');
function generateNestedList(data,field,level,callback) {
let treeUl = document.createElement('div');
treeUl.classList.add('tree-ul');
if (level > 0) {
treeUl.classList.add('no-expand');
}
data.forEach(item => {
let treeLi = document.createElement('div');
treeLi.classList.add('tree-li');
let treeLiInfo = document.createElement('div');
treeLiInfo.classList.add('tree-li-info');
treeLiInfo.setAttribute('data-level', level);
treeLiInfo.addEventListener("click", function(event) {
callback(item,level)
})
if (item.child && item.child.length > 0) {
let imgA = document.createElement('a');
imgA.classList.add('img-a');
imgA.classList.add('no-expand-img');
treeLiInfo.appendChild(imgA);
}
let nameA = document.createElement('a');
nameA.textContent = item[field];
treeLiInfo.appendChild(nameA);
treeLi.appendChild(treeLiInfo);
if (item.child && item.child.length > 0) {
let childTreeUl = generateNestedList(item.child,field, level + 1,callback);
treeLi.appendChild(childTreeUl);
}
treeUl.appendChild(treeLi);
});
return treeUl;
}
// function generateNestedList(data,level) {
// let result='';
// if(level>0){
// result+= '<div class="tree-ul no-expand">';
// }else{
// result+= '<div class="tree-ul ">';
// }
// data.forEach(item => {
// result += '<div class="tree-li">';
// result += '<div class="tree-li-info" data-level='+level+'>';
// if (item.child && item.child.length > 0) {
// result+=' <a class="img-a no-expand-img"></a>';
// }
// // result += ' <input type="checkbox" />'
// result += ' <a>'+item.name+'</a>'
// result += '</div>';
// if (item.child && item.child.length > 0) {
// result += generateNestedList(item.child,level+1);
// }
// result += '</div>';
// });
// result += '</div>';
// return result;
// }
function clickNode(item,level){
console.log('点击了',item,'层级是',level)
}
//开始渲染
const nestedList = generateNestedList(list,'name',0,clickNode);
div.appendChild(nestedList);
const listItem = document.querySelectorAll(".img-a");
listItem.forEach(function(item) {
item.addEventListener("click", function(event) {
event.stopPropagation();
if(item.classList.contains("no-expand-img")){
item.classList.remove("no-expand-img");
item.classList.add("expand-img")
}else{
item.classList.remove("expand-img");
item.classList.add("no-expand-img")
}
expand(item);
});
});
function expand(e){
let node=e.parentNode.parentNode;
const childNodes = Array.from(node.childNodes);
childNodes.forEach(node => {
if(node.classList.contains('tree-ul')){
if(node.classList.contains("no-expand")){
node.classList.remove("no-expand");
node.classList.add("expand")
}else{
node.classList.remove("expand");
node.classList.add("no-expand")
}
}
});
}
const liItemsDiv = document.querySelectorAll('.tree-li-info');
// 为每个 <li> 元素添加事件监听器
liItemsDiv.forEach((div) => {
div.addEventListener('click', function(event) {
// 移除所有的 selected 类名
liItemsDiv.forEach((item) => {
item.classList.remove('selected');
});
div.classList.add('selected');
});
});