381 lines
12 KiB
Vue
381 lines
12 KiB
Vue
![]() |
<template>
|
|||
|
<vue3-tree-vue
|
|||
|
dir="ltr"
|
|||
|
:items="treeItems"
|
|||
|
:isCheckable="false"
|
|||
|
:hideGuideLines="false"
|
|||
|
@onCheck="onItemChecked"
|
|||
|
@dropValidator="onBeforeItemDropped"
|
|||
|
@onSelect="onClick"
|
|||
|
@onExpand="onItemExpanded"
|
|||
|
>
|
|||
|
<!-- Applying some simple styling to tree-items -->
|
|||
|
<template v-slot:item-prepend-icon="treeViewItem">
|
|||
|
<!-- :src="prependIcon(treeViewItem)" -->
|
|||
|
<!-- <img
|
|||
|
src="assets/common/img/vue3-treevue/folder.svg"
|
|||
|
alt="folder"
|
|||
|
height="20"
|
|||
|
width="20"
|
|||
|
/> -->
|
|||
|
<svg class="icon icon-folder-2">
|
|||
|
<use xlink:href="#icon-folder-2"></use>
|
|||
|
</svg>
|
|||
|
</template>
|
|||
|
</vue3-tree-vue>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
import "vue3-tree-vue/dist/style.css"; // remember to add this in your component or maint.[ts/js]
|
|||
|
import { mapState } from "pinia";
|
|||
|
import entityApi from "~/apis/entityApi";
|
|||
|
import { useCommonStore } from "~/stores/commonStore";
|
|||
|
|
|||
|
export default {
|
|||
|
props: {
|
|||
|
treeItems: {
|
|||
|
default() {
|
|||
|
return [];
|
|||
|
},
|
|||
|
},
|
|||
|
},
|
|||
|
watch: {
|
|||
|
treeItems(newVal) {
|
|||
|
this.dicTreePath = {};
|
|||
|
this.rootTreeItems.children = newVal;
|
|||
|
// this.rootTreeItems.children.forEach((element) => {
|
|||
|
// this.dicTreePath[element.keyId] = "" + element.index;
|
|||
|
// });
|
|||
|
},
|
|||
|
},
|
|||
|
|
|||
|
data() {
|
|||
|
return {
|
|||
|
dicTreePath: {},
|
|||
|
rootTreeItems: [
|
|||
|
{
|
|||
|
name: "Rejection Emails",
|
|||
|
type: "emails",
|
|||
|
expanded: true,
|
|||
|
children: [
|
|||
|
{
|
|||
|
name: "List of Rejection Emails",
|
|||
|
type: "excel",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Handling Rejection Through Open-sourcing User Guide",
|
|||
|
type: "doc",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Leet Code Tracking Sheet",
|
|||
|
type: "folder",
|
|||
|
expanded: true,
|
|||
|
children: [
|
|||
|
{
|
|||
|
name: "Solved Problems (30)",
|
|||
|
type: "excel",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Unsolved Problems (5000)",
|
|||
|
type: "folder",
|
|||
|
id: 6,
|
|||
|
expanded: true,
|
|||
|
children: [
|
|||
|
{
|
|||
|
name: "Inverting a linked list (lol)",
|
|||
|
id: 7,
|
|||
|
type: "playlist",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Decoding hieroglyphs",
|
|||
|
id: 8,
|
|||
|
type: "playlist",
|
|||
|
children: [
|
|||
|
{
|
|||
|
name: "List of Rejection Emails",
|
|||
|
id: 24334,
|
|||
|
type: "excel",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Handling Rejection Through Open-sourcing User Guide",
|
|||
|
id: 354545,
|
|||
|
type: "doc",
|
|||
|
},
|
|||
|
],
|
|||
|
},
|
|||
|
],
|
|||
|
},
|
|||
|
],
|
|||
|
},
|
|||
|
],
|
|||
|
},
|
|||
|
{
|
|||
|
name: "System Design Playlist",
|
|||
|
id: 993029302938,
|
|||
|
type: "playlist",
|
|||
|
children: [
|
|||
|
{
|
|||
|
name: "Space Station Design",
|
|||
|
id: 10,
|
|||
|
type: "playlist",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Gabbage Collector Design",
|
|||
|
id: 11,
|
|||
|
type: "playlist",
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Design Extra-Terrestrial Life Support",
|
|||
|
type: "playlist",
|
|||
|
id: 12,
|
|||
|
},
|
|||
|
],
|
|||
|
},
|
|||
|
{
|
|||
|
name: "Lazyloaded",
|
|||
|
id: 389813,
|
|||
|
type: "playlist",
|
|||
|
disabled: true,
|
|||
|
},
|
|||
|
],
|
|||
|
dicTreePath: {},
|
|||
|
};
|
|||
|
},
|
|||
|
computed: {
|
|||
|
...mapState(useCommonStore, ["paginationGetter"]),
|
|||
|
},
|
|||
|
|
|||
|
methods: {
|
|||
|
// new tree methods.
|
|||
|
onItemChecked(checkedItems) {
|
|||
|
console.log(checkedItems);
|
|||
|
},
|
|||
|
onItemSelected(item) {
|
|||
|
console.log(item);
|
|||
|
},
|
|||
|
onBeforeItemDropped(droppedItem, dropHost) {
|
|||
|
// dropHost == undefined means dropping at the root of the tree.
|
|||
|
|
|||
|
// Here you can specify any kind of drop validation you will like.
|
|||
|
// this function should return true if the drop operation is valid.
|
|||
|
|
|||
|
if (dropHost.type !== playlist) return false;
|
|||
|
return true;
|
|||
|
},
|
|||
|
onItemExpanded(expandedItem) {
|
|||
|
console.log(expandedItem);
|
|||
|
|
|||
|
//to use this feature properly you need to set lazyLoad property as true
|
|||
|
//fetch data
|
|||
|
// const lazyLoadedItems = fetchFromApi(...);
|
|||
|
// expandedItem.children.push(...lazyLoadedItems)
|
|||
|
},
|
|||
|
|
|||
|
// old tree methods.
|
|||
|
/**
|
|||
|
* حذف یک نود.
|
|||
|
* این متد یک نود را حذف میکند.
|
|||
|
*
|
|||
|
* @param {HTMLElement} node - نودی که قرار است حذف شود
|
|||
|
*/
|
|||
|
onDel(node) {
|
|||
|
node.remove();
|
|||
|
},
|
|||
|
|
|||
|
/**
|
|||
|
* تغییر نام.
|
|||
|
* این متد نام یک مورد را تغییر میدهد.
|
|||
|
*
|
|||
|
* @param {object} params - پارامترهای مورد نیاز
|
|||
|
*/
|
|||
|
onChangeName(params) {},
|
|||
|
|
|||
|
/**
|
|||
|
* افزودن یک نود جدید.
|
|||
|
*
|
|||
|
* @param {object} params - پارامترهای مورد نیاز
|
|||
|
*/
|
|||
|
onAddNode(params) {},
|
|||
|
|
|||
|
/**
|
|||
|
* رویداد کلیک.
|
|||
|
* این متد وظیفه اعمال تغییرات پس از کلیک را دارد.
|
|||
|
*
|
|||
|
* @param {object} params - پارامترهای مورد نیاز
|
|||
|
*/
|
|||
|
onClick(params) {
|
|||
|
if (this.classId !== "") {
|
|||
|
let id = document.getElementById(this.classId);
|
|||
|
if (id) id.classList.remove("selected");
|
|||
|
const element = document.getElementById(params.id);
|
|||
|
if (element) element.classList.add("selected");
|
|||
|
} else {
|
|||
|
const element = document.getElementById(params.id);
|
|||
|
if (element) element.classList.add("selected");
|
|||
|
}
|
|||
|
this.classId = structuredClone(params.id);
|
|||
|
this.$emit("on-click", params);
|
|||
|
},
|
|||
|
|
|||
|
/**
|
|||
|
* اضافه کردن یک نود جدید.
|
|||
|
*
|
|||
|
* @param {object} item - مورد جدید برای افزودن
|
|||
|
*/
|
|||
|
addNode(item) {
|
|||
|
var node = new TreeNode({ item: item, isLeaf: false });
|
|||
|
if (!this.rootTreeItems.children) this.rootTreeItems.children = [];
|
|||
|
this.rootTreeItems.addChildren(node);
|
|||
|
},
|
|||
|
|
|||
|
/**
|
|||
|
* بهروزرسانی فرزندان.
|
|||
|
* این متد فرزندان یک المان را بهروزرسانی میکند.
|
|||
|
*
|
|||
|
* @param {object} element - المانی که فرزندانش باید بهروز شود
|
|||
|
* @param {array} childs - فرزندان جدید
|
|||
|
*/
|
|||
|
updateChildren(element, childs) {
|
|||
|
var parent = this.rootTreeItems;
|
|||
|
if (element.parent != null) parent = element.parent;
|
|||
|
|
|||
|
const index = parent.children.findIndex((item) => item.id === element.id);
|
|||
|
if (index !== -1) {
|
|||
|
parent.children[index].children = childs;
|
|||
|
}
|
|||
|
},
|
|||
|
/**
|
|||
|
* دریافت درخت جدید.
|
|||
|
* این متد یک درخت جدید از دادههای فعلی دریافت میکند.
|
|||
|
*/
|
|||
|
getNewTree() {
|
|||
|
var vm = this;
|
|||
|
function _dfs(oldNode) {
|
|||
|
var newNode = {};
|
|||
|
|
|||
|
for (var k in oldNode) {
|
|||
|
if (k !== "children" && k !== "parent") {
|
|||
|
newNode[k] = oldNode[k];
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (oldNode.children && oldNode.children.length > 0) {
|
|||
|
newNode.children = [];
|
|||
|
for (var i = 0, len = oldNode.children.length; i < len; i++) {
|
|||
|
newNode.children.push(_dfs(oldNode.children[i]));
|
|||
|
}
|
|||
|
}
|
|||
|
return newNode;
|
|||
|
}
|
|||
|
|
|||
|
vm.newTree = _dfs(vm.rootTreeItems);
|
|||
|
},
|
|||
|
},
|
|||
|
// components: {
|
|||
|
// VJstree,
|
|||
|
// VJstree: () => import("vue-jstree"),
|
|||
|
// },
|
|||
|
};
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss">
|
|||
|
.icon-tree {
|
|||
|
color: #dbc056;
|
|||
|
font-size: 0.6rem;
|
|||
|
}
|
|||
|
.vtl-node-main {
|
|||
|
padding: 5px 1rem 5px 0 !important;
|
|||
|
|
|||
|
.vtl-node-content {
|
|||
|
// margin-right: 7px;
|
|||
|
}
|
|||
|
|
|||
|
&.vtl-drag-disabled {
|
|||
|
// background-color: #d0cfcf;
|
|||
|
background-color: inherit !important;
|
|||
|
}
|
|||
|
|
|||
|
&:hover {
|
|||
|
cursor: pointer;
|
|||
|
background-color: #f0f0f0 !important;
|
|||
|
}
|
|||
|
|
|||
|
.vtl-caret {
|
|||
|
// margin-left: 0;
|
|||
|
// margin-right: -1rem;
|
|||
|
// margin-top: 8px;
|
|||
|
// transform: rotateY(180deg);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.vtl-tree-margin {
|
|||
|
// margin-right: 2em;
|
|||
|
// margin-left: auto;
|
|||
|
}
|
|||
|
|
|||
|
.vtl {
|
|||
|
.vtl-drag-disabled {
|
|||
|
background-color: #d0cfcf;
|
|||
|
|
|||
|
&:hover {
|
|||
|
background-color: #d0cfcf;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.vtl-disabled {
|
|||
|
background-color: #d0cfcf;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.vtl {
|
|||
|
}
|
|||
|
.tree-icon {
|
|||
|
background-image: url("") !important;
|
|||
|
}
|
|||
|
</style>
|
|||
|
|
|||
|
<style lang="scss" scoped>
|
|||
|
.label {
|
|||
|
position: relative;
|
|||
|
background-color: #fafafa;
|
|||
|
// background-color: #fff;
|
|||
|
padding: 0.3em;
|
|||
|
border-radius: 0.5em;
|
|||
|
margin-left: 1.4em;
|
|||
|
margin-top: 0.5rem;
|
|||
|
border: 1px solid #9e00b1;
|
|||
|
span {
|
|||
|
position: absolute;
|
|||
|
top: -0.5em;
|
|||
|
left: -1.2em;
|
|||
|
display: -webkit-box;
|
|||
|
display: -ms-flexbox;
|
|||
|
display: flex;
|
|||
|
}
|
|||
|
}
|
|||
|
.icon {
|
|||
|
&:hover {
|
|||
|
cursor: pointer;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.muted {
|
|||
|
color: gray;
|
|||
|
font-size: 80%;
|
|||
|
}
|
|||
|
|
|||
|
.tree {
|
|||
|
text-align: right;
|
|||
|
}
|
|||
|
|
|||
|
.tree-list-container {
|
|||
|
.tree-list-content {
|
|||
|
width: 100%;
|
|||
|
height: calc(100vh - 20em);
|
|||
|
overflow-y: auto;
|
|||
|
}
|
|||
|
|
|||
|
// }
|
|||
|
}
|
|||
|
</style>
|