三.数据输出(树)
前言 --> 树组件特点
- 下拉菜单组件应该由两部分组成:
- 选中项的文本
- 待选菜单(默认隐藏)
- 它的主要功能包括:
- 节点可以无限延伸(递归)
- 可以展开/收起子节点
- 节点可以选中,选中父节点,它的所有子节点也全部被选中,同样,反选父节点,其所有子节点也取消选择
- 同一级所有子节点选中时,它的父级也自动选中,一直递归判断到根节点
1.目录结构
sh
├── tree
│ ├── tree-node-content.vue
│ ├── tree-node.vue
│ ├── tree.vue
│ └── index.js
2.组件封装
3.使用案例
3.1 全部加载
<template>
<div class="box-tree">
<vue-tree :data="treeData">
<template #root="{ row, handleExpand }">
<div class="root-content">
<div class="expand-wrapper" @click="handleExpand">
<div class="expand-status">
<div class="tips">
<img class="close" :src="images[!row.expand ? 0 : 1]" />
<img class="icon" :src="images[2]" />
</div>
</div>
<div class="detail-content">
<span class="title">{{ row.label }}</span>
</div>
</div>
<vue-input v-model="searchName" placeholder="请输入名称搜索" @search="handleSearch"
:suggest-data="suggestData" />
</div>
</template>
<template #branch="{ row, handleExpand }">
<div class="branch-content node-content" @click="handleExpand">
<div class="expand-status">
<div class="tips">
<img class="close" :src="images[!row.expand ? 3 : 4]" />
</div>
</div>
<div class="detail-content">
<span class="title">{{ row.label }}(共{{ row.children.length }}份)</span>
</div>
</div>
</template>
<template #leaf="{ row }">
<div class="leaf-content node-content">
<div class="text">
<div class="title">{{ row.label }}</div>
<div class="sub-title">{{ row.label }}</div>
</div>
<div class="handle" v-if="!row.children || !row.children.length">
<button class="button" @click="scanAndCorrect(row)">xxxx</button>
<button class="button" :class="{ disabled: !row.studyTaskId }" @click="viewReport(row)">
xxx
</button>
</div>
</div>
</template>
</vue-tree>
</div>
</template>
<script setup lang="ts">
import { ref } from "vue";
const searchName = ref('')
const isLoading = ref(true)
const treeData = ref([])
const request = () => {
treeData.value = [{
label: '000',
children: [
{
label: '111',
},
{
label: '2222',
},
{
label: '3333',
},
]
}]
}
request();
const suggestData = ref([
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
{ label: "xxxxxxxxxxxxxxxxx" },
]);
const handleSearch = (val: string) => {
request();
};
const images = [
"",
"",
"",
"",
"",
"",
];
</script>
<style lang="scss" scoped>
.box-tree {
width: auto;
overflow-x: auto;
:deep(.vue-tree) {
width: 750px;
}
}
</style>
3.2 按需加载
<template>
<div class="box-tree">
<vue-tree :data="treeData">
<template #root="{ row, expand, handleExpand }">
<div class="root-content light-blue">
<div class="expand-wrapper" @click="handleLoad(expand, handleExpand, row)">
<div class="expand-status">
<div class="tips">
<img class="close" :src="images[!expand ? 0 : 4]" />
<img class="icon" :src="images[5]" />
</div>
</div>
<div class="detail-content">
<span class="title">{{ row.label }}</span>
</div>
</div>
</div>
</template>
<template #branch="{ row, expand, handleExpand }">
<div class="branch-content node-content" @click="handleLoad(expand, handleExpand, row)">
<div class="expand-status">
<div class="tips">
<img class="close" :src="images[!expand ? 3 : 4]" />
</div>
</div>
<div class="detail-content">
<span class="title">{{ row.label }}(共{{ row.qstCount }}题)</span>
</div>
</div>
</template>
<template #leaf="{ row }">
<div class="leaf-content node-content">
<div class="text">
<div class="title">{{ row.label }}</div>
<div class="sub-title">{{ row.label }}</div>
</div>
<div class="handle" v-if="!row.children || !row.children.length">
<button class="button" @click="scanAndCorrect(row)">xxxx</button>
<button class="button" :class="{ disabled: !row.studyTaskId }"
@click="viewReport(row)">xx</button>
</div>
</div>
</template>
</vue-tree>
</div>
</template>
<script setup lang="ts">
import { ref } from "vue";
const isLoading = ref(true)
const treeData = ref([{
id: 1,
label: '111',
children: []
}])
/**
* 点击某节点,获取其子节点
* @param nodeProps
*/
const loadNode = async (nodeProps: any) => {
const { nodeState, node } = nodeProps;
if (!node.id || node.children.length !== 0) {
return;
}
nodeState.isLoading = true;
const apiRes = {
data: [
{
label: '3333'
},
{
label: '3333'
},
{
label: '3333'
},
{
label: '3333'
}]
};
const childes = apiRes.data.map((item) => ({
id: item.nodeId,
// @ts-ignore
label: item.label || item.pageName,
children: [],
...item,
}));
if (childes.length === 0) {
node.children = [{ id: "", label: "暂无数据", children: [] }];
} else {
node.children = childes;
}
nodeState.isLoading = false;
};
const handleLoad = async (expand, fn, node) => {
if (!expand) {
await loadNode({ node, nodeState: { isLoading: false } });
}
fn();
};
const images = [
"",
"",
"",
"",
"",
"",
];
</script>
<style lang="scss" scoped>
.box-tree {
width: auto;
overflow-x: auto;
:deep(.vue-tree) {
width: 750px;
}
}
</style>
总结
通过对前端组件的分析,需要重点关注组件中易变性对组件封装的影响,它会对组件的可复用性、可扩展性产生很大影响