Vue3 组合式实现 带连接线的Tree型 架构图(一级树形图)
创建组件名称 TreeNodeView.vue
<template>
<div class="tree-node">
<div class="node">{{ rootNodeName }}</div>
<div class="children" :style="childrenLineStyle">
<div class="child-node" v-for="node in childNodes" :key="node.id">
<div class="node">{{ node.name }}</div>
</div>
</div>
</div>
</template>
<script setup>
import {computed, ref, onMounted} from 'vue';
const props = defineProps({
childNodes: {
type: Array,
required: true
},
rootNodeName: {
type: String,
required: true,
},
});
const childNode = ref(null);
const childrenWidth = ref(null);
const childrenMarginLeft = ref(null);
const childrenLineStyle = computed(() => {
if (!childNode.value) {
return {};
}
const getWidth = (element) => {
const style = getComputedStyle(element);
return parseFloat(style.width) + parseFloat(style.marginLeft) + parseFloat(style.marginRight);
};
let allNodeWidth = Array.from(childNode.value).reduce((total, node) => total + getWidth(node), 0);
const firstNodeWidth = getWidth(childNode.value[0]) / 2;
let lastNodeWidth = 0;
if (childNode.value.length > 1) {
lastNodeWidth = getWidth(childNode.value[childNode.value.length - 1]) / 2;
}
childrenWidth.value = `${allNodeWidth - firstNodeWidth - lastNodeWidth}px`;
childrenMarginLeft.value = `${firstNodeWidth}px`;
return {};
});
onMounted(() => {
childNode.value = document.querySelectorAll('.child-node');
});
</script>
<style>
.tree-node {
display: inline-flex;
flex-direction: column;
align-items: center;
}
.node {
border: 1px solid #000;
padding: 5px;
position: relative;
}
.children {
display: flex;
justify-content: center;
position: relative;
margin-top: 45px;
}
.children::before {
content: '';
height: 25px;
border: 0.00001rem solid #000;
position: absolute;
top: -46px;
}
.child-node {
position: relative;
margin-left: 10px;
margin-right: 10px;
}
.child-node::before {
content: '';
position: absolute;
top: -20px;
left: 50%;
transform: translateX(-50%);
height: 20px;
border-left: 1px solid #000;
}
.children::after {
content: '';
position: absolute;
top: -20px;
left: 0;
right: 0;
border-top: 1px solid #000;
width: v-bind(childrenWidth);
margin-left: v-bind(childrenMarginLeft);
}
</style>
<template>
<div class="tree-node">
<TreeNodeView v-bind="nodeViewConfig"></TreeNodeView>
</div>
</template>
<script setup>
import TreeNodeView from "./components/TreeNodeView.vue";
const nodeViewConfig = {
rootNodeName: '根节点',
childNodes: [
{id: 1, name: '节点1-1'},
{id: 2, name: '节点222222221-2'},
{id: 3, name: '节点1-3'},
{id: 4, name: '节点1-4222222'},
{id: 5, name: '节点1-4'},
{id: 6, name: '节点1-4'}
]
}
</script>
<style scoped>
</style>