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-15 07:20:57 +00:00
|
|
|
|
v-model="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
|
|
|
|
/>
|
2025-07-09 08:12:02 +00:00
|
|
|
|
<div class="tree-height" :style="treeStyle">
|
2025-08-21 07:40:22 +00:00
|
|
|
|
<div v-if="showShared && hasPermission(EditionConst.IS_EE, 'OR')" class="border-b mb-4">
|
|
|
|
|
|
<div
|
|
|
|
|
|
@click="handleSharedNodeClick"
|
|
|
|
|
|
class="shared-button flex cursor"
|
|
|
|
|
|
:class="currentNodeKey === 'share' && 'active'"
|
|
|
|
|
|
>
|
|
|
|
|
|
<AppIcon
|
|
|
|
|
|
iconName="app-shared-active"
|
|
|
|
|
|
style="font-size: 18px"
|
|
|
|
|
|
class="color-primary"
|
|
|
|
|
|
></AppIcon>
|
2025-08-29 09:03:21 +00:00
|
|
|
|
<span class="ml-8">{{ shareTitle }}</span>
|
2025-08-21 07:40:22 +00:00
|
|
|
|
</div>
|
2025-07-09 08:12:02 +00:00
|
|
|
|
</div>
|
2025-08-21 07:40:22 +00:00
|
|
|
|
|
2025-06-25 10:57:06 +00:00
|
|
|
|
<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">
|
2025-08-04 10:07:47 +00:00
|
|
|
|
<AppIcon iconName="app-folder" style="font-size: 20px"></AppIcon>
|
2025-09-26 09:44:26 +00:00
|
|
|
|
<span class="ml-8 ellipsis tree-label" style="max-width: 110px" :title="node.label">{{
|
2025-06-25 10:57:06 +00:00
|
|
|
|
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">
|
2025-08-01 09:05:32 +00:00
|
|
|
|
<el-button text class="w-full" v-if="MoreFilledPermission(node)">
|
|
|
|
|
|
<AppIcon iconName="app-more"></AppIcon>
|
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)"
|
2025-07-08 13:03:38 +00:00
|
|
|
|
v-if="node.level !== 3 && permissionPrecise.folderCreate()"
|
2025-06-27 02:59:58 +00:00
|
|
|
|
>
|
2025-08-05 06:23:02 +00:00
|
|
|
|
<AppIcon iconName="app-add-folder" class="color-secondary"></AppIcon>
|
2025-06-25 10:57:06 +00:00
|
|
|
|
{{ $t('components.folder.addChildFolder') }}
|
|
|
|
|
|
</el-dropdown-item>
|
2025-07-09 08:12:02 +00:00
|
|
|
|
<el-dropdown-item
|
|
|
|
|
|
@click.stop="openEditFolder(data)"
|
2025-07-08 13:03:38 +00:00
|
|
|
|
v-if="permissionPrecise.folderEdit()"
|
|
|
|
|
|
>
|
2025-08-05 06:23:02 +00:00
|
|
|
|
<AppIcon iconName="app-edit" class="color-secondary"></AppIcon>
|
2025-06-25 10:57:06 +00:00
|
|
|
|
{{ $t('common.edit') }}
|
|
|
|
|
|
</el-dropdown-item>
|
|
|
|
|
|
<el-dropdown-item
|
|
|
|
|
|
divided
|
|
|
|
|
|
@click.stop="deleteFolder(data)"
|
|
|
|
|
|
:disabled="!data.parent_id"
|
2025-07-08 13:03:38 +00:00
|
|
|
|
v-if="permissionPrecise.folderDelete()"
|
2025-06-25 10:57:06 +00:00
|
|
|
|
>
|
2025-08-05 06:23:02 +00:00
|
|
|
|
<AppIcon iconName="app-delete" class="color-secondary"></AppIcon>
|
2025-06-25 10:57:06 +00:00
|
|
|
|
{{ $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>
|
2025-07-15 09:15:20 +00:00
|
|
|
|
import { computed, onUnmounted, 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'
|
2025-06-12 08:51:49 +00:00
|
|
|
|
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'
|
2025-07-08 13:03:38 +00:00
|
|
|
|
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: {
|
2025-06-12 08:51:49 +00:00
|
|
|
|
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
|
|
|
|
})
|
2025-07-08 13:03:38 +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'
|
2025-08-19 10:51:09 +00:00
|
|
|
|
} else {
|
|
|
|
|
|
return 'application'
|
2025-07-08 13:03:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const permissionPrecise = computed(() => {
|
|
|
|
|
|
return permissionMap[resourceType.value!]['workspace']
|
|
|
|
|
|
})
|
2025-06-24 10:44:36 +00:00
|
|
|
|
|
2025-08-01 09:05:32 +00:00
|
|
|
|
const MoreFilledPermission = (node: any) => {
|
2025-07-21 07:13:14 +00:00
|
|
|
|
return (
|
|
|
|
|
|
(node.level !== 3 && permissionPrecise.value.folderCreate()) ||
|
|
|
|
|
|
permissionPrecise.value.folderEdit() ||
|
|
|
|
|
|
permissionPrecise.value.folderDelete()
|
|
|
|
|
|
)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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[]
|
2025-06-12 08:51:49 +00:00
|
|
|
|
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)
|
|
|
|
|
|
}
|
2025-06-12 08:51:49 +00:00
|
|
|
|
|
|
|
|
|
|
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-12 08:51:49 +00:00
|
|
|
|
}
|
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-07-15 09:15:20 +00:00
|
|
|
|
|
|
|
|
|
|
function clearCurrentKey() {
|
|
|
|
|
|
treeRef.value?.setCurrentKey(undefined)
|
|
|
|
|
|
}
|
|
|
|
|
|
defineExpose({
|
|
|
|
|
|
clearCurrentKey,
|
|
|
|
|
|
})
|
|
|
|
|
|
onUnmounted(() => {
|
|
|
|
|
|
treeRef.value?.setCurrentKey(undefined)
|
|
|
|
|
|
})
|
2025-05-08 08:23:03 +00:00
|
|
|
|
</script>
|
2025-06-12 08:51:49 +00:00
|
|
|
|
<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-09-11 09:17:13 +00:00
|
|
|
|
border-radius: var();
|
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-09-11 09:17:13 +00:00
|
|
|
|
border-radius: var();
|
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-12 08:51:49 +00:00
|
|
|
|
}
|
2025-06-25 10:57:06 +00:00
|
|
|
|
.tree-height {
|
|
|
|
|
|
padding-top: 4px;
|
2025-07-09 08:12:02 +00:00
|
|
|
|
height: calc(100vh - 210px);
|
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;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-06-12 08:51:49 +00:00
|
|
|
|
</style>
|