***忘记摘抄那个博客的了,闲的时候记录了下;现在vue3版本的element-ui已经有类似的组件了
<!--
-->
<template>
<div class="treeSelect">
<div
class="mask"
v-show="isShowSelect"
@click="isShowSelect = !isShowSelect"
></div>
<el-popover
placement="bottom-start"
:width="width"
trigger="manual"
v-model="isShowSelect"
@hide="popoverHide"
>
<el-tree
class="common-tree"
:style="style"
ref="tree"
:data="data"
:props="defaultProps"
:show-checkbox="multiple"
:node-key="nodeKey"
:check-strictly="checkStrictly"
:default-expand-all="defaultexpandall"
:expand-on-click-node="false"
:check-on-click-node="multiple"
:highlight-current="true"
:auto-expand-parent="true"
:default-expanded-keys="checkedKeys"
@node-click="handleNodeClick"
@check-change="handleCheckChange"
></el-tree>
<el-select
:style="selectStyle"
slot="reference"
ref="select"
:size="size"
v-model="selectedData"
:multiple="multiple"
:clearable="clearable"
:collapse-tags="collapseTags"
@click.native="isShowSelect = !isShowSelect"
@remove-tag="removeSelectedNodes"
@clear="removeSelectedNode"
@change="changeSelectedNodes"
@visible-change="changeIcon($event)"
class="tree-select"
>
<el-option
v-for="item in options"
:key="item.value"
:label="item.label"
:value="item.value"
></el-option>
</el-select>
</el-popover>
</div>
</template>
<script>
export default {
name: "tree-select",
props: {
data: {
type: Array,
default() {
return [];
},
},
defaultProps: {
type: Object,
default() {
return {};
},
},
defaultexpandall: {
type: Boolean,
default() {
return true;
},
},
multiple: {
type: Boolean,
default() {
return false;
},
},
clearable: {
type: Boolean,
default() {
return true;
},
},
collapseTags: {
type: Boolean,
default() {
return false;
},
},
nodeKey: {
type: String,
default() {
return "id";
},
},
checkStrictly: {
type: Boolean,
default() {
return false;
},
},
checkedKeys: {
type: Array,
default() {
return [];
},
},
size: {
type: String,
default() {
return "small";
},
},
width: {
type: Number,
default() {
return 220;
},
},
height: {
type: Number,
default() {
return 260;
},
},
single: {
type: Boolean,
default: false,
},
},
data() {
return {
isShowSelect: false,
options: [],
selectedData: [],
style: "height:" + this.height + "px;",
selectStyle: "width:" + this.width + "px;",
checkedIds: [],
checkedData: [],
};
},
mounted() {
this.initCheckedData();
},
methods: {
filterNode(value, data) {
if (!value) return true;
return data.label.indexOf(value) !== -1;
},
changeIcon(callBack) {
this.$emit("changeIcon", callBack);
},
setSelectOption(node) {
let nav_title1 = "",
nav_title2 = "",
nav_title3 = "",
nav_title4 = "";
switch (node.level) {
case 1:
nav_title1 = node.data.roomName;
nav_title2 = "";
nav_title3 = "";
nav_title4 = "";
break;
case 2:
nav_title1 = node.parent.data.roomName + " - ";
nav_title2 = node.data.roomName;
nav_title3 = "";
nav_title4 = "";
break;
case 3:
nav_title1 = node.parent.parent.data.roomName + " - ";
nav_title2 = node.parent.data.roomName + " - ";
nav_title3 = node.data.roomName;
nav_title4 = "";
break;
case 4:
nav_title1 = node.parent.parent.parent.data.roomName + " - ";
nav_title2 = node.parent.parent.data.roomName + " - ";
nav_title3 = node.parent.data.roomName + " - ";
nav_title4 = node.data.roomName;
break;
default:
break;
}
let value = nav_title1 + nav_title2 + nav_title3 + nav_title4;
let tmpMap = {};
tmpMap.value = node.key;
tmpMap.label = node.label;
this.options = [];
if (this.single) {
tmpMap.label = value;
}
this.options.push(tmpMap);
this.selectedData = node.key;
},
checkSelectedNode(checkedKeys) {
var item = checkedKeys[0];
this.$refs.tree.setCurrentKey(item);
var node = this.$refs.tree.getNode(item);
this.setSelectOption(node);
if (this.checkedKeys.length != 0) {
this.$emit("change", node);
}
},
checkSelectedNodes(checkedKeys) {
this.$refs.tree.setCheckedKeys(checkedKeys);
},
clearSelectedNode() {
this.selectedData = "";
this.$refs.tree.setCurrentKey(null);
},
clearSelectedNodes() {
var checkedKeys = this.$refs.tree.getCheckedKeys();
for (let i = 0; i < checkedKeys.length; i++) {
this.$refs.tree.setChecked(checkedKeys[i], false);
}
},
initCheckedData() {
if (this.multiple) {
if (this.checkedKeys.length > 0) {
this.checkSelectedNodes(this.checkedKeys);
} else {
this.clearSelectedNodes();
}
} else {
if (this.checkedKeys.length > 0) {
this.checkSelectedNode(this.checkedKeys);
} else {
this.clearSelectedNode();
}
}
},
popoverHide() {
if (this.multiple) {
this.checkedIds = this.$refs.tree.getCheckedKeys();
this.checkedData = this.$refs.tree.getCheckedNodes();
} else {
this.checkedIds = this.$refs.tree.getCurrentKey();
this.checkedData = this.$refs.tree.getCurrentNode();
}
this.$emit("popoverHide", this.checkedIds, this.checkedData);
},
handleNodeClick(data, node) {
if (!this.multiple) {
this.setSelectOption(node);
this.isShowSelect = !this.isShowSelect;
this.$emit("change", node);
}
},
handleCheckChange() {
var checkedKeys1 = this.$refs.tree.getCheckedKeys();
var checkedKeys = [];
checkedKeys1.map((ele, index) => {
if (ele != undefined) {
checkedKeys.push(ele);
}
});
this.options = checkedKeys.map((item) => {
var node = this.$refs.tree.getNode(item);
let tmpMap = {};
if (node.childNodes.length == 0) {
tmpMap.value = node.key;
tmpMap.label = node.label;
tmpMap.disabled = node.disabled;
tmpMap.childNodes = node.childNodes;
}
return tmpMap;
});
this.selectedData = this.options.map((item) => {
return item.value;
});
let arr_Data = this.selectedData.filter((item) => item);
this.$emit("change", arr_Data);
},
removeSelectedNodes(val) {
this.$refs.tree.setChecked(val, false);
var node = this.$refs.tree.getNode(val);
if (!this.checkStrictly && node.childNodes.length > 0) {
this.treeToList(node).map((item) => {
if (item.childNodes.length <= 0) {
this.$refs.tree.setChecked(item, false);
}
});
this.handleCheckChange();
}
let arr_Data = this.selectedData.filter((item) => item);
this.$emit("change", arr_Data);
},
treeToList(tree) {
var queen = [];
var out = [];
queen = queen.concat(tree);
while (queen.length) {
var first = queen.shift();
if (first.childNodes) {
queen = queen.concat(first.childNodes);
}
out.push(first);
}
return out;
},
removeSelectedNode() {
this.clearSelectedNode();
this.$emit("change", this.selectedData);
},
changeSelectedNodes(selectedData) {
if (this.multiple && selectedData.length <= 0) {
this.clearSelectedNodes();
}
console.log(this.selectedData);
this.$emit("change", this.selectedData);
},
},
watch: {
isShowSelect() {
this.$refs.select.blur();
},
checkedKeys(val) {
if (!val) return;
this.checkedKeys = val;
this.initCheckedData();
},
},
};
</script>
<style lang="scss" scoped>
.mask {
width: 100%;
height: 100%;
position: fixed;
top: 0;
left: 0;
opacity: 0;
z-index: 11;
}
.common-tree {
overflow: auto;
}
.tree-select {
z-index: 111;
}
</style>
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通