UnisKB/ui/src/components/folder-tree/index.vue

282 lines
7.9 KiB
Vue
Raw Normal View History

2025-05-08 08:23:03 +00:00
<template>
2025-06-17 18:10:36 +00:00
<div class="folder-tree">
2025-05-12 10:19:28 +00:00
<el-input
2025-07-03 09:28:35 +00:00
v-model.trim="filterText"
2025-05-12 10:19:28 +00:00
:placeholder="$t('common.search')"
prefix-icon="Search"
clearable
2025-05-15 02:25:05 +00:00
class="p-8"
2025-05-12 10:19:28 +00:00
/>
<div
@click="handleSharedNodeClick"
2025-06-20 19:14:59 +00:00
v-if="showShared && hasPermission(EditionConst.IS_EE, 'OR')"
2025-06-25 10:57:06 +00:00
class="shared-button flex cursor"
:class="currentNodeKey === 'share' && 'active'"
>
2025-06-26 08:36:08 +00:00
<AppIcon iconName="app-shared-active" style="font-size: 18px" class="color-primary"></AppIcon>
2025-06-24 06:49:09 +00:00
<span class="ml-8 lighter">{{ shareTitle }}</span>
</div>
2025-06-25 10:57:06 +00:00
<div class="tree-height border-t" :style="treeStyle">
<el-scrollbar>
<el-tree
ref="treeRef"
:data="data"
:props="defaultProps"
@node-click="handleNodeClick"
:filter-node-method="filterNode"
:default-expanded-keys="[currentNodeKey]"
:current-node-key="currentNodeKey"
highlight-current
class="overflow-inherit_node__children"
node-key="id"
v-loading="loading"
2025-06-30 13:05:06 +00:00
v-bind="$attrs"
2025-06-25 10:57:06 +00:00
>
<template #default="{ node, data }">
<div class="flex-between w-full" @mouseenter.stop="handleMouseEnter(data)">
<div class="flex align-center">
<AppIcon iconName="app-folder" style="font-size: 16px"></AppIcon>
<span class="ml-8 ellipsis" style="max-width: 110px" :title="node.label">{{
node.label
}}</span>
</div>
<div
2025-06-27 02:59:58 +00:00
v-if="canOperation"
2025-06-25 10:57:06 +00:00
@click.stop
v-show="hoverNodeId === data.id"
@mouseenter.stop="handleMouseEnter(data)"
@mouseleave.stop="handleMouseleave"
class="mr-16"
>
<el-dropdown trigger="click" :teleported="false">
<el-button text class="w-full">
2025-07-03 08:29:14 +00:00
<el-icon><MoreFilled /></el-icon>
2025-06-25 10:57:06 +00:00
</el-button>
<template #dropdown>
<el-dropdown-menu>
2025-06-27 02:59:58 +00:00
<el-dropdown-item
@click.stop="openCreateFolder(data)"
v-if="node.level !== 3 && permissionPrecise.folderCreate()"
2025-06-27 02:59:58 +00:00
>
2025-06-25 10:57:06 +00:00
<AppIcon iconName="app-add-folder"></AppIcon>
{{ $t('components.folder.addChildFolder') }}
</el-dropdown-item>
<el-dropdown-item @click.stop="openEditFolder(data)"
v-if="permissionPrecise.folderEdit()"
>
2025-06-25 10:57:06 +00:00
<el-icon><EditPen /></el-icon>
{{ $t('common.edit') }}
</el-dropdown-item>
<el-dropdown-item
divided
@click.stop="deleteFolder(data)"
:disabled="!data.parent_id"
v-if="permissionPrecise.folderDelete()"
2025-06-25 10:57:06 +00:00
>
<el-icon><Delete /></el-icon>
{{ $t('common.delete') }}
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</div>
</div>
</template>
</el-tree>
</el-scrollbar>
</div>
2025-06-17 18:10:36 +00:00
<CreateFolderDialog ref="CreateFolderDialogRef" @refresh="refreshFolder" :title="title" />
2025-05-12 10:19:28 +00:00
</div>
2025-05-08 08:23:03 +00:00
</template>
<script lang="ts" setup>
import { computed, ref, watch } from 'vue'
2025-06-24 10:44:36 +00:00
import { onBeforeRouteLeave } from 'vue-router'
2025-05-12 10:19:28 +00:00
import type { TreeInstance } from 'element-plus'
2025-06-17 18:10:36 +00:00
import CreateFolderDialog from '@/components/folder-tree/CreateFolderDialog.vue'
import { t } from '@/locales'
2025-06-17 18:10:36 +00:00
import folderApi from '@/api/folder'
2025-06-20 19:14:59 +00:00
import { EditionConst } from '@/utils/permission/data'
import { hasPermission } from '@/utils/permission/index'
2025-06-24 10:44:36 +00:00
import useStore from '@/stores'
2025-07-03 08:29:14 +00:00
import { TreeToFlatten } from '@/utils/array'
import { MsgConfirm } from '@/utils/message'
import permissionMap from '@/permission'
2025-06-24 10:44:36 +00:00
2025-05-08 08:23:03 +00:00
defineOptions({ name: 'FolderTree' })
2025-05-09 06:42:51 +00:00
const props = defineProps({
data: {
type: Array,
default: () => [],
},
2025-05-12 10:19:28 +00:00
currentNodeKey: {
type: String,
2025-07-03 08:29:14 +00:00
default: 'default',
2025-05-12 10:19:28 +00:00
},
2025-06-17 18:10:36 +00:00
source: {
type: String,
default: 'APPLICATION',
},
2025-06-20 19:14:59 +00:00
showShared: {
type: Boolean,
default: false,
},
2025-06-16 09:07:00 +00:00
shareTitle: {
type: String,
2025-06-24 08:53:58 +00:00
default: '',
2025-06-16 09:07:00 +00:00
},
2025-06-19 09:10:39 +00:00
canOperation: {
type: Boolean,
default: true,
},
2025-06-25 10:57:06 +00:00
treeStyle: {
type: Object,
default: () => ({}),
},
2025-05-09 06:42:51 +00:00
})
const resourceType = computed(() => {
if (props.source === 'APPLICATION') {
return 'application'
} else if (props.source === 'KNOWLEDGE') {
return 'knowledge'
} else if (props.source === 'MODEL') {
return 'model'
} else if (props.source === 'TOOL') {
return 'tool'
}
})
const permissionPrecise = computed(() => {
return permissionMap[resourceType.value!]['workspace']
})
2025-06-24 10:44:36 +00:00
const { folder } = useStore()
onBeforeRouteLeave((to, from) => {
folder.setCurrentFolder({})
})
2025-05-08 08:23:03 +00:00
interface Tree {
2025-05-09 06:42:51 +00:00
name: string
2025-05-08 08:23:03 +00:00
children?: Tree[]
id?: string
2025-06-17 18:10:36 +00:00
show?: boolean
2025-07-03 08:29:14 +00:00
parent_id?: string
2025-05-08 08:23:03 +00:00
}
const defaultProps = {
children: 'children',
2025-05-09 06:42:51 +00:00
label: 'name',
2025-05-08 08:23:03 +00:00
}
2025-05-12 10:19:28 +00:00
2025-06-17 18:10:36 +00:00
const emit = defineEmits(['handleNodeClick', 'refreshTree'])
2025-05-12 10:19:28 +00:00
const treeRef = ref<TreeInstance>()
const filterText = ref('')
2025-06-17 18:10:36 +00:00
const hoverNodeId = ref<string | undefined>('')
const title = ref('')
2025-06-19 09:10:39 +00:00
const loading = ref(false)
2025-05-12 10:19:28 +00:00
watch(filterText, (val) => {
treeRef.value!.filter(val)
})
2025-07-03 09:28:35 +00:00
const filterNode = (value: string, data: Tree) => {
if (!value) return true
return data.name.toLowerCase().includes(value.toLowerCase())
}
2025-06-19 09:10:39 +00:00
let time: any
2025-05-12 10:19:28 +00:00
2025-06-17 18:10:36 +00:00
function handleMouseEnter(data: Tree) {
2025-06-18 09:22:13 +00:00
clearTimeout(time)
2025-06-17 18:10:36 +00:00
hoverNodeId.value = data.id
}
function handleMouseleave() {
2025-06-18 09:22:13 +00:00
time = setTimeout(() => {
clearTimeout(time)
document.body.click()
}, 300)
2025-06-17 18:10:36 +00:00
}
2025-05-12 10:19:28 +00:00
const handleNodeClick = (data: Tree) => {
emit('handleNodeClick', data)
}
const handleSharedNodeClick = () => {
2025-06-13 09:17:01 +00:00
treeRef.value?.setCurrentKey(undefined)
2025-06-24 06:49:09 +00:00
emit('handleNodeClick', { id: 'share', name: props.shareTitle })
}
2025-06-17 18:10:36 +00:00
function deleteFolder(row: Tree) {
2025-07-03 08:29:14 +00:00
MsgConfirm(
`${t('common.deleteConfirm')}${row.name}`,
t('components.folder.deleteConfirmMessage'),
{
confirmButtonText: t('common.delete'),
confirmButtonClass: 'danger',
},
)
.then(() => {
folderApi.delFolder(row.id as string, props.source, loading).then(() => {
treeRef.value?.setCurrentKey(row.parent_id || 'default')
const prevFolder = TreeToFlatten(props.data).find((item: any) => item.id === row.parent_id)
folder.setCurrentFolder(prevFolder)
emit('refreshTree')
})
})
.catch(() => {})
2025-06-17 18:10:36 +00:00
}
const CreateFolderDialogRef = ref()
function openCreateFolder(row: Tree) {
2025-06-25 10:57:06 +00:00
title.value = t('components.folder.addChildFolder')
2025-06-17 18:10:36 +00:00
CreateFolderDialogRef.value.open(props.source, row.id)
}
function openEditFolder(row: Tree) {
2025-06-25 10:57:06 +00:00
title.value = t('components.folder.editFolder')
2025-06-17 18:10:36 +00:00
CreateFolderDialogRef.value.open(props.source, row.id, row)
}
function refreshFolder() {
emit('refreshTree')
}
2025-05-08 08:23:03 +00:00
</script>
<style lang="scss" scoped>
2025-06-25 10:57:06 +00:00
.folder-tree {
.shared-button {
padding: 10px 8px;
font-weight: 400;
font-size: 14px;
margin-bottom: 4px;
&.active {
background: var(--el-color-primary-light-9);
2025-06-26 08:36:08 +00:00
border-radius: var(--app-border-radius-base);
2025-06-25 10:57:06 +00:00
color: var(--el-color-primary);
font-weight: 500;
&:hover {
background: var(--el-color-primary-light-9);
}
}
&:hover {
2025-06-26 08:36:08 +00:00
border-radius: var(--app-border-radius-base);
2025-06-25 10:57:06 +00:00
background: var(--app-text-color-light-1);
}
&.is-active {
&:hover {
color: var(--el-color-primary);
background: var(--el-color-primary-light-9);
}
}
}
2025-06-25 10:57:06 +00:00
.tree-height {
padding-top: 4px;
2025-07-07 06:23:36 +00:00
height: calc(100vh - 175px);
2025-06-19 03:20:41 +00:00
}
}
2025-06-25 11:58:48 +00:00
:deep(.overflow-inherit_node__children) {
.el-tree-node__children {
overflow: inherit !important;
}
}
</style>