feat: implement persona folder for advanced persona management (#4443)

* feat(db): add persona folder management for hierarchical organization

Implement hierarchical folder structure for organizing personas:
- Add PersonaFolder model with recursive parent-child relationships
- Add folder_id and sort_order fields to Persona model
- Implement CRUD operations for persona folders in database layer
- Add migration support for existing databases
- Extend PersonaManager with folder management methods
- Add dashboard API routes for folder operations

* feat(persona): add batch sort order update endpoint for personas and folders

Add new API endpoint POST /persona/reorder to batch update sort_order
for both personas and folders. This enables drag-and-drop reordering
in the dashboard UI.

Changes:
- Add abstract batch_update_sort_order method to BaseDatabase
- Implement batch_update_sort_order in SQLiteDatabase
- Add batch_update_sort_order to PersonaManager with cache refresh
- Add reorder_items route handler with input validation

* feat(persona): add folder_id and sort_order params to persona creation

Extend persona creation flow to support folder placement and ordering:
- Add folder_id and sort_order parameters to insert_persona in db layer
- Update PersonaManager.create_persona to accept and pass folder params
- Add get_folder_detail API endpoint for retrieving folder information
- Include folder_id and sort_order in persona creation response
- Add session flush/refresh to return complete persona object

* feat(dashboard): implement persona folder management UI

- Add folder management system with tree view and breadcrumbs
- Implement create, rename, delete, and move operations for folders
- Add drag-and-drop support for organizing personas and folders
- Create new PersonaManager component and Pinia store for state management
- Refactor PersonaPage to support hierarchical structure
- Update locale files with folder-related translations
- Handle empty parent_id correctly in backend route

* feat(dashboard): centralize folder expansion state in persona store

Move folder expansion logic from local component state to global Pinia
store to persist expansion state.
- Add `expandedFolderIds` state and toggle actions to `personaStore`
- Update `FolderTreeNode` to use store state instead of local data
- Automatically navigate to target folder after moving a persona

* feat(dashboard): add reusable folder management component library

Extract folder management UI into reusable base components and create
persona-specific wrapper components that integrate with personaStore.

- Add base folder components (tree, breadcrumb, card, dialogs) with
  customizable labels for i18n support
- Create useFolderManager composable for folder state management
- Implement drag-and-drop support for moving personas between folders
- Add persona-specific wrapper components connecting to personaStore
- Reorganize PersonaManager into views/persona directory structure
- Include comprehensive README documentation for component usage

* refactor(dashboard): remove legacy persona folder management components

Remove deprecated persona folder management Vue components that have been
superseded by the new reusable folder management component library.

Deleted components:
- CreateFolderDialog.vue
- FolderBreadcrumb.vue
- FolderCard.vue
- FolderTree.vue
- FolderTreeNode.vue
- MoveTargetNode.vue
- MoveToFolderDialog.vue
- PersonaCard.vue
- PersonaManager.vue

These components are replaced by the centralized folder management
implementation introduced in commit 3fbb3db2.

* fix(dashboard): add delayed skeleton loading to prevent UI flicker

Implement a 150ms delay before showing the skeleton loader in
PersonaManager to prevent visual flicker during fast loading operations.

- Add showSkeleton state with timer-based delay mechanism
- Use v-fade-transition for smooth skeleton visibility transitions
- Clean up timer on component unmount to prevent memory leaks
- Only display skeleton when loading exceeds threshold duration

* feat(dashboard): add generic folder item selector component for persona selection

Introduce BaseFolderItemSelector.vue as a reusable component for selecting
items within folder hierarchies. Refactor PersonaSelector to use this new
base component instead of its previous flat list implementation.

Changes:
- Add BaseFolderItemSelector with folder tree navigation and item selection
- Extend folder types with SelectableItem and FolderItemSelectorLabels
- Refactor PersonaSelector to leverage the new base component
- Add i18n translations for rootFolder and emptyFolder labels

* feat(persona): add tree-view display for persona list command

Add hierarchical folder tree output for the persona list command,
showing personas organized by folders with visual tree connectors.

- Add _build_tree_output method for recursive tree structure rendering
- Display folders with 📁 icon and personas with 👤 icon
- Show root-level personas separately from folder contents
- Include total persona count in output

* refactor(persona): simplify tree-view output with shorter indentation lines

Replace complex tree connector logic with simpler depth-based indentation
using "│ " prefix. Remove unnecessary parameters (prefix, is_last) and
computed variables (has_content, total_items, item_idx) in favor of a
cleaner depth-based approach.

* feat(dashboard): add duplicate persona ID validation in create form

Add frontend validation to prevent creating personas with duplicate IDs.
Load existing persona IDs when opening the create form and validate
against them in real-time.

- Add existingPersonaIds array and loadExistingPersonaIds method
- Add validation rule to check for duplicate persona IDs
- Add i18n messages for duplicate ID error (en-US and zh-CN)
- Fix minLength validation to require at least 1 character

* i18n(persona): add createButton translation key for folder dialog

Move create button label to folder-specific translation path
instead of using generic buttons.create key.

* feat(persona): show target folder name in persona creation dialog

Add visual feedback showing which folder a new persona will be created in.

- Add info alert in PersonaForm displaying the target folder name
- Pass currentFolderName prop from PersonaManager and PersonaSelector
- Add recursive findFolderName helper to resolve folder ID to name
- Add i18n translations for createInFolder and rootFolder labels

* style:format code

* fix: remove 'persistent' attribute from dialog components

---------

Co-authored-by: Soulter <905617992@qq.com>
This commit is contained in:
Ruochen Pan
2026-01-21 13:05:33 +08:00
committed by GitHub
parent c09bbfb8ac
commit 8910ab3a47
36 changed files with 5722 additions and 436 deletions
@@ -1,13 +1,55 @@
import builtins
from typing import TYPE_CHECKING
from astrbot.api import sp, star
from astrbot.api.event import AstrMessageEvent, MessageEventResult
if TYPE_CHECKING:
from astrbot.core.db.po import Persona
class PersonaCommands:
def __init__(self, context: star.Context):
self.context = context
def _build_tree_output(
self,
folder_tree: list[dict],
all_personas: list["Persona"],
depth: int = 0,
) -> list[str]:
"""递归构建树状输出,使用短线条表示层级"""
lines: list[str] = []
# 使用短线条作为缩进前缀,每层只用 "│" 加一个空格
prefix = "" * depth
for folder in folder_tree:
# 输出文件夹
lines.append(f"{prefix}├ 📁 {folder['name']}/")
# 获取该文件夹下的人格
folder_personas = [
p for p in all_personas if p.folder_id == folder["folder_id"]
]
child_prefix = "" * (depth + 1)
# 输出该文件夹下的人格
for persona in folder_personas:
lines.append(f"{child_prefix}├ 👤 {persona.persona_id}")
# 递归处理子文件夹
children = folder.get("children", [])
if children:
lines.extend(
self._build_tree_output(
children,
all_personas,
depth + 1,
)
)
return lines
async def persona(self, message: AstrMessageEvent):
l = message.message_str.split(" ") # noqa: E741
umo = message.unified_msg_origin
@@ -69,12 +111,32 @@ class PersonaCommands:
.use_t2i(False),
)
elif l[1] == "list":
parts = ["人格列表:\n"]
for persona in self.context.provider_manager.personas:
parts.append(f"- {persona['name']}\n")
parts.append("\n\n*输入 `/persona view 人格名` 查看人格详细信息")
msg = "".join(parts)
message.set_result(MessageEventResult().message(msg))
# 获取文件夹树和所有人格
folder_tree = await self.context.persona_manager.get_folder_tree()
all_personas = self.context.persona_manager.personas
lines = ["📂 人格列表:\n"]
# 构建树状输出
tree_lines = self._build_tree_output(folder_tree, all_personas)
lines.extend(tree_lines)
# 输出根目录下的人格(没有文件夹的)
root_personas = [p for p in all_personas if p.folder_id is None]
if root_personas:
if tree_lines: # 如果有文件夹内容,加个空行
lines.append("")
for persona in root_personas:
lines.append(f"👤 {persona.persona_id}")
# 统计信息
total_count = len(all_personas)
lines.append(f"\n{total_count} 个人格")
lines.append("\n*使用 `/persona <人格名>` 设置人格")
lines.append("*使用 `/persona view <人格名>` 查看详细信息")
msg = "\n".join(lines)
message.set_result(MessageEventResult().message(msg).use_t2i(False))
elif l[1] == "view":
if len(l) == 2:
message.set_result(MessageEventResult().message("请输入人格情景名"))
+91 -1
View File
@@ -14,6 +14,7 @@ from astrbot.core.db.po import (
CommandConflict,
ConversationV2,
Persona,
PersonaFolder,
PlatformMessageHistory,
PlatformSession,
PlatformStat,
@@ -253,8 +254,19 @@ class BaseDatabase(abc.ABC):
system_prompt: str,
begin_dialogs: list[str] | None = None,
tools: list[str] | None = None,
folder_id: str | None = None,
sort_order: int = 0,
) -> Persona:
"""Insert a new persona record."""
"""Insert a new persona record.
Args:
persona_id: Unique identifier for the persona
system_prompt: System prompt for the persona
begin_dialogs: Optional list of initial dialog strings
tools: Optional list of tool names (None means all tools, [] means no tools)
folder_id: Optional folder ID to place the persona in (None means root)
sort_order: Sort order within the folder (default 0)
"""
...
@abc.abstractmethod
@@ -283,6 +295,84 @@ class BaseDatabase(abc.ABC):
"""Delete a persona by its ID."""
...
# ====
# Persona Folder Management
# ====
@abc.abstractmethod
async def insert_persona_folder(
self,
name: str,
parent_id: str | None = None,
description: str | None = None,
sort_order: int = 0,
) -> PersonaFolder:
"""Insert a new persona folder."""
...
@abc.abstractmethod
async def get_persona_folder_by_id(self, folder_id: str) -> PersonaFolder | None:
"""Get a persona folder by its folder_id."""
...
@abc.abstractmethod
async def get_persona_folders(
self, parent_id: str | None = None
) -> list[PersonaFolder]:
"""Get all persona folders, optionally filtered by parent_id."""
...
@abc.abstractmethod
async def get_all_persona_folders(self) -> list[PersonaFolder]:
"""Get all persona folders."""
...
@abc.abstractmethod
async def update_persona_folder(
self,
folder_id: str,
name: str | None = None,
parent_id: T.Any = None,
description: T.Any = None,
sort_order: int | None = None,
) -> PersonaFolder | None:
"""Update a persona folder."""
...
@abc.abstractmethod
async def delete_persona_folder(self, folder_id: str) -> None:
"""Delete a persona folder by its folder_id."""
...
@abc.abstractmethod
async def move_persona_to_folder(
self, persona_id: str, folder_id: str | None
) -> Persona | None:
"""Move a persona to a folder (or root if folder_id is None)."""
...
@abc.abstractmethod
async def get_personas_by_folder(
self, folder_id: str | None = None
) -> list[Persona]:
"""Get all personas in a specific folder."""
...
@abc.abstractmethod
async def batch_update_sort_order(
self,
items: list[dict],
) -> None:
"""Batch update sort_order for personas and/or folders.
Args:
items: List of dicts with keys:
- id: The persona_id or folder_id
- type: Either "persona" or "folder"
- sort_order: The new sort_order value
"""
...
@abc.abstractmethod
async def insert_preference_or_update(
self,
+42
View File
@@ -68,6 +68,44 @@ class ConversationV2(SQLModel, table=True):
)
class PersonaFolder(SQLModel, table=True):
"""Persona 文件夹,支持递归层级结构。
用于组织和管理多个 Persona,类似于文件系统的目录结构。
"""
__tablename__: str = "persona_folders"
id: int | None = Field(
primary_key=True,
sa_column_kwargs={"autoincrement": True},
default=None,
)
folder_id: str = Field(
max_length=36,
nullable=False,
unique=True,
default_factory=lambda: str(uuid.uuid4()),
)
name: str = Field(max_length=255, nullable=False)
parent_id: str | None = Field(default=None, max_length=36)
"""父文件夹IDNULL表示根目录"""
description: str | None = Field(default=None, sa_type=Text)
sort_order: int = Field(default=0)
created_at: datetime = Field(default_factory=lambda: datetime.now(timezone.utc))
updated_at: datetime = Field(
default_factory=lambda: datetime.now(timezone.utc),
sa_column_kwargs={"onupdate": datetime.now(timezone.utc)},
)
__table_args__ = (
UniqueConstraint(
"folder_id",
name="uix_persona_folder_id",
),
)
class Persona(SQLModel, table=True):
"""Persona is a set of instructions for LLMs to follow.
@@ -87,6 +125,10 @@ class Persona(SQLModel, table=True):
"""a list of strings, each representing a dialog to start with"""
tools: list | None = Field(default=None, sa_type=JSON)
"""None means use ALL tools for default, empty list means no tools, otherwise a list of tool names."""
folder_id: str | None = Field(default=None, max_length=36)
"""所属文件夹IDNULL 表示在根目录"""
sort_order: int = Field(default=0)
"""排序顺序"""
created_at: datetime = Field(default_factory=lambda: datetime.now(timezone.utc))
updated_at: datetime = Field(
default_factory=lambda: datetime.now(timezone.utc),
+230
View File
@@ -16,6 +16,7 @@ from astrbot.core.db.po import (
CommandConflict,
ConversationV2,
Persona,
PersonaFolder,
PlatformMessageHistory,
PlatformSession,
PlatformStat,
@@ -51,8 +52,30 @@ class SQLiteDatabase(BaseDatabase):
await conn.execute(text("PRAGMA temp_store=MEMORY"))
await conn.execute(text("PRAGMA mmap_size=134217728"))
await conn.execute(text("PRAGMA optimize"))
# 确保 personas 表有 folder_id 和 sort_order 列(前向兼容)
await self._ensure_persona_folder_columns(conn)
await conn.commit()
async def _ensure_persona_folder_columns(self, conn) -> None:
"""确保 personas 表有 folder_id 和 sort_order 列。
这是为了支持旧版数据库的平滑升级。新版数据库通过 SQLModel
的 metadata.create_all 自动创建这些列。
"""
result = await conn.execute(text("PRAGMA table_info(personas)"))
columns = {row[1] for row in result.fetchall()}
if "folder_id" not in columns:
await conn.execute(
text(
"ALTER TABLE personas ADD COLUMN folder_id VARCHAR(36) DEFAULT NULL"
)
)
if "sort_order" not in columns:
await conn.execute(
text("ALTER TABLE personas ADD COLUMN sort_order INTEGER DEFAULT 0")
)
# ====
# Platform Statistics
# ====
@@ -541,6 +564,8 @@ class SQLiteDatabase(BaseDatabase):
system_prompt,
begin_dialogs=None,
tools=None,
folder_id=None,
sort_order=0,
):
"""Insert a new persona record."""
async with self.get_db() as session:
@@ -551,8 +576,12 @@ class SQLiteDatabase(BaseDatabase):
system_prompt=system_prompt,
begin_dialogs=begin_dialogs or [],
tools=tools,
folder_id=folder_id,
sort_order=sort_order,
)
session.add(new_persona)
await session.flush()
await session.refresh(new_persona)
return new_persona
async def get_persona_by_id(self, persona_id):
@@ -605,6 +634,207 @@ class SQLiteDatabase(BaseDatabase):
delete(Persona).where(col(Persona.persona_id) == persona_id),
)
# ====
# Persona Folder Management
# ====
async def insert_persona_folder(
self,
name: str,
parent_id: str | None = None,
description: str | None = None,
sort_order: int = 0,
) -> PersonaFolder:
"""Insert a new persona folder."""
async with self.get_db() as session:
session: AsyncSession
async with session.begin():
new_folder = PersonaFolder(
name=name,
parent_id=parent_id,
description=description,
sort_order=sort_order,
)
session.add(new_folder)
await session.flush()
await session.refresh(new_folder)
return new_folder
async def get_persona_folder_by_id(self, folder_id: str) -> PersonaFolder | None:
"""Get a persona folder by its folder_id."""
async with self.get_db() as session:
session: AsyncSession
query = select(PersonaFolder).where(PersonaFolder.folder_id == folder_id)
result = await session.execute(query)
return result.scalar_one_or_none()
async def get_persona_folders(
self, parent_id: str | None = None
) -> list[PersonaFolder]:
"""Get all persona folders, optionally filtered by parent_id.
Args:
parent_id: If None, returns root folders only. If specified, returns
children of that folder.
"""
async with self.get_db() as session:
session: AsyncSession
if parent_id is None:
# Get root folders (parent_id is NULL)
query = (
select(PersonaFolder)
.where(col(PersonaFolder.parent_id).is_(None))
.order_by(col(PersonaFolder.sort_order), col(PersonaFolder.name))
)
else:
query = (
select(PersonaFolder)
.where(PersonaFolder.parent_id == parent_id)
.order_by(col(PersonaFolder.sort_order), col(PersonaFolder.name))
)
result = await session.execute(query)
return list(result.scalars().all())
async def get_all_persona_folders(self) -> list[PersonaFolder]:
"""Get all persona folders."""
async with self.get_db() as session:
session: AsyncSession
query = select(PersonaFolder).order_by(
col(PersonaFolder.sort_order), col(PersonaFolder.name)
)
result = await session.execute(query)
return list(result.scalars().all())
async def update_persona_folder(
self,
folder_id: str,
name: str | None = None,
parent_id: T.Any = NOT_GIVEN,
description: T.Any = NOT_GIVEN,
sort_order: int | None = None,
) -> PersonaFolder | None:
"""Update a persona folder."""
async with self.get_db() as session:
session: AsyncSession
async with session.begin():
query = update(PersonaFolder).where(
col(PersonaFolder.folder_id) == folder_id
)
values: dict[str, T.Any] = {}
if name is not None:
values["name"] = name
if parent_id is not NOT_GIVEN:
values["parent_id"] = parent_id
if description is not NOT_GIVEN:
values["description"] = description
if sort_order is not None:
values["sort_order"] = sort_order
if not values:
return None
query = query.values(**values)
await session.execute(query)
return await self.get_persona_folder_by_id(folder_id)
async def delete_persona_folder(self, folder_id: str) -> None:
"""Delete a persona folder by its folder_id.
Note: This will also set folder_id to NULL for all personas in this folder,
moving them to the root directory.
"""
async with self.get_db() as session:
session: AsyncSession
async with session.begin():
# Move personas to root directory
await session.execute(
update(Persona)
.where(col(Persona.folder_id) == folder_id)
.values(folder_id=None)
)
# Delete the folder
await session.execute(
delete(PersonaFolder).where(
col(PersonaFolder.folder_id) == folder_id
),
)
async def move_persona_to_folder(
self, persona_id: str, folder_id: str | None
) -> Persona | None:
"""Move a persona to a folder (or root if folder_id is None)."""
async with self.get_db() as session:
session: AsyncSession
async with session.begin():
await session.execute(
update(Persona)
.where(col(Persona.persona_id) == persona_id)
.values(folder_id=folder_id)
)
return await self.get_persona_by_id(persona_id)
async def get_personas_by_folder(
self, folder_id: str | None = None
) -> list[Persona]:
"""Get all personas in a specific folder.
Args:
folder_id: If None, returns personas in root directory.
"""
async with self.get_db() as session:
session: AsyncSession
if folder_id is None:
query = (
select(Persona)
.where(col(Persona.folder_id).is_(None))
.order_by(col(Persona.sort_order), col(Persona.persona_id))
)
else:
query = (
select(Persona)
.where(Persona.folder_id == folder_id)
.order_by(col(Persona.sort_order), col(Persona.persona_id))
)
result = await session.execute(query)
return list(result.scalars().all())
async def batch_update_sort_order(
self,
items: list[dict],
) -> None:
"""Batch update sort_order for personas and/or folders.
Args:
items: List of dicts with keys:
- id: The persona_id or folder_id
- type: Either "persona" or "folder"
- sort_order: The new sort_order value
"""
if not items:
return
async with self.get_db() as session:
session: AsyncSession
async with session.begin():
for item in items:
item_id = item.get("id")
item_type = item.get("type")
sort_order = item.get("sort_order")
if item_id is None or item_type is None or sort_order is None:
continue
if item_type == "persona":
await session.execute(
update(Persona)
.where(col(Persona.persona_id) == item_id)
.values(sort_order=sort_order)
)
elif item_type == "folder":
await session.execute(
update(PersonaFolder)
.where(col(PersonaFolder.folder_id) == item_id)
.values(sort_order=sort_order)
)
async def insert_preference_or_update(self, scope, scope_id, key, value):
"""Insert a new preference record or update if it exists."""
async with self.get_db() as session:
+154 -2
View File
@@ -1,7 +1,7 @@
from astrbot import logger
from astrbot.core.astrbot_config_mgr import AstrBotConfigManager
from astrbot.core.db import BaseDatabase
from astrbot.core.db.po import Persona, Personality
from astrbot.core.db.po import Persona, PersonaFolder, Personality
from astrbot.core.platform.message_session import MessageSession
DEFAULT_PERSONALITY = Personality(
@@ -94,14 +94,164 @@ class PersonaManager:
"""获取所有 personas"""
return await self.db.get_personas()
async def get_personas_by_folder(
self, folder_id: str | None = None
) -> list[Persona]:
"""获取指定文件夹中的 personas
Args:
folder_id: 文件夹 IDNone 表示根目录
"""
return await self.db.get_personas_by_folder(folder_id)
async def move_persona_to_folder(
self, persona_id: str, folder_id: str | None
) -> Persona | None:
"""移动 persona 到指定文件夹
Args:
persona_id: Persona ID
folder_id: 目标文件夹 IDNone 表示移动到根目录
"""
persona = await self.db.move_persona_to_folder(persona_id, folder_id)
if persona:
for i, p in enumerate(self.personas):
if p.persona_id == persona_id:
self.personas[i] = persona
break
return persona
# ====
# Persona Folder Management
# ====
async def create_folder(
self,
name: str,
parent_id: str | None = None,
description: str | None = None,
sort_order: int = 0,
) -> PersonaFolder:
"""创建新的文件夹"""
return await self.db.insert_persona_folder(
name=name,
parent_id=parent_id,
description=description,
sort_order=sort_order,
)
async def get_folder(self, folder_id: str) -> PersonaFolder | None:
"""获取指定文件夹"""
return await self.db.get_persona_folder_by_id(folder_id)
async def get_folders(self, parent_id: str | None = None) -> list[PersonaFolder]:
"""获取文件夹列表
Args:
parent_id: 父文件夹 ID,None 表示获取根目录下的文件夹
"""
return await self.db.get_persona_folders(parent_id)
async def get_all_folders(self) -> list[PersonaFolder]:
"""获取所有文件夹"""
return await self.db.get_all_persona_folders()
async def update_folder(
self,
folder_id: str,
name: str | None = None,
parent_id: str | None = None,
description: str | None = None,
sort_order: int | None = None,
) -> PersonaFolder | None:
"""更新文件夹信息"""
return await self.db.update_persona_folder(
folder_id=folder_id,
name=name,
parent_id=parent_id,
description=description,
sort_order=sort_order,
)
async def delete_folder(self, folder_id: str) -> None:
"""删除文件夹
Note: 文件夹内的 personas 会被移动到根目录
"""
await self.db.delete_persona_folder(folder_id)
async def batch_update_sort_order(self, items: list[dict]) -> None:
"""批量更新 personas 和/或 folders 的排序顺序
Args:
items: 包含以下键的字典列表:
- id: persona_id 或 folder_id
- type: "persona""folder"
- sort_order: 新的排序顺序值
"""
await self.db.batch_update_sort_order(items)
# 刷新缓存
self.personas = await self.get_all_personas()
self.get_v3_persona_data()
async def get_folder_tree(self) -> list[dict]:
"""获取文件夹树形结构
Returns:
树形结构的文件夹列表,每个文件夹包含 children 子列表
"""
all_folders = await self.get_all_folders()
folder_map: dict[str, dict] = {}
# 创建文件夹字典
for folder in all_folders:
folder_map[folder.folder_id] = {
"folder_id": folder.folder_id,
"name": folder.name,
"parent_id": folder.parent_id,
"description": folder.description,
"sort_order": folder.sort_order,
"children": [],
}
# 构建树形结构
root_folders = []
for folder_id, folder_data in folder_map.items():
parent_id = folder_data["parent_id"]
if parent_id is None:
root_folders.append(folder_data)
elif parent_id in folder_map:
folder_map[parent_id]["children"].append(folder_data)
# 递归排序
def sort_folders(folders: list[dict]) -> list[dict]:
folders.sort(key=lambda f: (f["sort_order"], f["name"]))
for folder in folders:
if folder["children"]:
folder["children"] = sort_folders(folder["children"])
return folders
return sort_folders(root_folders)
async def create_persona(
self,
persona_id: str,
system_prompt: str,
begin_dialogs: list[str] | None = None,
tools: list[str] | None = None,
folder_id: str | None = None,
sort_order: int = 0,
) -> Persona:
"""创建新的 persona。tools 参数为 None 时表示使用所有工具,空列表表示不使用任何工具"""
"""创建新的 persona。
Args:
persona_id: Persona 唯一标识
system_prompt: 系统提示词
begin_dialogs: 预设对话列表
tools: 工具列表,None 表示使用所有工具,空列表表示不使用任何工具
folder_id: 所属文件夹 IDNone 表示根目录
sort_order: 排序顺序
"""
if await self.db.get_persona_by_id(persona_id):
raise ValueError(f"Persona with ID {persona_id} already exists.")
new_persona = await self.db.insert_persona(
@@ -109,6 +259,8 @@ class PersonaManager:
system_prompt,
begin_dialogs,
tools=tools,
folder_id=folder_id,
sort_order=sort_order,
)
self.personas.append(new_persona)
self.get_v3_persona_data()
+258 -1
View File
@@ -23,6 +23,15 @@ class PersonaRoute(Route):
"/persona/create": ("POST", self.create_persona),
"/persona/update": ("POST", self.update_persona),
"/persona/delete": ("POST", self.delete_persona),
"/persona/move": ("POST", self.move_persona),
"/persona/reorder": ("POST", self.reorder_items),
# Folder routes
"/persona/folder/list": ("GET", self.list_folders),
"/persona/folder/tree": ("GET", self.get_folder_tree),
"/persona/folder/detail": ("POST", self.get_folder_detail),
"/persona/folder/create": ("POST", self.create_folder),
"/persona/folder/update": ("POST", self.update_folder),
"/persona/folder/delete": ("POST", self.delete_folder),
}
self.db_helper = db_helper
self.persona_mgr = core_lifecycle.persona_mgr
@@ -31,7 +40,14 @@ class PersonaRoute(Route):
async def list_personas(self):
"""获取所有人格列表"""
try:
personas = await self.persona_mgr.get_all_personas()
# 支持按文件夹筛选
folder_id = request.args.get("folder_id")
if folder_id is not None:
personas = await self.persona_mgr.get_personas_by_folder(
folder_id if folder_id else None
)
else:
personas = await self.persona_mgr.get_all_personas()
return (
Response()
.ok(
@@ -41,6 +57,8 @@ class PersonaRoute(Route):
"system_prompt": persona.system_prompt,
"begin_dialogs": persona.begin_dialogs or [],
"tools": persona.tools,
"folder_id": persona.folder_id,
"sort_order": persona.sort_order,
"created_at": persona.created_at.isoformat()
if persona.created_at
else None,
@@ -78,6 +96,8 @@ class PersonaRoute(Route):
"system_prompt": persona.system_prompt,
"begin_dialogs": persona.begin_dialogs or [],
"tools": persona.tools,
"folder_id": persona.folder_id,
"sort_order": persona.sort_order,
"created_at": persona.created_at.isoformat()
if persona.created_at
else None,
@@ -100,6 +120,8 @@ class PersonaRoute(Route):
system_prompt = data.get("system_prompt", "").strip()
begin_dialogs = data.get("begin_dialogs", [])
tools = data.get("tools")
folder_id = data.get("folder_id") # None 表示根目录
sort_order = data.get("sort_order", 0)
if not persona_id:
return Response().error("人格ID不能为空").__dict__
@@ -120,6 +142,8 @@ class PersonaRoute(Route):
system_prompt=system_prompt,
begin_dialogs=begin_dialogs if begin_dialogs else None,
tools=tools if tools else None,
folder_id=folder_id,
sort_order=sort_order,
)
return (
@@ -132,6 +156,8 @@ class PersonaRoute(Route):
"system_prompt": persona.system_prompt,
"begin_dialogs": persona.begin_dialogs or [],
"tools": persona.tools or [],
"folder_id": persona.folder_id,
"sort_order": persona.sort_order,
"created_at": persona.created_at.isoformat()
if persona.created_at
else None,
@@ -200,3 +226,234 @@ class PersonaRoute(Route):
except Exception as e:
logger.error(f"删除人格失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"删除人格失败: {e!s}").__dict__
async def move_persona(self):
"""移动人格到指定文件夹"""
try:
data = await request.get_json()
persona_id = data.get("persona_id")
folder_id = data.get("folder_id") # None 表示移动到根目录
if not persona_id:
return Response().error("缺少必要参数: persona_id").__dict__
await self.persona_mgr.move_persona_to_folder(persona_id, folder_id)
return Response().ok({"message": "人格移动成功"}).__dict__
except ValueError as e:
return Response().error(str(e)).__dict__
except Exception as e:
logger.error(f"移动人格失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"移动人格失败: {e!s}").__dict__
# ====
# Folder Routes
# ====
async def list_folders(self):
"""获取文件夹列表"""
try:
parent_id = request.args.get("parent_id")
# 空字符串视为 None(根目录)
if parent_id == "":
parent_id = None
folders = await self.persona_mgr.get_folders(parent_id)
return (
Response()
.ok(
[
{
"folder_id": folder.folder_id,
"name": folder.name,
"parent_id": folder.parent_id,
"description": folder.description,
"sort_order": folder.sort_order,
"created_at": folder.created_at.isoformat()
if folder.created_at
else None,
"updated_at": folder.updated_at.isoformat()
if folder.updated_at
else None,
}
for folder in folders
],
)
.__dict__
)
except Exception as e:
logger.error(f"获取文件夹列表失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"获取文件夹列表失败: {e!s}").__dict__
async def get_folder_tree(self):
"""获取文件夹树形结构"""
try:
tree = await self.persona_mgr.get_folder_tree()
return Response().ok(tree).__dict__
except Exception as e:
logger.error(f"获取文件夹树失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"获取文件夹树失败: {e!s}").__dict__
async def get_folder_detail(self):
"""获取指定文件夹的详细信息"""
try:
data = await request.get_json()
folder_id = data.get("folder_id")
if not folder_id:
return Response().error("缺少必要参数: folder_id").__dict__
folder = await self.persona_mgr.get_folder(folder_id)
if not folder:
return Response().error("文件夹不存在").__dict__
return (
Response()
.ok(
{
"folder_id": folder.folder_id,
"name": folder.name,
"parent_id": folder.parent_id,
"description": folder.description,
"sort_order": folder.sort_order,
"created_at": folder.created_at.isoformat()
if folder.created_at
else None,
"updated_at": folder.updated_at.isoformat()
if folder.updated_at
else None,
},
)
.__dict__
)
except Exception as e:
logger.error(f"获取文件夹详情失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"获取文件夹详情失败: {e!s}").__dict__
async def create_folder(self):
"""创建文件夹"""
try:
data = await request.get_json()
name = data.get("name", "").strip()
parent_id = data.get("parent_id")
description = data.get("description")
sort_order = data.get("sort_order", 0)
if not name:
return Response().error("文件夹名称不能为空").__dict__
folder = await self.persona_mgr.create_folder(
name=name,
parent_id=parent_id,
description=description,
sort_order=sort_order,
)
return (
Response()
.ok(
{
"message": "文件夹创建成功",
"folder": {
"folder_id": folder.folder_id,
"name": folder.name,
"parent_id": folder.parent_id,
"description": folder.description,
"sort_order": folder.sort_order,
"created_at": folder.created_at.isoformat()
if folder.created_at
else None,
"updated_at": folder.updated_at.isoformat()
if folder.updated_at
else None,
},
},
)
.__dict__
)
except Exception as e:
logger.error(f"创建文件夹失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"创建文件夹失败: {e!s}").__dict__
async def update_folder(self):
"""更新文件夹信息"""
try:
data = await request.get_json()
folder_id = data.get("folder_id")
name = data.get("name")
parent_id = data.get("parent_id")
description = data.get("description")
sort_order = data.get("sort_order")
if not folder_id:
return Response().error("缺少必要参数: folder_id").__dict__
await self.persona_mgr.update_folder(
folder_id=folder_id,
name=name,
parent_id=parent_id,
description=description,
sort_order=sort_order,
)
return Response().ok({"message": "文件夹更新成功"}).__dict__
except Exception as e:
logger.error(f"更新文件夹失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"更新文件夹失败: {e!s}").__dict__
async def delete_folder(self):
"""删除文件夹"""
try:
data = await request.get_json()
folder_id = data.get("folder_id")
if not folder_id:
return Response().error("缺少必要参数: folder_id").__dict__
await self.persona_mgr.delete_folder(folder_id)
return Response().ok({"message": "文件夹删除成功"}).__dict__
except Exception as e:
logger.error(f"删除文件夹失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"删除文件夹失败: {e!s}").__dict__
async def reorder_items(self):
"""批量更新排序顺序
请求体格式:
{
"items": [
{"id": "persona_id_1", "type": "persona", "sort_order": 0},
{"id": "persona_id_2", "type": "persona", "sort_order": 1},
{"id": "folder_id_1", "type": "folder", "sort_order": 0},
...
]
}
"""
try:
data = await request.get_json()
items = data.get("items", [])
if not items:
return Response().error("items 不能为空").__dict__
# 验证每个 item 的格式
for item in items:
if not all(k in item for k in ("id", "type", "sort_order")):
return (
Response()
.error("每个 item 必须包含 id, type, sort_order 字段")
.__dict__
)
if item["type"] not in ("persona", "folder"):
return (
Response()
.error("type 字段必须是 'persona''folder'")
.__dict__
)
await self.persona_mgr.batch_update_sort_order(items)
return Response().ok({"message": "排序更新成功"}).__dict__
except Exception as e:
logger.error(f"更新排序失败: {e!s}\n{traceback.format_exc()}")
return Response().error(f"更新排序失败: {e!s}").__dict__
@@ -0,0 +1,132 @@
<template>
<v-dialog v-model="showDialog" max-width="450px">
<v-card>
<v-card-title>
<v-icon class="mr-2">mdi-folder-plus</v-icon>
{{ labels.title }}
</v-card-title>
<v-card-text>
<v-form ref="form" v-model="formValid">
<v-text-field v-model="formData.name" :label="mergedLabels.nameLabel"
:rules="[(v: any) => !!v || mergedLabels.nameRequired]" variant="outlined"
density="comfortable" autofocus class="mb-3" />
<v-textarea v-model="formData.description" :label="labels.descriptionLabel" variant="outlined"
rows="3" density="comfortable" hide-details />
</v-form>
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="closeDialog">
{{ labels.cancelButton }}
</v-btn>
<v-btn color="primary" variant="flat" @click="submitForm" :loading="loading" :disabled="!formValid">
{{ labels.createButton }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { CreateFolderData } from './types';
interface DefaultLabels {
title: string;
nameLabel: string;
descriptionLabel: string;
nameRequired: string;
cancelButton: string;
createButton: string;
}
const defaultLabels: DefaultLabels = {
title: '创建文件夹',
nameLabel: '名称',
descriptionLabel: '描述',
nameRequired: '请输入文件夹名称',
cancelButton: '取消',
createButton: '创建'
};
export default defineComponent({
name: 'BaseCreateFolderDialog',
props: {
modelValue: {
type: Boolean,
default: false
},
parentFolderId: {
type: String as PropType<string | null>,
default: null
},
labels: {
type: Object as PropType<Partial<DefaultLabels>>,
default: () => ({})
}
},
emits: ['update:modelValue', 'create'],
data() {
return {
formValid: false,
loading: false,
formData: {
name: '',
description: ''
}
};
},
computed: {
showDialog: {
get(): boolean {
return this.modelValue;
},
set(value: boolean) {
this.$emit('update:modelValue', value);
}
},
mergedLabels(): DefaultLabels {
return { ...defaultLabels, ...this.labels };
}
},
watch: {
modelValue(newValue: boolean) {
if (newValue) {
this.resetForm();
}
}
},
methods: {
resetForm() {
this.formData = {
name: '',
description: ''
};
if (this.$refs.form) {
(this.$refs.form as any).resetValidation();
}
},
closeDialog() {
this.showDialog = false;
},
async submitForm() {
if (!this.formValid) return;
const data: CreateFolderData = {
name: this.formData.name,
description: this.formData.description || undefined,
parent_id: this.parentFolderId
};
this.$emit('create', data);
},
setLoading(value: boolean) {
this.loading = value;
}
}
});
</script>
@@ -0,0 +1,84 @@
<template>
<v-breadcrumbs :items="computedItems" class="base-folder-breadcrumb pa-0">
<template v-slot:prepend>
<v-icon size="small" class="mr-1">mdi-folder-outline</v-icon>
</template>
<template v-slot:item="{ item }">
<v-breadcrumbs-item :disabled="(item as any).disabled" @click="!(item as any).disabled && handleClick((item as any).folderId)"
:class="{ 'breadcrumb-link': !(item as any).disabled }">
<v-icon v-if="(item as any).isRoot" size="small" class="mr-1">mdi-home</v-icon>
{{ (item as any).title }}
</v-breadcrumbs-item>
</template>
<template v-slot:divider>
<v-icon size="small">mdi-chevron-right</v-icon>
</template>
</v-breadcrumbs>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { BreadcrumbItem, FolderTreeNode } from './types';
export default defineComponent({
name: 'BaseFolderBreadcrumb',
props: {
breadcrumbPath: {
type: Array as PropType<FolderTreeNode[]>,
required: true
},
currentFolderId: {
type: String as PropType<string | null>,
default: null
},
rootFolderName: {
type: String,
default: '根目录'
}
},
emits: ['navigate'],
computed: {
computedItems(): BreadcrumbItem[] {
const items: BreadcrumbItem[] = [
{
title: this.rootFolderName,
folderId: null,
disabled: this.currentFolderId === null,
isRoot: true
}
];
this.breadcrumbPath.forEach((folder, index) => {
items.push({
title: folder.name,
folderId: folder.folder_id,
disabled: index === this.breadcrumbPath.length - 1,
isRoot: false
});
});
return items;
}
},
methods: {
handleClick(folderId: string | null) {
this.$emit('navigate', folderId);
}
}
});
</script>
<style scoped>
.base-folder-breadcrumb {
font-size: 14px;
}
.breadcrumb-link {
cursor: pointer;
transition: color 0.2s;
}
.breadcrumb-link:hover {
color: rgb(var(--v-theme-primary));
}
</style>
@@ -0,0 +1,143 @@
<template>
<v-card class="base-folder-card" :class="{ 'drag-over': isDragOver }" rounded="lg" @click="$emit('click')" @contextmenu.prevent="$emit('contextmenu', $event)"
elevation="1" hover @dragover.prevent="handleDragOver" @dragleave="handleDragLeave" @drop.prevent="handleDrop">
<v-card-text class="d-flex align-center pa-3">
<v-icon size="40" color="amber-darken-2" class="mr-3">mdi-folder</v-icon>
<div class="folder-info flex-grow-1 overflow-hidden">
<div class="text-subtitle-1 font-weight-medium text-truncate">{{ folder.name }}</div>
<div v-if="folder.description" class="text-body-2 text-medium-emphasis text-truncate">
{{ folder.description }}
</div>
</div>
<v-menu offset-y>
<template v-slot:activator="{ props }">
<v-btn icon="mdi-dots-vertical" variant="text" size="small" v-bind="props" @click.stop />
</template>
<v-list density="compact">
<v-list-item @click.stop="$emit('open')">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-open</v-icon>
</template>
<v-list-item-title>{{ labels.open }}</v-list-item-title>
</v-list-item>
<v-list-item @click.stop="$emit('rename')">
<template v-slot:prepend>
<v-icon size="small">mdi-pencil</v-icon>
</template>
<v-list-item-title>{{ labels.rename }}</v-list-item-title>
</v-list-item>
<v-list-item @click.stop="$emit('move')">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-move</v-icon>
</template>
<v-list-item-title>{{ labels.moveTo }}</v-list-item-title>
</v-list-item>
<v-divider class="my-1" />
<v-list-item @click.stop="$emit('delete')" class="text-error">
<template v-slot:prepend>
<v-icon size="small" color="error">mdi-delete</v-icon>
</template>
<v-list-item-title>{{ labels.delete }}</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
</v-card-text>
</v-card>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { Folder } from './types';
interface DefaultLabels {
open: string;
rename: string;
moveTo: string;
delete: string;
}
const defaultLabels: DefaultLabels = {
open: '打开',
rename: '重命名',
moveTo: '移动到...',
delete: '删除'
};
export default defineComponent({
name: 'BaseFolderCard',
props: {
folder: {
type: Object as PropType<Folder>,
required: true
},
acceptDropTypes: {
type: Array as PropType<string[]>,
default: () => []
},
labels: {
type: Object as PropType<Partial<DefaultLabels>>,
default: () => ({})
}
},
emits: ['click', 'contextmenu', 'open', 'rename', 'move', 'delete', 'item-dropped'],
data() {
return {
isDragOver: false
};
},
computed: {
mergedLabels(): DefaultLabels {
return { ...defaultLabels, ...this.labels };
}
},
methods: {
handleDragOver(event: DragEvent) {
if (!event.dataTransfer) return;
event.dataTransfer.dropEffect = 'move';
this.isDragOver = true;
},
handleDragLeave() {
this.isDragOver = false;
},
handleDrop(event: DragEvent) {
this.isDragOver = false;
if (!event.dataTransfer) return;
try {
const data = JSON.parse(event.dataTransfer.getData('application/json'));
if (this.acceptDropTypes.length === 0 || this.acceptDropTypes.includes(data.type)) {
this.$emit('item-dropped', {
item_id: data.id || data.persona_id || data.item_id,
item_type: data.type,
target_folder_id: this.folder.folder_id,
source_data: data
});
}
} catch (e) {
console.error('Failed to parse drop data:', e);
}
}
}
});
</script>
<style scoped>
.base-folder-card {
cursor: pointer;
transition: all 0.2s ease;
}
.base-folder-card:hover {
transform: translateY(-2px);
}
.base-folder-card.drag-over {
background-color: rgba(var(--v-theme-primary), 0.15);
border: 2px dashed rgb(var(--v-theme-primary));
transform: scale(1.02);
}
.folder-info {
min-width: 0;
}
</style>
@@ -0,0 +1,513 @@
<template>
<div class="folder-item-selector">
<!-- 触发按钮区域 -->
<div class="d-flex align-center justify-space-between">
<span v-if="!modelValue" style="color: rgb(var(--v-theme-primaryText));">
{{ labels.notSelected || '未选择' }}
</span>
<span v-else>
{{ displayValue }}
</span>
<v-btn size="small" color="primary" variant="tonal" @click="openDialog">
{{ labels.buttonText || '选择...' }}
</v-btn>
</div>
<!-- 选择对话框 -->
<v-dialog v-model="dialog" max-width="1000px" min-width="800px">
<v-card class="selector-dialog-card">
<v-card-title class="dialog-title d-flex align-center py-4 px-5">
<v-icon class="mr-3" color="primary">mdi-account-circle</v-icon>
<span>{{ labels.dialogTitle || '选择项目' }}</span>
</v-card-title>
<v-divider />
<v-card-text class="pa-0" style="height: 600px; max-height: 80vh; overflow: hidden;">
<div class="selector-layout">
<!-- 左侧文件夹树 -->
<div class="folder-sidebar">
<div class="sidebar-header pa-3 pb-2">
<span class="text-caption text-medium-emphasis font-weight-medium">
<v-icon size="small" class="mr-1">mdi-folder-multiple</v-icon>
文件夹
</span>
</div>
<v-list density="compact" nav class="tree-list pa-2" bg-color="transparent">
<!-- 根目录 -->
<v-list-item :active="currentFolderId === null" @click="navigateToFolder(null)"
rounded="lg" class="mb-1 root-item">
<template v-slot:prepend>
<v-icon size="20" :color="currentFolderId === null ? 'primary' : ''">mdi-home</v-icon>
</template>
<v-list-item-title class="text-body-2">{{ labels.rootFolder || '根目录' }}</v-list-item-title>
</v-list-item>
<!-- 文件夹树 -->
<template v-if="!treeLoading">
<BaseMoveTargetNode v-for="folder in folderTree" :key="folder.folder_id"
:folder="folder" :depth="0" :selected-folder-id="currentFolderId"
:disabled-folder-ids="[]" @select="navigateToFolder" />
</template>
<div v-if="treeLoading" class="text-center pa-4">
<v-progress-circular indeterminate size="20" color="primary" />
</div>
</v-list>
</div>
<!-- 右侧项目列表 -->
<div class="items-panel">
<!-- 面包屑导航 -->
<div class="breadcrumb-bar px-4 py-3">
<v-breadcrumbs :items="breadcrumbItems" density="compact" class="pa-0">
<template v-slot:item="{ item }">
<v-breadcrumbs-item :disabled="(item as any).disabled"
@click="!(item as any).disabled && navigateToFolder((item as any).folderId)"
:class="{ 'breadcrumb-link': !(item as any).disabled }">
<v-icon v-if="(item as any).isRoot" size="small"
class="mr-1">mdi-home</v-icon>
{{ item.title }}
</v-breadcrumbs-item>
</template>
<template v-slot:divider>
<v-icon size="small" color="grey">mdi-chevron-right</v-icon>
</template>
</v-breadcrumbs>
</div>
<v-divider />
<!-- 项目列表 -->
<div class="items-list">
<v-progress-linear v-if="itemsLoading" indeterminate
color="primary" height="2"></v-progress-linear>
<!-- 子文件夹 -->
<v-list v-if="!itemsLoading" lines="two" class="pa-3 items-content">
<template v-if="currentSubFolders.length > 0">
<div class="section-label text-caption text-medium-emphasis mb-2 px-2">子文件夹</div>
<v-list-item v-for="folder in currentSubFolders" :key="'folder-' + folder.folder_id"
@click="navigateToFolder(folder.folder_id)" rounded="lg" class="mb-1 folder-item">
<template v-slot:prepend>
<v-avatar size="36" color="amber-lighten-4" class="mr-3">
<v-icon color="amber-darken-2" size="20">mdi-folder</v-icon>
</v-avatar>
</template>
<v-list-item-title class="font-weight-medium">{{ folder.name }}</v-list-item-title>
<template v-slot:append>
<v-icon size="20" color="grey">mdi-chevron-right</v-icon>
</template>
</v-list-item>
</template>
<!-- 项目列表 -->
<template v-if="currentItems.length > 0">
<div class="section-label text-caption text-medium-emphasis mb-2 px-2" :class="{ 'mt-4': currentSubFolders.length > 0 }">可选项目</div>
<v-list-item v-for="item in currentItems" :key="'item-' + getItemId(item)"
:value="getItemId(item)" @click="selectItem(item)"
:active="selectedItemId === getItemId(item)" rounded="lg" class="mb-1 persona-item"
:class="{ 'selected-item': selectedItemId === getItemId(item) }">
<template v-slot:prepend>
<v-avatar size="36" :color="selectedItemId === getItemId(item) ? 'primary-lighten-4' : 'grey-lighten-3'" class="mr-3">
<v-icon :color="selectedItemId === getItemId(item) ? 'primary' : 'grey-darken-1'" size="20">mdi-account</v-icon>
</v-avatar>
</template>
<v-list-item-title class="font-weight-medium">{{ getItemName(item) }}</v-list-item-title>
<v-list-item-subtitle v-if="getItemDescription(item)" class="text-truncate">
{{ truncateText(getItemDescription(item), 80) }}
</v-list-item-subtitle>
<template v-slot:append>
<v-icon v-if="selectedItemId === getItemId(item)"
color="primary" size="22">mdi-check-circle</v-icon>
</template>
</v-list-item>
</template>
<!-- 空状态 -->
<div v-if="currentSubFolders.length === 0 && currentItems.length === 0"
class="empty-state text-center py-12">
<v-icon size="64" color="grey-lighten-2">mdi-folder-open-outline</v-icon>
<p class="text-grey mt-4 text-body-2">{{ labels.emptyFolder || labels.noItems || '此文件夹为空' }}</p>
</div>
</v-list>
</div>
</div>
</div>
</v-card-text>
<v-card-actions class="pa-4">
<v-btn v-if="showCreateButton" variant="text" color="primary" prepend-icon="mdi-plus"
@click="$emit('create')">
{{ labels.createButton || '新建' }}
</v-btn>
<v-spacer></v-spacer>
<v-btn variant="text" @click="cancelSelection">{{ labels.cancelButton || '取消' }}</v-btn>
<v-btn color="primary" @click="confirmSelection" :disabled="!selectedItemId">
{{ labels.confirmButton || '确认' }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</div>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import BaseMoveTargetNode from './BaseMoveTargetNode.vue';
import type { FolderTreeNode, FolderItemSelectorLabels, SelectableItem } from './types';
export default defineComponent({
name: 'BaseFolderItemSelector',
components: {
BaseMoveTargetNode
},
props: {
modelValue: {
type: String,
default: ''
},
//
folderTree: {
type: Array as PropType<FolderTreeNode[]>,
default: () => []
},
//
items: {
type: Array as PropType<SelectableItem[]>,
default: () => []
},
//
treeLoading: {
type: Boolean,
default: false
},
itemsLoading: {
type: Boolean,
default: false
},
//
labels: {
type: Object as PropType<Partial<FolderItemSelectorLabels>>,
default: () => ({})
},
//
showCreateButton: {
type: Boolean,
default: false
},
// ""
defaultItem: {
type: Object as PropType<SelectableItem | null>,
default: null
},
//
itemIdField: {
type: String,
default: 'id'
},
itemNameField: {
type: String,
default: 'name'
},
itemDescriptionField: {
type: String,
default: 'description'
},
//
displayValueFormatter: {
type: Function as unknown as PropType<((value: string) => string) | null>,
default: null
}
},
emits: ['update:modelValue', 'navigate', 'create'],
data() {
return {
dialog: false,
selectedItemId: '' as string,
currentFolderId: null as string | null,
breadcrumbPath: [] as FolderTreeNode[]
};
},
computed: {
displayValue(): string {
if (this.displayValueFormatter) {
return this.displayValueFormatter(this.modelValue);
}
//
if (this.defaultItem && this.modelValue === this.getItemId(this.defaultItem)) {
return this.labels.defaultItem || this.getItemName(this.defaultItem);
}
return this.modelValue;
},
currentItems(): SelectableItem[] {
const items: SelectableItem[] = [];
//
if (this.currentFolderId === null && this.defaultItem) {
items.push(this.defaultItem);
}
//
items.push(...this.items);
return items;
},
currentSubFolders(): FolderTreeNode[] {
if (this.currentFolderId === null) {
return this.folderTree;
}
const folder = this.findFolderInTree(this.currentFolderId);
return folder?.children || [];
},
breadcrumbItems(): any[] {
const items: any[] = [
{
title: this.labels.rootFolder || '根目录',
folderId: null,
disabled: this.currentFolderId === null,
isRoot: true
}
];
this.breadcrumbPath.forEach((folder, index) => {
items.push({
title: folder.name,
folderId: folder.folder_id,
disabled: index === this.breadcrumbPath.length - 1,
isRoot: false
});
});
return items;
}
},
methods: {
getItemId(item: SelectableItem): string {
return String(item[this.itemIdField] || item.id || '');
},
getItemName(item: SelectableItem): string {
return String(item[this.itemNameField] || item.name || '');
},
getItemDescription(item: SelectableItem): string {
return String(item[this.itemDescriptionField] || item.description || '');
},
truncateText(text: string, maxLength: number): string {
if (!text) return '';
return text.length > maxLength ? text.substring(0, maxLength) + '...' : text;
},
openDialog() {
this.selectedItemId = this.modelValue || '';
this.currentFolderId = null;
this.breadcrumbPath = [];
this.dialog = true;
this.$emit('navigate', null);
},
navigateToFolder(folderId: string | null) {
this.currentFolderId = folderId;
this.updateBreadcrumb(folderId);
this.$emit('navigate', folderId);
},
findFolderInTree(folderId: string): FolderTreeNode | null {
const findNode = (nodes: FolderTreeNode[]): FolderTreeNode | null => {
for (const node of nodes) {
if (node.folder_id === folderId) {
return node;
}
if (node.children && node.children.length > 0) {
const found = findNode(node.children);
if (found) return found;
}
}
return null;
};
return findNode(this.folderTree);
},
findPathToFolder(folderId: string): FolderTreeNode[] {
const findPath = (nodes: FolderTreeNode[], path: FolderTreeNode[]): FolderTreeNode[] | null => {
for (const node of nodes) {
if (node.folder_id === folderId) {
return [...path, node];
}
if (node.children && node.children.length > 0) {
const result = findPath(node.children, [...path, node]);
if (result) return result;
}
}
return null;
};
return findPath(this.folderTree, []) || [];
},
updateBreadcrumb(folderId: string | null) {
if (folderId === null) {
this.breadcrumbPath = [];
} else {
this.breadcrumbPath = this.findPathToFolder(folderId);
}
},
selectItem(item: SelectableItem) {
this.selectedItemId = this.getItemId(item);
},
confirmSelection() {
this.$emit('update:modelValue', this.selectedItemId);
this.dialog = false;
},
cancelSelection() {
this.selectedItemId = this.modelValue || '';
this.dialog = false;
}
}
});
</script>
<style scoped>
.selector-dialog-card {
border-radius: 12px;
overflow: hidden;
}
.dialog-title {
font-size: 1.25rem;
font-weight: 500;
}
.selector-layout {
display: flex;
height: 100%;
}
.folder-sidebar {
width: 280px;
border-right: 1px solid rgba(var(--v-border-color), var(--v-border-opacity));
overflow-y: auto;
flex-shrink: 0;
background-color: transparent;
}
.sidebar-header {
border-bottom: 1px solid rgba(var(--v-border-color), 0.5);
}
.items-panel {
flex: 1;
display: flex;
flex-direction: column;
min-width: 0;
background-color: rgb(var(--v-theme-surface));
}
.breadcrumb-bar {
background-color: transparent;
min-height: 56px;
display: flex;
align-items: center;
}
.items-list {
flex: 1;
overflow-y: auto;
}
.items-content {
background-color: transparent;
}
.tree-list {
padding: 0;
}
.section-label {
text-transform: uppercase;
letter-spacing: 0.5px;
font-size: 0.7rem;
}
.breadcrumb-link {
cursor: pointer;
transition: color 0.2s;
}
.breadcrumb-link:hover {
color: rgb(var(--v-theme-primary));
}
.root-item {
margin-bottom: 4px;
}
.folder-item {
transition: all 0.15s ease;
}
.folder-item:hover {
background-color: rgba(var(--v-theme-primary), 0.06);
}
.persona-item {
transition: all 0.15s ease;
border: 1px solid transparent;
}
.persona-item:hover {
background-color: rgba(var(--v-theme-primary), 0.04);
}
.persona-item.selected-item {
background-color: rgba(var(--v-theme-primary), 0.08);
border-color: rgba(var(--v-theme-primary), 0.3);
}
.empty-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
min-height: 200px;
}
.v-list-item {
transition: all 0.15s ease;
}
.v-list-item:hover {
background-color: rgba(var(--v-theme-primary), 0.04);
}
.v-list-item.v-list-item--active {
background-color: rgba(var(--v-theme-primary), 0.08);
}
@media (max-width: 600px) {
.selector-layout {
flex-direction: column;
height: auto;
max-height: 500px;
}
.folder-sidebar {
width: 100%;
border-right: none;
border-bottom: 1px solid rgba(var(--v-border-color), var(--v-border-opacity));
max-height: 150px;
}
.items-list {
max-height: 300px;
}
}
</style>
@@ -0,0 +1,272 @@
<template>
<div class="base-folder-tree">
<!-- 搜索框 -->
<v-text-field v-model="searchQuery" :placeholder="labels.searchPlaceholder" prepend-inner-icon="mdi-magnify"
variant="outlined" density="compact" hide-details clearable class="mb-3" />
<!-- 根目录节点 -->
<v-list density="compact" nav class="tree-list" bg-color="transparent">
<v-list-item :active="currentFolderId === null" @click="handleFolderClick(null)" rounded="lg"
:class="['root-item', { 'drag-over': isRootDragOver }]"
@dragover.prevent="handleRootDragOver" @dragleave="handleRootDragLeave" @drop.prevent="handleRootDrop">
<template v-slot:prepend>
<v-icon>mdi-home</v-icon>
</template>
<v-list-item-title>{{ labels.rootFolder }}</v-list-item-title>
</v-list-item>
<!-- 文件夹树 -->
<template v-if="!treeLoading">
<BaseFolderTreeNode v-for="folder in filteredFolderTree" :key="folder.folder_id" :folder="folder"
:depth="0" :current-folder-id="currentFolderId" :search-query="searchQuery"
:expanded-folder-ids="expandedFolderIds" :accept-drop-types="acceptDropTypes"
@folder-click="handleFolderClick" @folder-context-menu="handleContextMenu"
@item-dropped="$emit('item-dropped', $event)"
@toggle-expansion="$emit('toggle-expansion', $event)"
@set-expansion="$emit('set-expansion', $event)" />
</template>
<!-- 加载状态 -->
<div v-if="treeLoading" class="text-center pa-4">
<v-progress-circular indeterminate size="24" />
</div>
<!-- 空状态 -->
<div v-if="!treeLoading && folderTree.length === 0" class="text-center pa-4 text-medium-emphasis">
<v-icon size="32" class="mb-2">mdi-folder-outline</v-icon>
<div class="text-body-2">{{ labels.noFolders }}</div>
</div>
</v-list>
<!-- 右键菜单 -->
<v-menu v-model="contextMenu.show" :target="contextMenu.target as any" location="end" :close-on-content-click="true">
<v-list density="compact">
<v-list-item @click="openFolder">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-open</v-icon>
</template>
<v-list-item-title>{{ mergedLabels.contextMenu.open }}</v-list-item-title>
</v-list-item>
<v-list-item @click="$emit('rename-folder', contextMenu.folder)">
<template v-slot:prepend>
<v-icon size="small">mdi-pencil</v-icon>
</template>
<v-list-item-title>{{ mergedLabels.contextMenu.rename }}</v-list-item-title>
</v-list-item>
<v-list-item @click="$emit('move-folder', contextMenu.folder)">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-move</v-icon>
</template>
<v-list-item-title>{{ mergedLabels.contextMenu.moveTo }}</v-list-item-title>
</v-list-item>
<v-divider class="my-1" />
<v-list-item @click="$emit('delete-folder', contextMenu.folder)" class="text-error">
<template v-slot:prepend>
<v-icon size="small" color="error">mdi-delete</v-icon>
</template>
<v-list-item-title>{{ mergedLabels.contextMenu.delete }}</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
</div>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { FolderTreeNode, ContextMenuEvent } from './types';
import BaseFolderTreeNode from './BaseFolderTreeNode.vue';
interface ContextMenuState {
show: boolean;
target: [number, number] | null;
folder: FolderTreeNode | null;
}
interface Folder {
folder_id: string;
name: string;
parent_id: string | null;
description?: string | null;
sort_order?: number;
created_at?: string;
updated_at?: string;
}
interface DefaultLabels {
searchPlaceholder: string;
rootFolder: string;
noFolders: string;
contextMenu: {
open: string;
rename: string;
moveTo: string;
delete: string;
};
}
const defaultLabels: DefaultLabels = {
searchPlaceholder: '搜索文件夹...',
rootFolder: '根目录',
noFolders: '暂无文件夹',
contextMenu: {
open: '打开',
rename: '重命名',
moveTo: '移动到...',
delete: '删除'
}
};
export default defineComponent({
name: 'BaseFolderTree',
components: {
BaseFolderTreeNode
},
props: {
folderTree: {
type: Array as PropType<FolderTreeNode[]>,
required: true
},
currentFolderId: {
type: String as PropType<string | null>,
default: null
},
expandedFolderIds: {
type: Array as PropType<string[]>,
default: () => []
},
treeLoading: {
type: Boolean,
default: false
},
acceptDropTypes: {
type: Array as PropType<string[]>,
default: () => []
},
labels: {
type: Object as PropType<Partial<DefaultLabels>>,
default: () => ({})
}
},
emits: [
'folder-click',
'rename-folder',
'move-folder',
'delete-folder',
'item-dropped',
'toggle-expansion',
'set-expansion'
],
data() {
return {
searchQuery: '',
isRootDragOver: false,
contextMenu: {
show: false,
target: null,
folder: null
} as ContextMenuState
};
},
computed: {
mergedLabels(): DefaultLabels {
return {
...defaultLabels,
...this.labels,
contextMenu: {
...defaultLabels.contextMenu,
...(this.labels?.contextMenu || {})
}
};
},
filteredFolderTree(): FolderTreeNode[] {
if (!this.searchQuery) {
return this.folderTree;
}
const query = this.searchQuery.toLowerCase();
return this.filterTreeBySearch(this.folderTree, query);
}
},
methods: {
filterTreeBySearch(nodes: FolderTreeNode[], query: string): FolderTreeNode[] {
return nodes.filter(node => {
const matches = node.name.toLowerCase().includes(query);
const childMatches = this.filterTreeBySearch(node.children || [], query);
return matches || childMatches.length > 0;
}).map(node => ({
...node,
children: this.filterTreeBySearch(node.children || [], query)
}));
},
handleFolderClick(folderId: string | null) {
this.$emit('folder-click', folderId);
},
handleRootDragOver(event: DragEvent) {
if (!event.dataTransfer) return;
event.dataTransfer.dropEffect = 'move';
this.isRootDragOver = true;
},
handleRootDragLeave() {
this.isRootDragOver = false;
},
handleRootDrop(event: DragEvent) {
this.isRootDragOver = false;
if (!event.dataTransfer) return;
try {
const data = JSON.parse(event.dataTransfer.getData('application/json'));
if (this.acceptDropTypes.length === 0 || this.acceptDropTypes.includes(data.type)) {
this.$emit('item-dropped', {
item_id: data.id || data.persona_id || data.item_id,
item_type: data.type,
target_folder_id: null,
source_data: data
});
}
} catch (e) {
console.error('Failed to parse drop data:', e);
}
},
handleContextMenu(eventData: ContextMenuEvent) {
const { event, folder } = eventData;
this.contextMenu.target = [event.clientX, event.clientY];
this.contextMenu.folder = folder as FolderTreeNode;
this.contextMenu.show = true;
},
openFolder() {
if (this.contextMenu.folder) {
this.$emit('folder-click', this.contextMenu.folder.folder_id);
}
}
}
});
</script>
<style scoped>
.base-folder-tree {
height: 100%;
display: flex;
flex-direction: column;
}
.tree-list {
flex: 1;
overflow-y: auto;
}
.root-item {
margin-bottom: 4px;
transition: all 0.2s ease;
}
.root-item.drag-over {
background-color: rgba(var(--v-theme-primary), 0.15);
border: 2px dashed rgb(var(--v-theme-primary));
border-radius: 8px;
}
</style>
@@ -0,0 +1,154 @@
<template>
<div class="base-folder-tree-node">
<v-list-item :active="currentFolderId === folder.folder_id" @click.stop="$emit('folder-click', folder.folder_id)"
@contextmenu.prevent="handleContextMenu" rounded="lg" :style="{ paddingLeft: `${(depth + 1) * 16}px` }"
:class="['folder-item', { 'drag-over': isDragOver }]"
@dragover.prevent="handleDragOver" @dragleave="handleDragLeave" @drop.prevent="handleDrop">
<template v-slot:prepend>
<v-btn v-if="hasChildren" icon variant="text" size="x-small" @click.stop="toggleExpand"
class="expand-btn">
<v-icon size="16">{{ isExpanded ? 'mdi-chevron-down' : 'mdi-chevron-right' }}</v-icon>
</v-btn>
<div v-else class="expand-placeholder"></div>
<v-icon :color="currentFolderId === folder.folder_id ? 'primary' : ''">
{{ isExpanded ? 'mdi-folder-open' : 'mdi-folder' }}
</v-icon>
</template>
<v-list-item-title class="text-truncate">{{ folder.name }}</v-list-item-title>
</v-list-item>
<!-- 子文件夹 -->
<v-expand-transition>
<div v-show="isExpanded && hasChildren">
<BaseFolderTreeNode v-for="child in folder.children" :key="child.folder_id" :folder="child" :depth="depth + 1"
:current-folder-id="currentFolderId" :search-query="searchQuery"
:expanded-folder-ids="expandedFolderIds" :accept-drop-types="acceptDropTypes"
@folder-click="$emit('folder-click', $event)"
@folder-context-menu="$emit('folder-context-menu', $event)"
@item-dropped="$emit('item-dropped', $event)"
@toggle-expansion="$emit('toggle-expansion', $event)"
@set-expansion="$emit('set-expansion', $event)" />
</div>
</v-expand-transition>
</div>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { FolderTreeNode } from './types';
export default defineComponent({
name: 'BaseFolderTreeNode',
props: {
folder: {
type: Object as PropType<FolderTreeNode>,
required: true
},
depth: {
type: Number,
default: 0
},
currentFolderId: {
type: String as PropType<string | null>,
default: null
},
searchQuery: {
type: String,
default: ''
},
expandedFolderIds: {
type: Array as PropType<string[]>,
default: () => []
},
acceptDropTypes: {
type: Array as PropType<string[]>,
default: () => []
}
},
emits: ['folder-click', 'folder-context-menu', 'item-dropped', 'toggle-expansion', 'set-expansion'],
data() {
return {
isDragOver: false
};
},
computed: {
hasChildren(): boolean {
return this.folder.children && this.folder.children.length > 0;
},
isExpanded(): boolean {
return this.expandedFolderIds.includes(this.folder.folder_id);
}
},
watch: {
searchQuery: {
immediate: true,
handler(newQuery: string) {
//
if (newQuery && this.hasChildren) {
this.$emit('set-expansion', { folderId: this.folder.folder_id, expanded: true });
}
}
}
},
methods: {
toggleExpand() {
this.$emit('toggle-expansion', this.folder.folder_id);
},
handleContextMenu(event: MouseEvent) {
this.$emit('folder-context-menu', { event, folder: this.folder });
},
handleDragOver(event: DragEvent) {
if (!event.dataTransfer) return;
event.dataTransfer.dropEffect = 'move';
this.isDragOver = true;
},
handleDragLeave() {
this.isDragOver = false;
},
handleDrop(event: DragEvent) {
this.isDragOver = false;
if (!event.dataTransfer) return;
try {
const data = JSON.parse(event.dataTransfer.getData('application/json'));
if (this.acceptDropTypes.length === 0 || this.acceptDropTypes.includes(data.type)) {
this.$emit('item-dropped', {
item_id: data.id || data.persona_id || data.item_id,
item_type: data.type,
target_folder_id: this.folder.folder_id,
source_data: data
});
}
} catch (e) {
console.error('Failed to parse drop data:', e);
}
}
}
});
</script>
<style scoped>
.base-folder-tree-node {
width: 100%;
}
.folder-item {
min-height: 36px;
transition: all 0.2s ease;
}
.folder-item.drag-over {
background-color: rgba(var(--v-theme-primary), 0.15);
border: 2px dashed rgb(var(--v-theme-primary));
border-radius: 8px;
}
.expand-btn {
margin-right: 4px;
}
.expand-placeholder {
width: 28px;
flex-shrink: 0;
}
</style>
@@ -0,0 +1,93 @@
<template>
<div class="base-move-target-node">
<v-list-item :active="selectedFolderId === folder.folder_id" :disabled="isDisabled"
@click.stop="!isDisabled && $emit('select', folder.folder_id)" rounded="lg"
:style="{ paddingLeft: `${(depth + 1) * 16}px` }" class="folder-item">
<template v-slot:prepend>
<v-btn v-if="hasChildren" icon variant="text" size="x-small" @click.stop="toggleExpand"
class="expand-btn" :disabled="isDisabled">
<v-icon size="16">{{ isExpanded ? 'mdi-chevron-down' : 'mdi-chevron-right' }}</v-icon>
</v-btn>
<div v-else class="expand-placeholder"></div>
<v-icon :color="isDisabled ? 'grey' : (selectedFolderId === folder.folder_id ? 'primary' : '')">
{{ isExpanded ? 'mdi-folder-open' : 'mdi-folder' }}
</v-icon>
</template>
<v-list-item-title class="text-truncate">{{ folder.name }}</v-list-item-title>
</v-list-item>
<!-- 子文件夹 -->
<v-expand-transition>
<div v-show="isExpanded && hasChildren">
<BaseMoveTargetNode v-for="child in folder.children" :key="child.folder_id" :folder="child" :depth="depth + 1"
:selected-folder-id="selectedFolderId" :disabled-folder-ids="disabledFolderIds"
@select="$emit('select', $event)" />
</div>
</v-expand-transition>
</div>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { FolderTreeNode } from './types';
export default defineComponent({
name: 'BaseMoveTargetNode',
props: {
folder: {
type: Object as PropType<FolderTreeNode>,
required: true
},
depth: {
type: Number,
default: 0
},
selectedFolderId: {
type: String as PropType<string | null>,
default: null
},
disabledFolderIds: {
type: Array as PropType<string[]>,
default: () => []
}
},
emits: ['select'],
data() {
return {
isExpanded: true
};
},
computed: {
hasChildren(): boolean {
return this.folder.children && this.folder.children.length > 0;
},
isDisabled(): boolean {
return this.disabledFolderIds.includes(this.folder.folder_id);
}
},
methods: {
toggleExpand() {
this.isExpanded = !this.isExpanded;
}
}
});
</script>
<style scoped>
.base-move-target-node {
width: 100%;
}
.folder-item {
min-height: 36px;
}
.expand-btn {
margin-right: 4px;
}
.expand-placeholder {
width: 28px;
flex-shrink: 0;
}
</style>
@@ -0,0 +1,178 @@
<template>
<v-dialog v-model="showDialog" max-width="500px" persistent>
<v-card>
<v-card-title>
<v-icon class="mr-2">mdi-folder-move</v-icon>
{{ labels.title }}
</v-card-title>
<v-card-text>
<p class="text-body-2 text-medium-emphasis mb-4">
{{ labels.description }}
</p>
<!-- 文件夹选择树 -->
<div class="folder-select-tree">
<v-list density="compact" nav class="tree-list">
<!-- 根目录选项 -->
<v-list-item :active="selectedFolderId === null" @click="selectFolder(null)" rounded="lg"
class="mb-1">
<template v-slot:prepend>
<v-icon>mdi-home</v-icon>
</template>
<v-list-item-title>{{ labels.rootFolder }}</v-list-item-title>
</v-list-item>
<!-- 文件夹树 -->
<template v-if="!treeLoading">
<BaseMoveTargetNode v-for="folder in folderTree" :key="folder.folder_id" :folder="folder"
:depth="0" :selected-folder-id="selectedFolderId" :disabled-folder-ids="disabledFolderIds"
@select="selectFolder" />
</template>
<!-- 加载状态 -->
<div v-if="treeLoading" class="text-center pa-4">
<v-progress-circular indeterminate size="24" />
</div>
</v-list>
</div>
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="closeDialog">
{{ labels.cancelButton }}
</v-btn>
<v-btn color="primary" variant="flat" @click="submitMove" :loading="loading">
{{ labels.moveButton }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import type { FolderTreeNode } from './types';
import BaseMoveTargetNode from './BaseMoveTargetNode.vue';
import { collectFolderAndChildrenIds } from './useFolderManager';
interface DefaultLabels {
title: string;
description: string;
rootFolder: string;
cancelButton: string;
moveButton: string;
}
const defaultLabels: DefaultLabels = {
title: '移动到文件夹',
description: '选择目标文件夹',
rootFolder: '根目录',
cancelButton: '取消',
moveButton: '移动'
};
export default defineComponent({
name: 'BaseMoveToFolderDialog',
components: {
BaseMoveTargetNode
},
props: {
modelValue: {
type: Boolean,
default: false
},
folderTree: {
type: Array as PropType<FolderTreeNode[]>,
required: true
},
treeLoading: {
type: Boolean,
default: false
},
// ID
currentFolderId: {
type: String as PropType<string | null>,
default: null
},
// ID
itemCurrentFolderId: {
type: String as PropType<string | null>,
default: null
},
//
isMovingFolder: {
type: Boolean,
default: false
},
labels: {
type: Object as PropType<Partial<DefaultLabels>>,
default: () => ({})
}
},
emits: ['update:modelValue', 'move'],
data() {
return {
selectedFolderId: null as string | null,
loading: false
};
},
computed: {
showDialog: {
get(): boolean {
return this.modelValue;
},
set(value: boolean) {
this.$emit('update:modelValue', value);
}
},
mergedLabels(): DefaultLabels {
return { ...defaultLabels, ...this.labels };
},
// ID
disabledFolderIds(): string[] {
if (!this.isMovingFolder || !this.currentFolderId) return [];
return collectFolderAndChildrenIds(this.folderTree, this.currentFolderId);
}
},
watch: {
modelValue(newValue: boolean) {
if (newValue) {
//
this.selectedFolderId = this.itemCurrentFolderId;
}
}
},
methods: {
selectFolder(folderId: string | null) {
//
if (folderId && this.disabledFolderIds.includes(folderId)) return;
this.selectedFolderId = folderId;
},
closeDialog() {
this.showDialog = false;
},
submitMove() {
this.$emit('move', this.selectedFolderId);
},
setLoading(value: boolean) {
this.loading = value;
}
}
});
</script>
<style scoped>
.folder-select-tree {
max-height: 400px;
overflow-y: auto;
border: 1px solid rgba(var(--v-border-color), var(--v-border-opacity));
border-radius: 8px;
}
.tree-list {
padding: 8px;
}
</style>
+349
View File
@@ -0,0 +1,349 @@
# 通用文件夹管理组件库
这是一个可复用的文件夹管理 UI 组件库,提供了完整的文件夹树、面包屑导航、拖放操作等功能。可用于管理各种类型的项目,如 Persona、模板、知识库等。
## 组件列表
| 组件 | 说明 |
|------|------|
| `BaseFolderTree` | 文件夹树组件,支持搜索、展开/折叠、右键菜单、拖放 |
| `BaseFolderTreeNode` | 文件夹树节点组件(内部使用) |
| `BaseFolderCard` | 文件夹卡片组件,用于网格布局展示 |
| `BaseFolderBreadcrumb` | 面包屑导航组件 |
| `BaseCreateFolderDialog` | 创建文件夹对话框 |
| `BaseMoveToFolderDialog` | 移动项目到文件夹对话框 |
| `BaseMoveTargetNode` | 移动对话框中的目标文件夹节点(内部使用) |
## Composable
### `useFolderManager`
提供文件夹管理的核心逻辑,包括状态管理、导航、CRUD 操作等。
```typescript
import { useFolderManager } from '@/components/folder';
const {
// 状态
folderTree,
currentFolderId,
currentFolders,
breadcrumbPath,
expandedFolderIds,
loading,
treeLoading,
// 计算属性
currentFolderName,
breadcrumbItems,
// 方法
loadFolderTree,
navigateToFolder,
refreshCurrentFolder,
createFolder,
updateFolder,
deleteFolder,
moveFolder,
toggleFolderExpansion,
setFolderExpansion,
findFolderInTree,
findPathToFolder,
filterTreeBySearch,
} = useFolderManager({
operations: {
loadFolderTree: async () => {
const response = await axios.get('/api/your-module/folder/tree');
return response.data.data;
},
loadSubFolders: async (parentId) => {
const response = await axios.get('/api/your-module/folder/list', {
params: { parent_id: parentId ?? '' }
});
return response.data.data;
},
createFolder: async (data) => {
const response = await axios.post('/api/your-module/folder/create', data);
return response.data.data.folder;
},
updateFolder: async (data) => {
await axios.post('/api/your-module/folder/update', data);
},
deleteFolder: async (folderId) => {
await axios.post('/api/your-module/folder/delete', { folder_id: folderId });
},
},
rootFolderName: '根目录',
autoLoad: true,
});
```
## 使用示例
### 基础用法
```vue
<template>
<div class="folder-manager">
<!-- 侧边栏 -->
<div class="sidebar">
<BaseFolderTree
:folder-tree="folderTree"
:current-folder-id="currentFolderId"
:expanded-folder-ids="expandedFolderIds"
:tree-loading="treeLoading"
:accept-drop-types="['item']"
:labels="treeLabels"
@folder-click="navigateToFolder"
@rename-folder="handleRenameFolder"
@move-folder="handleMoveFolder"
@delete-folder="handleDeleteFolder"
@item-dropped="handleItemDropped"
@toggle-expansion="toggleFolderExpansion"
/>
</div>
<!-- 主内容区 -->
<div class="main-content">
<!-- 面包屑 -->
<BaseFolderBreadcrumb
:breadcrumb-path="breadcrumbPath"
:current-folder-id="currentFolderId"
root-folder-name="根目录"
@navigate="navigateToFolder"
/>
<!-- 文件夹卡片 -->
<v-row>
<v-col v-for="folder in currentFolders" :key="folder.folder_id" cols="3">
<BaseFolderCard
:folder="folder"
:accept-drop-types="['item']"
:labels="cardLabels"
@click="navigateToFolder(folder.folder_id)"
@open="navigateToFolder(folder.folder_id)"
@rename="handleRenameFolder(folder)"
@move="handleMoveFolder(folder)"
@delete="handleDeleteFolder(folder)"
@item-dropped="handleItemDropped"
/>
</v-col>
</v-row>
</div>
<!-- 创建文件夹对话框 -->
<BaseCreateFolderDialog
v-model="showCreateDialog"
:parent-folder-id="currentFolderId"
:labels="createDialogLabels"
@create="handleCreateFolder"
/>
<!-- 移动对话框 -->
<BaseMoveToFolderDialog
v-model="showMoveDialog"
:folder-tree="folderTree"
:tree-loading="treeLoading"
:current-folder-id="movingFolder?.folder_id"
:item-current-folder-id="movingFolder?.parent_id"
:is-moving-folder="true"
:labels="moveDialogLabels"
@move="handleMove"
/>
</div>
</template>
<script setup lang="ts">
import { ref } from 'vue';
import {
BaseFolderTree,
BaseFolderCard,
BaseFolderBreadcrumb,
BaseCreateFolderDialog,
BaseMoveToFolderDialog,
useFolderManager,
} from '@/components/folder';
const folderManager = useFolderManager({
operations: {
// ... 实现你的 API 调用
},
});
const {
folderTree,
currentFolderId,
currentFolders,
breadcrumbPath,
expandedFolderIds,
treeLoading,
navigateToFolder,
toggleFolderExpansion,
createFolder,
} = folderManager;
const showCreateDialog = ref(false);
const showMoveDialog = ref(false);
const movingFolder = ref(null);
// 自定义标签
const treeLabels = {
searchPlaceholder: '搜索文件夹...',
rootFolder: '根目录',
noFolders: '暂无文件夹',
contextMenu: {
open: '打开',
rename: '重命名',
moveTo: '移动到...',
delete: '删除',
},
};
const cardLabels = {
open: '打开',
rename: '重命名',
moveTo: '移动到...',
delete: '删除',
};
const createDialogLabels = {
title: '创建文件夹',
nameLabel: '名称',
descriptionLabel: '描述',
nameRequired: '请输入名称',
cancelButton: '取消',
createButton: '创建',
};
// 处理函数
async function handleCreateFolder(data) {
await createFolder(data);
showCreateDialog.value = false;
}
function handleRenameFolder(folder) {
// 打开重命名对话框
}
function handleMoveFolder(folder) {
movingFolder.value = folder;
showMoveDialog.value = true;
}
function handleDeleteFolder(folder) {
// 确认并删除
}
function handleItemDropped({ item_id, item_type, target_folder_id }) {
// 处理拖放
}
async function handleMove(targetFolderId) {
// 执行移动
showMoveDialog.value = false;
}
</script>
```
## 类型定义
```typescript
// 文件夹基础接口
interface Folder {
folder_id: string;
name: string;
parent_id: string | null;
description?: string | null;
sort_order?: number;
created_at?: string;
updated_at?: string;
}
// 文件夹树节点接口
interface FolderTreeNode extends Folder {
children: FolderTreeNode[];
}
// 拖放事件数据
interface DropEventData {
item_id: string;
item_type: string;
target_folder_id: string | null;
source_data?: any;
}
// 创建文件夹数据
interface CreateFolderData {
name: string;
parent_id?: string | null;
description?: string;
}
```
## 国际化支持
所有组件都支持通过 `labels` prop 自定义文本,方便集成到不同的国际化方案中:
```vue
<BaseFolderTree
:labels="{
searchPlaceholder: t('folder.search'),
rootFolder: t('folder.root'),
noFolders: t('folder.empty'),
contextMenu: {
open: t('folder.menu.open'),
rename: t('folder.menu.rename'),
moveTo: t('folder.menu.move'),
delete: t('folder.menu.delete'),
},
}"
/>
```
## 拖放支持
组件内置了拖放支持,可以通过 `acceptDropTypes` 指定接受的拖放类型:
```vue
<!-- 只接受 'persona' 类型的拖放 -->
<BaseFolderTree
:accept-drop-types="['persona']"
@item-dropped="handleDrop"
/>
<!-- 拖放事件处理 -->
<script setup>
function handleDrop({ item_id, item_type, target_folder_id, source_data }) {
if (item_type === 'persona') {
// 移动 persona 到目标文件夹
movePersonaToFolder(item_id, target_folder_id);
}
}
</script>
```
## 与 Pinia Store 集成
如果你更喜欢使用 Pinia Store 管理状态,可以参考现有的 `personaStore.ts` 实现:
```typescript
// stores/myFolderStore.ts
import { defineStore } from 'pinia';
import type { FolderTreeNode, Folder } from '@/components/folder';
export const useMyFolderStore = defineStore('myFolder', {
state: () => ({
folderTree: [] as FolderTreeNode[],
currentFolderId: null as string | null,
currentFolders: [] as Folder[],
// ...
}),
actions: {
async loadFolderTree() {
// ...
},
// ...
},
});
```
+46
View File
@@ -0,0 +1,46 @@
/**
*
*
* UI
* persona
*
* 使:
* ```vue
* <script setup>
* import {
* BaseFolderTree,
* BaseFolderCard,
* BaseFolderBreadcrumb,
* BaseCreateFolderDialog,
* BaseMoveToFolderDialog,
* useFolderManager
* } from '@/components/folder';
*
* const folderManager = useFolderManager({
* operations: {
* loadFolderTree: async () => { ... },
* loadSubFolders: async (parentId) => { ... },
* createFolder: async (data) => { ... },
* updateFolder: async (data) => { ... },
* deleteFolder: async (folderId) => { ... },
* }
* });
* </script>
* ```
*/
// 类型导出
export * from './types';
// Composable 导出
export { useFolderManager, collectFolderAndChildrenIds } from './useFolderManager';
export type { UseFolderManagerOptions, UseFolderManagerReturn } from './useFolderManager';
// 组件导出
export { default as BaseFolderTree } from './BaseFolderTree.vue';
export { default as BaseFolderTreeNode } from './BaseFolderTreeNode.vue';
export { default as BaseFolderCard } from './BaseFolderCard.vue';
export { default as BaseFolderBreadcrumb } from './BaseFolderBreadcrumb.vue';
export { default as BaseCreateFolderDialog } from './BaseCreateFolderDialog.vue';
export { default as BaseMoveToFolderDialog } from './BaseMoveToFolderDialog.vue';
export { default as BaseMoveTargetNode } from './BaseMoveTargetNode.vue';
+249
View File
@@ -0,0 +1,249 @@
/**
*
*
* persona
*/
/**
*
*/
export interface Folder {
folder_id: string;
name: string;
parent_id: string | null;
description?: string | null;
sort_order?: number;
created_at?: string;
updated_at?: string;
}
/**
*
*/
export interface FolderTreeNode extends Folder {
children: FolderTreeNode[];
}
/**
*
*/
export interface DraggableItem {
id: string;
type: string;
[key: string]: any;
}
/**
*
*/
export interface DropEventData {
item_id: string;
item_type: string;
target_folder_id: string | null;
source_data?: any;
}
/**
* - 使
*/
export interface FolderOperations {
// 加载文件夹树
loadFolderTree: () => Promise<FolderTreeNode[]>;
// 加载指定文件夹的子文件夹
loadSubFolders: (parentId: string | null) => Promise<Folder[]>;
// 创建文件夹
createFolder: (data: CreateFolderData) => Promise<Folder>;
// 更新文件夹
updateFolder: (data: UpdateFolderData) => Promise<void>;
// 删除文件夹
deleteFolder: (folderId: string) => Promise<void>;
// 移动文件夹
moveFolder?: (folderId: string, targetParentId: string | null) => Promise<void>;
}
/**
*
*/
export interface CreateFolderData {
name: string;
parent_id?: string | null;
description?: string;
}
/**
*
*/
export interface UpdateFolderData {
folder_id: string;
name?: string;
description?: string;
parent_id?: string | null;
}
/**
*
*/
export interface FolderManagerState {
folderTree: FolderTreeNode[];
currentFolderId: string | null;
currentFolders: Folder[];
breadcrumbPath: FolderTreeNode[];
expandedFolderIds: string[];
loading: boolean;
treeLoading: boolean;
}
/**
*
*/
export interface BreadcrumbItem {
title: string;
folderId: string | null;
disabled: boolean;
isRoot: boolean;
}
/**
*
*/
export interface ContextMenuEvent {
event: MouseEvent;
folder: Folder;
}
/**
* i18n
* 使
*/
export interface FolderI18nKeys {
// 搜索框
searchPlaceholder?: string;
// 根目录
rootFolder?: string;
// 侧边栏标题
sidebarTitle?: string;
// 空状态
noFolders?: string;
// 文件夹标题
foldersTitle?: string;
// 按钮
buttons?: {
create?: string;
cancel?: string;
save?: string;
delete?: string;
move?: string;
};
// 表单
form?: {
name?: string;
description?: string;
};
// 验证
validation?: {
nameRequired?: string;
};
// 右键菜单
contextMenu?: {
open?: string;
rename?: string;
moveTo?: string;
delete?: string;
};
// 对话框
dialogs?: {
createTitle?: string;
renameTitle?: string;
deleteTitle?: string;
deleteMessage?: string;
deleteWarning?: string;
moveTitle?: string;
moveDescription?: string;
};
// 消息
messages?: {
createSuccess?: string;
createError?: string;
renameSuccess?: string;
renameError?: string;
deleteSuccess?: string;
deleteError?: string;
moveSuccess?: string;
moveError?: string;
};
}
/**
* Props
*/
export interface BaseFolderProps {
// i18n 翻译函数
t?: (key: string, params?: Record<string, any>) => string;
// i18n 键配置
i18nKeys?: FolderI18nKeys;
}
/**
*
*/
export interface SelectableItem {
id: string;
name: string;
description?: string | null;
folder_id?: string | null;
[key: string]: any;
}
/**
*
*/
export interface FolderItemSelectorOperations<T extends SelectableItem> {
// 加载文件夹树
loadFolderTree: () => Promise<FolderTreeNode[]>;
// 加载指定文件夹下的项目
loadItemsInFolder: (folderId: string | null) => Promise<T[]>;
// 创建项目(可选)
createItem?: (data: any) => Promise<T>;
}
/**
*
*/
export interface FolderItemSelectorLabels {
// 对话框
dialogTitle?: string;
notSelected?: string;
buttonText?: string;
// 项目列表
noItems?: string;
defaultItem?: string;
noDescription?: string;
emptyFolder?: string;
// 按钮
createButton?: string;
confirmButton?: string;
cancelButton?: string;
// 文件夹
rootFolder?: string;
}
@@ -0,0 +1,324 @@
/**
* Composable
*
*
*/
import { ref, computed, reactive, type Ref, type ComputedRef } from 'vue';
import type {
Folder,
FolderTreeNode,
FolderOperations,
CreateFolderData,
UpdateFolderData,
BreadcrumbItem,
} from './types';
export interface UseFolderManagerOptions {
// 文件夹操作实现
operations: FolderOperations;
// 根目录显示名称
rootFolderName?: string;
// 是否自动加载
autoLoad?: boolean;
}
export interface UseFolderManagerReturn {
// 状态
folderTree: Ref<FolderTreeNode[]>;
currentFolderId: Ref<string | null>;
currentFolders: Ref<Folder[]>;
breadcrumbPath: Ref<FolderTreeNode[]>;
expandedFolderIds: Ref<string[]>;
loading: Ref<boolean>;
treeLoading: Ref<boolean>;
// 计算属性
currentFolderName: ComputedRef<string>;
breadcrumbItems: ComputedRef<BreadcrumbItem[]>;
// 方法
loadFolderTree: () => Promise<void>;
navigateToFolder: (folderId: string | null) => Promise<void>;
refreshCurrentFolder: () => Promise<void>;
createFolder: (data: CreateFolderData) => Promise<Folder>;
updateFolder: (data: UpdateFolderData) => Promise<void>;
deleteFolder: (folderId: string) => Promise<void>;
moveFolder: (folderId: string, targetParentId: string | null) => Promise<void>;
toggleFolderExpansion: (folderId: string) => void;
setFolderExpansion: (folderId: string, expanded: boolean) => void;
findFolderInTree: (folderId: string) => FolderTreeNode | null;
findPathToFolder: (folderId: string) => FolderTreeNode[];
filterTreeBySearch: (query: string) => FolderTreeNode[];
}
/**
* composable
*/
export function useFolderManager(options: UseFolderManagerOptions): UseFolderManagerReturn {
const { operations, rootFolderName = '根目录', autoLoad = false } = options;
// 状态
const folderTree = ref<FolderTreeNode[]>([]);
const currentFolderId = ref<string | null>(null);
const currentFolders = ref<Folder[]>([]);
const breadcrumbPath = ref<FolderTreeNode[]>([]);
const expandedFolderIds = ref<string[]>([]);
const loading = ref(false);
const treeLoading = ref(false);
// 计算属性
const currentFolderName = computed(() => {
if (breadcrumbPath.value.length === 0) {
return rootFolderName;
}
return breadcrumbPath.value[breadcrumbPath.value.length - 1]?.name || rootFolderName;
});
const breadcrumbItems = computed((): BreadcrumbItem[] => {
const items: BreadcrumbItem[] = [
{
title: rootFolderName,
folderId: null,
disabled: currentFolderId.value === null,
isRoot: true,
},
];
breadcrumbPath.value.forEach((folder, index) => {
items.push({
title: folder.name,
folderId: folder.folder_id,
disabled: index === breadcrumbPath.value.length - 1,
isRoot: false,
});
});
return items;
});
// 内部方法
const findPathToFolderInternal = (
nodes: FolderTreeNode[],
targetId: string,
path: FolderTreeNode[] = []
): FolderTreeNode[] | null => {
for (const node of nodes) {
if (node.folder_id === targetId) {
return [...path, node];
}
if (node.children && node.children.length > 0) {
const result = findPathToFolderInternal(node.children, targetId, [...path, node]);
if (result) return result;
}
}
return null;
};
const updateBreadcrumb = (folderId: string | null): void => {
if (folderId === null) {
breadcrumbPath.value = [];
return;
}
const path = findPathToFolderInternal(folderTree.value, folderId);
breadcrumbPath.value = path || [];
};
// 公开方法
const loadFolderTree = async (): Promise<void> => {
treeLoading.value = true;
try {
folderTree.value = await operations.loadFolderTree();
} finally {
treeLoading.value = false;
}
};
const navigateToFolder = async (folderId: string | null): Promise<void> => {
loading.value = true;
try {
currentFolderId.value = folderId;
currentFolders.value = await operations.loadSubFolders(folderId);
updateBreadcrumb(folderId);
} finally {
loading.value = false;
}
};
const refreshCurrentFolder = async (): Promise<void> => {
await navigateToFolder(currentFolderId.value);
};
const createFolder = async (data: CreateFolderData): Promise<Folder> => {
const folder = await operations.createFolder({
...data,
parent_id: data.parent_id ?? currentFolderId.value,
});
await Promise.all([refreshCurrentFolder(), loadFolderTree()]);
return folder;
};
const updateFolder = async (data: UpdateFolderData): Promise<void> => {
await operations.updateFolder(data);
await Promise.all([refreshCurrentFolder(), loadFolderTree()]);
};
const deleteFolder = async (folderId: string): Promise<void> => {
await operations.deleteFolder(folderId);
await Promise.all([refreshCurrentFolder(), loadFolderTree()]);
};
const moveFolder = async (folderId: string, targetParentId: string | null): Promise<void> => {
if (operations.moveFolder) {
await operations.moveFolder(folderId, targetParentId);
} else {
// 如果没有专门的移动方法,使用更新方法
await operations.updateFolder({
folder_id: folderId,
parent_id: targetParentId,
});
}
await Promise.all([refreshCurrentFolder(), loadFolderTree()]);
};
const toggleFolderExpansion = (folderId: string): void => {
const index = expandedFolderIds.value.indexOf(folderId);
if (index === -1) {
expandedFolderIds.value.push(folderId);
} else {
expandedFolderIds.value.splice(index, 1);
}
};
const setFolderExpansion = (folderId: string, expanded: boolean): void => {
const index = expandedFolderIds.value.indexOf(folderId);
if (expanded && index === -1) {
expandedFolderIds.value.push(folderId);
} else if (!expanded && index !== -1) {
expandedFolderIds.value.splice(index, 1);
}
};
const findFolderInTree = (folderId: string): FolderTreeNode | null => {
const findNode = (nodes: FolderTreeNode[]): FolderTreeNode | null => {
for (const node of nodes) {
if (node.folder_id === folderId) {
return node;
}
if (node.children && node.children.length > 0) {
const found = findNode(node.children);
if (found) return found;
}
}
return null;
};
return findNode(folderTree.value);
};
const findPathToFolder = (folderId: string): FolderTreeNode[] => {
return findPathToFolderInternal(folderTree.value, folderId) || [];
};
const filterTreeBySearch = (query: string): FolderTreeNode[] => {
if (!query) return folderTree.value;
const lowerQuery = query.toLowerCase();
const filterNodes = (nodes: FolderTreeNode[]): FolderTreeNode[] => {
return nodes
.filter((node) => {
const matches = node.name.toLowerCase().includes(lowerQuery);
const childMatches = filterNodes(node.children || []);
return matches || childMatches.length > 0;
})
.map((node) => ({
...node,
children: filterNodes(node.children || []),
}));
};
return filterNodes(folderTree.value);
};
// 自动加载
if (autoLoad) {
loadFolderTree();
navigateToFolder(null);
}
return {
// 状态
folderTree,
currentFolderId,
currentFolders,
breadcrumbPath,
expandedFolderIds,
loading,
treeLoading,
// 计算属性
currentFolderName,
breadcrumbItems,
// 方法
loadFolderTree,
navigateToFolder,
refreshCurrentFolder,
createFolder,
updateFolder,
deleteFolder,
moveFolder,
toggleFolderExpansion,
setFolderExpansion,
findFolderInTree,
findPathToFolder,
filterTreeBySearch,
};
}
/**
* ID
*
*/
export function collectFolderAndChildrenIds(
folderTree: FolderTreeNode[],
folderId: string
): string[] {
const ids: string[] = [folderId];
const collectChildIds = (nodes: FolderTreeNode[]): boolean => {
for (const node of nodes) {
if (node.folder_id === folderId) {
const collectAllChildren = (children: FolderTreeNode[]) => {
for (const child of children) {
ids.push(child.folder_id);
if (child.children) {
collectAllChildren(child.children);
}
}
};
if (node.children) {
collectAllChildren(node.children);
}
return true;
}
if (node.children && collectChildIds(node.children)) {
return true;
}
}
return false;
};
collectChildIds(folderTree);
return ids;
}
export default useFolderManager;
@@ -1,11 +1,23 @@
<template>
<v-dialog v-model="showDialog" max-width="500px" persistent>
<v-dialog v-model="showDialog" max-width="500px">
<v-card>
<v-card-title class="text-h2">
{{ editingPersona ? tm('dialog.edit.title') : tm('dialog.create.title') }}
</v-card-title>
<v-card-text>
<!-- 创建位置提示 -->
<v-alert
v-if="!editingPersona"
type="info"
variant="tonal"
density="compact"
class="mb-4"
icon="mdi-folder-outline"
>
{{ tm('form.createInFolder', { folder: folderDisplayName }) }}
</v-alert>
<v-form ref="personaForm" v-model="formValid">
<v-text-field v-model="personaForm.persona_id" :label="tm('form.personaId')"
:rules="personaIdRules" :disabled="editingPersona" variant="outlined" density="comfortable"
@@ -209,6 +221,14 @@ export default {
editingPersona: {
type: Object,
default: null
},
currentFolderId: {
type: String,
default: null
},
currentFolderName: {
type: String,
default: null
}
},
emits: ['update:modelValue', 'saved', 'error'],
@@ -225,15 +245,18 @@ export default {
mcpServers: [],
availableTools: [],
loadingTools: false,
existingPersonaIds: [], // ID
personaForm: {
persona_id: '',
system_prompt: '',
begin_dialogs: [],
tools: []
tools: [],
folder_id: null
},
personaIdRules: [
v => !!v || this.tm('validation.required'),
v => (v && v.length >= 0) || this.tm('validation.minLength', { min: 2 }),
v => (v && v.length >= 1) || this.tm('validation.minLength', { min: 1 }),
v => !this.existingPersonaIds.includes(v) || this.tm('validation.personaIdExists'),
],
systemPromptRules: [
v => !!v || this.tm('validation.required'),
@@ -262,6 +285,18 @@ export default {
(tool.description && tool.description.toLowerCase().includes(search)) ||
(tool.mcp_server_name && tool.mcp_server_name.toLowerCase().includes(search))
);
},
folderDisplayName() {
// 使
if (this.currentFolderName) {
return this.currentFolderName;
}
// ID
if (!this.currentFolderId) {
return this.tm('form.rootFolder');
}
// ID
return this.currentFolderId;
}
},
@@ -273,6 +308,8 @@ export default {
this.initFormWithPersona(this.editingPersona);
} else {
this.initForm();
//
this.loadExistingPersonaIds();
}
this.loadMcpServers();
this.loadTools();
@@ -310,7 +347,8 @@ export default {
persona_id: '',
system_prompt: '',
begin_dialogs: [],
tools: []
tools: [],
folder_id: this.currentFolderId
};
this.toolSelectValue = '0';
this.expandedPanels = [];
@@ -321,7 +359,8 @@ export default {
persona_id: persona.persona_id,
system_prompt: persona.system_prompt,
begin_dialogs: [...(persona.begin_dialogs || [])],
tools: persona.tools === null ? null : [...(persona.tools || [])]
tools: persona.tools === null ? null : [...(persona.tools || [])],
folder_id: persona.folder_id
};
// tools toolSelectValue
this.toolSelectValue = persona.tools === null ? '0' : '1';
@@ -363,6 +402,18 @@ export default {
}
},
async loadExistingPersonaIds() {
try {
const response = await axios.get('/api/persona/list');
if (response.data.status === 'ok') {
this.existingPersonaIds = (response.data.data || []).map(p => p.persona_id);
}
} catch (error) {
// 使
this.existingPersonaIds = [];
}
},
async savePersona() {
if (!this.formValid) return;
@@ -1,84 +1,46 @@
<template>
<div class="d-flex align-center justify-space-between">
<span v-if="!modelValue" style="color: rgb(var(--v-theme-primaryText));">
{{ tm('personaSelector.notSelected') }}
</span>
<span v-else>
{{ modelValue === 'default' ? tm('personaSelector.defaultPersona') : modelValue }}
</span>
<v-btn size="small" color="primary" variant="tonal" @click="openDialog">
{{ buttonText || tm('personaSelector.buttonText') }}
</v-btn>
</div>
<!-- Persona Selection Dialog -->
<v-dialog v-model="dialog" max-width="600px">
<v-card>
<v-card-title class="text-h3 py-4" style="font-weight: normal;">
{{ tm('personaSelector.dialogTitle') }}
</v-card-title>
<v-card-text class="pa-2" style="max-height: 400px; overflow-y: auto;">
<v-progress-linear v-if="loading" indeterminate color="primary"></v-progress-linear>
<v-list v-if="!loading && personaList.length > 0" density="compact">
<v-list-item
v-for="persona in personaList"
:key="persona.persona_id"
:value="persona.persona_id"
@click="selectPersona(persona)"
:active="selectedPersona === persona.persona_id"
rounded="md"
class="ma-1">
<v-list-item-title>{{ persona.persona_id === 'default' ? tm('personaSelector.defaultPersona') : persona.persona_id }}</v-list-item-title>
<v-list-item-subtitle>
{{ persona.system_prompt ? persona.system_prompt.substring(0, 50) + '...' : tm('personaSelector.noDescription') }}
</v-list-item-subtitle>
<template v-slot:append>
<v-icon v-if="selectedPersona === persona.persona_id" color="primary">mdi-check-circle</v-icon>
</template>
</v-list-item>
</v-list>
<div v-else-if="!loading && personaList.length === 0" class="text-center py-8">
<v-icon size="64" color="grey-lighten-1">mdi-account-off</v-icon>
<p class="text-grey mt-4">{{ tm('personaSelector.noPersonas') }}</p>
</div>
</v-card-text>
<v-card-actions class="pa-4">
<v-btn variant="text" color="primary" prepend-icon="mdi-plus" @click="openCreatePersona">
{{ tm('personaSelector.createPersona') }}
</v-btn>
<v-spacer></v-spacer>
<v-btn variant="text" @click="cancelSelection">{{ t('core.common.cancel') }}</v-btn>
<v-btn
color="primary"
@click="confirmSelection"
:disabled="!selectedPersona">
{{ t('core.common.confirm') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
<BaseFolderItemSelector
:model-value="modelValue"
@update:model-value="handleUpdate"
:folder-tree="folderTree"
:items="currentPersonas as any"
:tree-loading="treeLoading"
:items-loading="itemsLoading"
:labels="labels"
:show-create-button="true"
:default-item="defaultPersona"
item-id-field="persona_id"
item-name-field="persona_id"
item-description-field="system_prompt"
:display-value-formatter="formatDisplayValue"
@navigate="handleNavigate"
@create="openCreatePersona"
/>
<!-- 创建人格对话框 -->
<PersonaForm
<PersonaForm
v-model="showCreateDialog"
:editing-persona="null"
:mcp-servers="mcpServers"
:available-tools="availableTools"
:loading-tools="loadingTools"
:editing-persona="undefined"
:current-folder-id="currentFolderId ?? undefined"
:current-folder-name="currentFolderName ?? undefined"
@saved="handlePersonaCreated"
@error="handleError" />
</template>
<script setup>
import { ref, watch } from 'vue'
<script setup lang="ts">
import { ref, computed, onMounted } from 'vue'
import axios from 'axios'
import BaseFolderItemSelector from '@/components/folder/BaseFolderItemSelector.vue'
import PersonaForm from './PersonaForm.vue'
import { useI18n, useModuleI18n } from '@/i18n/composables'
import type { FolderTreeNode, SelectableItem } from '@/components/folder/types'
interface Persona {
persona_id: string
system_prompt: string
folder_id?: string | null
[key: string]: any
}
const props = defineProps({
modelValue: {
@@ -95,91 +57,142 @@ const emit = defineEmits(['update:modelValue'])
const { t } = useI18n()
const { tm } = useModuleI18n('core.shared')
const dialog = ref(false)
const personaList = ref([])
const loading = ref(false)
const selectedPersona = ref('')
//
const folderTree = ref<FolderTreeNode[]>([])
const currentPersonas = ref<Persona[]>([])
const treeLoading = ref(false)
const itemsLoading = ref(false)
const showCreateDialog = ref(false)
const currentFolderId = ref<string | null>(null)
// modelValue selectedPersona
watch(() => props.modelValue, (newValue) => {
selectedPersona.value = newValue || ''
}, { immediate: true })
async function openDialog() {
selectedPersona.value = props.modelValue || ''
dialog.value = true
await loadPersonas()
//
const defaultPersona: SelectableItem = {
id: 'default',
persona_id: 'default',
name: tm('personaSelector.defaultPersona'),
system_prompt: 'You are a helpful and friendly assistant.'
}
async function loadPersonas() {
loading.value = true
//
function findFolderName(nodes: FolderTreeNode[], folderId: string): string | null {
for (const node of nodes) {
if (node.folder_id === folderId) {
return node.name
}
if (node.children && node.children.length > 0) {
const found = findFolderName(node.children, folderId)
if (found) return found
}
}
return null
}
//
const currentFolderName = computed(() => {
if (!currentFolderId.value) {
return null // PersonaForm 使 tm('form.rootFolder')
}
return findFolderName(folderTree.value, currentFolderId.value)
})
//
const labels = computed(() => ({
dialogTitle: tm('personaSelector.dialogTitle'),
notSelected: tm('personaSelector.notSelected'),
buttonText: props.buttonText || tm('personaSelector.buttonText'),
noItems: tm('personaSelector.noPersonas'),
defaultItem: tm('personaSelector.defaultPersona'),
noDescription: tm('personaSelector.noDescription'),
createButton: tm('personaSelector.createPersona'),
confirmButton: t('core.common.confirm'),
cancelButton: t('core.common.cancel'),
rootFolder: tm('personaSelector.rootFolder') || '全部人格',
emptyFolder: tm('personaSelector.emptyFolder') || '此文件夹为空'
}))
//
function formatDisplayValue(value: string): string {
if (value === 'default') {
return tm('personaSelector.defaultPersona')
}
return value
}
//
function handleUpdate(value: string) {
emit('update:modelValue', value)
}
//
async function loadFolderTree() {
treeLoading.value = true
try {
const response = await axios.get('/api/persona/list')
const response = await axios.get('/api/persona/folder/tree')
if (response.data.status === 'ok') {
const personas = response.data.data || []
//
personaList.value = [
{
persona_id: 'default',
system_prompt: 'You are a helpful and friendly assistant.'
},
...personas
]
folderTree.value = response.data.data || []
}
} catch (error) {
console.error('加载人格列表失败:', error)
personaList.value = [
{
persona_id: 'default',
system_prompt: 'You are a helpful and friendly assistant.'
}
]
console.error('加载文件夹树失败:', error)
folderTree.value = []
} finally {
loading.value = false
treeLoading.value = false
}
}
function selectPersona(persona) {
selectedPersona.value = persona.persona_id
//
async function loadPersonasInFolder(folderId: string | null) {
itemsLoading.value = true
try {
// 使 /api/persona/list folder_id
const params = new URLSearchParams()
if (folderId !== null) {
params.set('folder_id', folderId)
} else {
// folder_id
params.set('folder_id', '')
}
const response = await axios.get(`/api/persona/list?${params.toString()}`)
if (response.data.status === 'ok') {
currentPersonas.value = response.data.data || []
}
} catch (error) {
console.error('加载人格列表失败:', error)
currentPersonas.value = []
} finally {
itemsLoading.value = false
}
}
function confirmSelection() {
emit('update:modelValue', selectedPersona.value)
dialog.value = false
}
function cancelSelection() {
selectedPersona.value = props.modelValue || ''
dialog.value = false
//
async function handleNavigate(folderId: string | null) {
currentFolderId.value = folderId
await loadPersonasInFolder(folderId)
}
//
function openCreatePersona() {
showCreateDialog.value = true
}
async function handlePersonaCreated(message) {
//
async function handlePersonaCreated(message: string) {
console.log('人格创建成功:', message)
showCreateDialog.value = false
//
await loadPersonas()
//
await loadPersonasInFolder(currentFolderId.value)
}
function handleError(error) {
//
function handleError(error: string) {
console.error('创建人格失败:', error)
}
//
onMounted(() => {
loadFolderTree()
})
</script>
<style scoped>
.v-list-item {
transition: all 0.2s ease;
}
.v-list-item:hover {
background-color: rgba(var(--v-theme-primary), 0.04);
}
.v-list-item.v-list-item--active {
background-color: rgba(var(--v-theme-primary), 0.08);
}
/* 样式继承自 BaseFolderItemSelector */
</style>
@@ -57,7 +57,9 @@
"createPersona": "Create New Persona",
"cancelSelection": "Cancel",
"confirmSelection": "Confirm Selection",
"selectPersonaPool": "Select Persona Pool..."
"selectPersonaPool": "Select Persona Pool...",
"rootFolder": "All Personas",
"emptyFolder": "This folder is empty"
},
"t2iTemplateEditor": {
"buttonText": "Customize T2I Template",
@@ -9,6 +9,7 @@
"delete": "Delete",
"cancel": "Cancel",
"save": "Save",
"move": "Move",
"addDialogPair": "Add Dialog Pair"
},
"labels": {
@@ -36,7 +37,9 @@
"noToolsFound": "No matching tools found",
"loadingTools": "Loading tools...",
"allToolsAvailable": "Use all available tools",
"noToolsSelected": "No tools selected"
"noToolsSelected": "No tools selected",
"createInFolder": "Will be created in \"{folder}\"",
"rootFolder": "All Personas"
},
"dialog": {
"create": {
@@ -48,13 +51,16 @@
},
"empty": {
"title": "No Persona Configured",
"description": "Create your first persona to start using personalized chatbots"
"description": "Create your first persona to start using personalized chatbots",
"folderEmpty": "This folder is empty",
"folderEmptyDescription": "Create a new persona or folder to get started"
},
"validation": {
"required": "This field is required",
"minLength": "Minimum {min} characters required",
"alphanumeric": "Only letters, numbers, underscores and hyphens are allowed",
"dialogRequired": "{type} cannot be empty"
"dialogRequired": "{type} cannot be empty",
"personaIdExists": "This persona name already exists"
},
"messages": {
"loadError": "Failed to load persona list",
@@ -63,5 +69,63 @@
"deleteConfirm": "Are you sure you want to delete persona \"{id}\"? This action cannot be undone.",
"deleteSuccess": "Deleted successfully",
"deleteError": "Delete failed"
},
"persona": {
"personasTitle": "Personas",
"toolsCount": "tools",
"contextMenu": {
"moveTo": "Move to..."
},
"messages": {
"moveSuccess": "Persona moved successfully",
"moveError": "Failed to move persona"
}
},
"folder": {
"sidebarTitle": "Folders",
"rootFolder": "Root",
"foldersTitle": "Folders",
"noFolders": "No folders yet",
"createButton": "New Folder",
"searchPlaceholder": "Search folders...",
"form": {
"name": "Folder Name",
"description": "Description (optional)"
},
"validation": {
"nameRequired": "Folder name is required"
},
"contextMenu": {
"open": "Open",
"rename": "Rename",
"moveTo": "Move to...",
"delete": "Delete"
},
"createDialog": {
"title": "Create New Folder",
"createButton": "Create"
},
"renameDialog": {
"title": "Rename Folder"
},
"deleteDialog": {
"title": "Delete Folder",
"message": "Are you sure you want to delete folder \"{name}\"?",
"warning": "All personas inside will be moved to root folder."
},
"messages": {
"createSuccess": "Folder created successfully",
"createError": "Failed to create folder",
"renameSuccess": "Folder renamed successfully",
"renameError": "Failed to rename folder",
"deleteSuccess": "Folder deleted successfully",
"deleteError": "Failed to delete folder"
}
},
"moveDialog": {
"title": "Move to Folder",
"description": "Select a destination folder for \"{name}\"",
"success": "Moved successfully",
"error": "Failed to move"
}
}
@@ -57,7 +57,9 @@
"createPersona": "创建新人格",
"cancelSelection": "取消",
"confirmSelection": "确认选择",
"selectPersonaPool": "选择人格池..."
"selectPersonaPool": "选择人格池...",
"rootFolder": "全部人格",
"emptyFolder": "此文件夹为空"
},
"t2iTemplateEditor": {
"buttonText": "自定义 T2I 模板",
@@ -9,6 +9,7 @@
"delete": "删除",
"cancel": "取消",
"save": "保存",
"move": "移动",
"addDialogPair": "添加对话对"
},
"labels": {
@@ -36,7 +37,9 @@
"noToolsFound": "未找到匹配的工具",
"loadingTools": "正在加载工具...",
"allToolsAvailable": "使用所有可用工具",
"noToolsSelected": "未选择任何工具"
"noToolsSelected": "未选择任何工具",
"createInFolder": "将在「{folder}」中创建",
"rootFolder": "全部人格"
},
"dialog": {
"create": {
@@ -48,13 +51,16 @@
},
"empty": {
"title": "暂无人格配置",
"description": "来创建一个吧!"
"description": "来创建一个吧!",
"folderEmpty": "此文件夹为空",
"folderEmptyDescription": "创建新的人格或文件夹开始使用"
},
"validation": {
"required": "此字段为必填项",
"minLength": "最少需要 {min} 个字符",
"alphanumeric": "只能包含字母、数字、下划线和连字符",
"dialogRequired": "{type}不能为空"
"dialogRequired": "{type}不能为空",
"personaIdExists": "该人格名称已存在"
},
"messages": {
"loadError": "加载人格列表失败",
@@ -63,5 +69,63 @@
"deleteConfirm": "确定要删除人格 \"{id}\" 吗?此操作不可撤销。",
"deleteSuccess": "删除成功",
"deleteError": "删除失败"
},
"persona": {
"personasTitle": "人格",
"toolsCount": "个工具",
"contextMenu": {
"moveTo": "移动到..."
},
"messages": {
"moveSuccess": "人格移动成功",
"moveError": "移动人格失败"
}
},
"folder": {
"sidebarTitle": "文件夹",
"rootFolder": "根目录",
"foldersTitle": "文件夹",
"noFolders": "暂无文件夹",
"createButton": "新建文件夹",
"searchPlaceholder": "搜索文件夹...",
"form": {
"name": "文件夹名称",
"description": "描述(可选)"
},
"validation": {
"nameRequired": "文件夹名称不能为空"
},
"contextMenu": {
"open": "打开",
"rename": "重命名",
"moveTo": "移动到...",
"delete": "删除"
},
"createDialog": {
"title": "创建新文件夹",
"createButton": "创建"
},
"renameDialog": {
"title": "重命名文件夹"
},
"deleteDialog": {
"title": "删除文件夹",
"message": "确定要删除文件夹 \"{name}\" 吗?",
"warning": "文件夹内的所有人格将被移动到根目录。"
},
"messages": {
"createSuccess": "文件夹创建成功",
"createError": "创建文件夹失败",
"renameSuccess": "文件夹重命名成功",
"renameError": "重命名文件夹失败",
"deleteSuccess": "文件夹删除成功",
"deleteError": "删除文件夹失败"
}
},
"moveDialog": {
"title": "移动到文件夹",
"description": "为 \"{name}\" 选择目标文件夹",
"success": "移动成功",
"error": "移动失败"
}
}
+333
View File
@@ -0,0 +1,333 @@
/**
* Persona Store
*/
import { defineStore } from 'pinia';
import axios from 'axios';
// 类型定义
export interface PersonaFolder {
folder_id: string;
name: string;
parent_id: string | null;
description: string | null;
sort_order: number;
created_at: string;
updated_at: string;
}
export interface Persona {
persona_id: string;
system_prompt: string;
begin_dialogs: string[];
tools: string[] | null;
folder_id: string | null;
sort_order: number;
created_at: string;
updated_at: string;
}
export interface FolderTreeNode {
folder_id: string;
name: string;
parent_id: string | null;
description: string | null;
sort_order: number;
children: FolderTreeNode[];
}
export interface ReorderItem {
id: string;
type: 'persona' | 'folder';
sort_order: number;
}
export const usePersonaStore = defineStore({
id: 'persona',
state: () => ({
folderTree: [] as FolderTreeNode[],
currentFolderId: null as string | null,
currentFolders: [] as PersonaFolder[],
currentPersonas: [] as Persona[],
breadcrumbPath: [] as FolderTreeNode[],
expandedFolderIds: [] as string[], // Store expanded folder IDs
loading: false,
treeLoading: false,
}),
getters: {
// 当前文件夹名称
currentFolderName(): string {
if (this.breadcrumbPath.length === 0) {
return '根目录';
}
return this.breadcrumbPath[this.breadcrumbPath.length - 1]?.name || '根目录';
},
},
actions: {
/**
* Toggle folder expansion state
*/
toggleFolderExpansion(folderId: string) {
const index = this.expandedFolderIds.indexOf(folderId);
if (index === -1) {
this.expandedFolderIds.push(folderId);
} else {
this.expandedFolderIds.splice(index, 1);
}
},
/**
* Set folder expansion state
*/
setFolderExpansion(folderId: string, expanded: boolean) {
const index = this.expandedFolderIds.indexOf(folderId);
if (expanded && index === -1) {
this.expandedFolderIds.push(folderId);
} else if (!expanded && index !== -1) {
this.expandedFolderIds.splice(index, 1);
}
},
/**
*
*/
async loadFolderTree(): Promise<void> {
this.treeLoading = true;
try {
const response = await axios.get('/api/persona/folder/tree');
if (response.data.status === 'ok') {
this.folderTree = response.data.data || [];
} else {
throw new Error(response.data.message || '获取文件夹树失败');
}
} finally {
this.treeLoading = false;
}
},
/**
*
*/
async navigateToFolder(folderId: string | null): Promise<void> {
this.loading = true;
try {
this.currentFolderId = folderId;
// 并行加载子文件夹和 Persona
const [foldersRes, personasRes] = await Promise.all([
axios.get('/api/persona/folder/list', {
params: { parent_id: folderId ?? '' }
}),
axios.get('/api/persona/list', {
params: { folder_id: folderId ?? '' }
}),
]);
if (foldersRes.data.status === 'ok') {
this.currentFolders = foldersRes.data.data || [];
}
if (personasRes.data.status === 'ok') {
this.currentPersonas = personasRes.data.data || [];
}
// 更新面包屑
this.updateBreadcrumb(folderId);
} finally {
this.loading = false;
}
},
/**
*
*/
updateBreadcrumb(folderId: string | null): void {
if (folderId === null) {
this.breadcrumbPath = [];
return;
}
// 从树中查找路径
const path: FolderTreeNode[] = [];
const findPath = (nodes: FolderTreeNode[], targetId: string): boolean => {
for (const node of nodes) {
if (node.folder_id === targetId) {
path.push(node);
return true;
}
if (node.children.length > 0 && findPath(node.children, targetId)) {
path.unshift(node);
return true;
}
}
return false;
};
findPath(this.folderTree, folderId);
this.breadcrumbPath = path;
},
/**
*
*/
async refreshCurrentFolder(): Promise<void> {
await this.navigateToFolder(this.currentFolderId);
},
/**
* Persona
*/
async movePersonaToFolder(personaId: string, targetFolderId: string | null): Promise<void> {
const response = await axios.post('/api/persona/move', {
persona_id: personaId,
folder_id: targetFolderId
});
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '移动人格失败');
}
// 刷新当前文件夹内容和文件夹树
await Promise.all([
this.refreshCurrentFolder(),
this.loadFolderTree(),
]);
},
/**
*
*/
async moveFolderToFolder(folderId: string, targetParentId: string | null): Promise<void> {
const response = await axios.post('/api/persona/folder/update', {
folder_id: folderId,
parent_id: targetParentId
});
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '移动文件夹失败');
}
// 刷新当前文件夹内容和文件夹树
await Promise.all([
this.refreshCurrentFolder(),
this.loadFolderTree(),
]);
},
/**
*
*/
async createFolder(data: {
name: string;
parent_id?: string | null;
description?: string;
}): Promise<PersonaFolder> {
const response = await axios.post('/api/persona/folder/create', {
...data,
parent_id: data.parent_id ?? this.currentFolderId,
});
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '创建文件夹失败');
}
// 刷新当前文件夹内容和文件夹树
await Promise.all([
this.refreshCurrentFolder(),
this.loadFolderTree(),
]);
return response.data.data.folder;
},
/**
*
*/
async updateFolder(data: {
folder_id: string;
name?: string;
description?: string;
}): Promise<void> {
const response = await axios.post('/api/persona/folder/update', data);
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '更新文件夹失败');
}
// 刷新当前文件夹内容和文件夹树
await Promise.all([
this.refreshCurrentFolder(),
this.loadFolderTree(),
]);
},
/**
*
*/
async deleteFolder(folderId: string): Promise<void> {
const response = await axios.post('/api/persona/folder/delete', {
folder_id: folderId
});
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '删除文件夹失败');
}
// 刷新当前文件夹内容和文件夹树
await Promise.all([
this.refreshCurrentFolder(),
this.loadFolderTree(),
]);
},
/**
* Persona
*/
async deletePersona(personaId: string): Promise<void> {
const response = await axios.post('/api/persona/delete', {
persona_id: personaId
});
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '删除人格失败');
}
// 刷新当前文件夹内容
await this.refreshCurrentFolder();
},
/**
*
*/
async reorderItems(items: ReorderItem[]): Promise<void> {
const response = await axios.post('/api/persona/reorder', { items });
if (response.data.status !== 'ok') {
throw new Error(response.data.message || '更新排序失败');
}
// 刷新当前文件夹内容
await this.refreshCurrentFolder();
},
/**
* ID
*/
findFolderInTree(folderId: string): FolderTreeNode | null {
const findNode = (nodes: FolderTreeNode[]): FolderTreeNode | null => {
for (const node of nodes) {
if (node.folder_id === folderId) {
return node;
}
if (node.children.length > 0) {
const found = findNode(node.children);
if (found) return found;
}
}
return null;
};
return findNode(this.folderTree);
},
}
});
+7 -285
View File
@@ -2,277 +2,38 @@
<div class="persona-page">
<v-container fluid class="pa-0">
<!-- 页面标题 -->
<v-row class="d-flex justify-space-between align-center px-4 py-3 pb-8">
<v-row class="d-flex justify-space-between align-center px-4 py-3 pb-6">
<div>
<h1 class="text-h1 font-weight-bold mb-2">
<v-icon color="black" class="me-2">mdi-heart</v-icon>{{ t('core.navigation.persona') }}
</h1>
<p class="text-subtitle-1 text-medium-emphasis mb-4">
<p class="text-subtitle-1 text-medium-emphasis mb-0">
{{ tm('page.description') }}
</p>
</div>
<div>
<v-btn color="primary" variant="tonal" prepend-icon="mdi-plus" @click="openCreateDialog"
rounded="xl" size="x-large">
{{ tm('buttons.create') }}
</v-btn>
</div>
</v-row>
<!-- 人格卡片网格 -->
<v-row>
<v-col v-for="persona in personas" :key="persona.persona_id" cols="12" md="6" lg="4" xl="3">
<v-card class="persona-card" rounded="md" @click="viewPersona(persona)">
<v-card-title class="d-flex justify-space-between align-center">
<div class="text-truncate ml-2">
{{ persona.persona_id }}
</div>
<v-menu offset-y>
<template v-slot:activator="{ props }">
<v-btn icon="mdi-dots-vertical" variant="text" size="small" v-bind="props"
@click.stop />
</template>
<v-list density="compact">
<v-list-item @click="editPersona(persona)">
<v-list-item-title>
<v-icon class="mr-2" size="small">mdi-pencil</v-icon>
{{ tm('buttons.edit') }}
</v-list-item-title>
</v-list-item>
<v-list-item @click="deletePersona(persona)" class="text-error">
<v-list-item-title>
<v-icon class="mr-2" size="small">mdi-delete</v-icon>
{{ tm('buttons.delete') }}
</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
</v-card-title>
<v-card-text>
<div class="system-prompt-preview">
{{ truncateText(persona.system_prompt, 100) }}
</div>
<div class="mt-3" v-if="persona.begin_dialogs && persona.begin_dialogs.length > 0">
<v-chip size="small" color="secondary" variant="tonal" prepend-icon="mdi-chat">
{{ tm('labels.presetDialogs', { count: persona.begin_dialogs.length / 2 }) }}
</v-chip>
</div>
<div class="mt-3 text-caption text-medium-emphasis">
{{ tm('labels.createdAt') }}: {{ formatDate(persona.created_at) }}
</div>
</v-card-text>
</v-card>
</v-col>
<!-- 空状态 -->
<v-col v-if="personas.length === 0 && !loading" cols="12">
<v-card class="text-center pa-8" elevation="0">
<v-icon size="64" color="grey-lighten-1" class="mb-4">mdi-account-group</v-icon>
<h3 class="text-h5 mb-2">{{ tm('empty.title') }}</h3>
<p class="text-body-1 text-medium-emphasis mb-4">{{ tm('empty.description') }}</p>
<v-btn color="primary" variant="tonal" prepend-icon="mdi-plus" @click="openCreateDialog">
{{ tm('buttons.createFirst') }}
</v-btn>
</v-card>
</v-col>
</v-row>
<!-- 加载状态 -->
<v-row v-if="loading">
<v-col v-for="n in 6" :key="n" cols="12" md="6" lg="4" xl="3">
<v-skeleton-loader type="card" rounded="lg"></v-skeleton-loader>
</v-col>
</v-row>
<!-- 主容器组件 -->
<PersonaManager />
</v-container>
<!-- 创建/编辑人格对话框 -->
<PersonaForm
v-model="showPersonaDialog"
:editing-persona="editingPersona"
@saved="handlePersonaSaved"
@error="showError" />
<!-- 查看人格详情对话框 -->
<v-dialog v-model="showViewDialog" max-width="700px">
<v-card v-if="viewingPersona">
<v-card-title class="d-flex justify-space-between align-center">
<span class="text-h5">{{ viewingPersona.persona_id }}</span>
<v-btn icon="mdi-close" variant="text" @click="showViewDialog = false" />
</v-card-title>
<v-card-text>
<div class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.systemPrompt') }}</h4>
<pre class="system-prompt-content">
{{ viewingPersona.system_prompt }}
</pre>
</div>
<div v-if="viewingPersona.begin_dialogs && viewingPersona.begin_dialogs.length > 0" class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.presetDialogs') }}</h4>
<div v-for="(dialog, index) in viewingPersona.begin_dialogs" :key="index" class="mb-2">
<v-chip :color="index % 2 === 0 ? 'primary' : 'secondary'" variant="tonal" size="small"
class="mb-1">
{{ index % 2 === 0 ? tm('form.userMessage') : tm('form.assistantMessage') }}
</v-chip>
<div class="dialog-content ml-2">
{{ dialog }}
</div>
</div>
</div>
<div class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.tools') }}</h4>
<div v-if="viewingPersona.tools === null" class="text-body-2 text-medium-emphasis">
<v-chip size="small" color="success" variant="tonal" prepend-icon="mdi-check-all">
{{ tm('form.allToolsAvailable') }}
</v-chip>
</div>
<div v-else-if="viewingPersona.tools && viewingPersona.tools.length > 0"
class="d-flex flex-wrap ga-1">
<v-chip v-for="toolName in viewingPersona.tools" :key="toolName" size="small"
color="primary" variant="tonal">
{{ toolName }}
</v-chip>
</div>
<div v-else class="text-body-2 text-medium-emphasis">
{{ tm('form.noToolsSelected') }}
</div>
</div>
<div class="text-caption text-medium-emphasis">
<div>{{ tm('labels.createdAt') }}: {{ formatDate(viewingPersona.created_at) }}</div>
<div v-if="viewingPersona.updated_at">{{ tm('labels.updatedAt') }}: {{
formatDate(viewingPersona.updated_at) }}</div>
</div>
</v-card-text>
</v-card>
</v-dialog>
<!-- 消息提示 -->
<v-snackbar :timeout="3000" elevation="24" :color="messageType" v-model="showMessage" location="top">
{{ message }}
</v-snackbar>
</div>
</template>
<script>
import axios from 'axios';
import { useI18n, useModuleI18n } from '@/i18n/composables';
import PersonaForm from '@/components/shared/PersonaForm.vue';
import { PersonaManager } from '@/views/persona';
export default {
name: 'PersonaPage',
components: {
PersonaForm
PersonaManager
},
setup() {
const { t } = useI18n();
const { tm } = useModuleI18n('features/persona');
return { t, tm };
},
data() {
return {
personas: [],
loading: false,
showPersonaDialog: false,
showViewDialog: false,
editingPersona: null,
viewingPersona: null,
showMessage: false,
message: '',
messageType: 'success'
}
},
mounted() {
this.loadPersonas();
},
methods: {
async loadPersonas() {
this.loading = true;
try {
const response = await axios.get('/api/persona/list');
if (response.data.status === 'ok') {
this.personas = response.data.data;
} else {
this.showError(response.data.message || this.tm('messages.loadError'));
}
} catch (error) {
this.showError(error.response?.data?.message || this.tm('messages.loadError'));
}
this.loading = false;
},
openCreateDialog() {
this.editingPersona = null;
this.showPersonaDialog = true;
},
editPersona(persona) {
this.editingPersona = persona;
this.showPersonaDialog = true;
},
viewPersona(persona) {
this.viewingPersona = persona;
this.showViewDialog = true;
},
handlePersonaSaved(message) {
this.showSuccess(message);
this.loadPersonas();
},
async deletePersona(persona) {
if (!confirm(this.tm('messages.deleteConfirm', { id: persona.persona_id }))) {
return;
}
try {
const response = await axios.post('/api/persona/delete', {
persona_id: persona.persona_id
});
if (response.data.status === 'ok') {
this.showSuccess(response.data.message || this.tm('messages.deleteSuccess'));
await this.loadPersonas();
} else {
this.showError(response.data.message || this.tm('messages.deleteError'));
}
} catch (error) {
this.showError(error.response?.data?.message || this.tm('messages.deleteError'));
}
},
truncateText(text, maxLength) {
if (!text) return '';
return text.length > maxLength ? text.substring(0, maxLength) + '...' : text;
},
formatDate(dateString) {
if (!dateString) return '';
return new Date(dateString).toLocaleString();
},
showSuccess(message) {
this.message = message;
this.messageType = 'success';
this.showMessage = true;
},
showError(message) {
this.message = message;
this.messageType = 'error';
this.showMessage = true;
}
}
}
};
</script>
<style scoped>
@@ -280,43 +41,4 @@ export default {
padding: 20px;
padding-top: 8px;
}
.persona-card {
transition: all 0.3s ease;
height: 100%;
cursor: pointer;
}
.system-prompt-preview {
font-size: 14px;
line-height: 1.4;
color: rgba(var(--v-theme-on-surface), 0.7);
overflow: hidden;
display: -webkit-box;
-webkit-line-clamp: 3;
line-clamp: 3;
-webkit-box-orient: vertical;
}
.system-prompt-content {
max-height: 400px;
overflow: auto;
padding: 12px;
border-radius: 8px;
font-size: 14px;
line-height: 1.5;
white-space: pre-wrap;
word-break: break-word;
}
.dialog-content {
background-color: rgba(var(--v-theme-surface-variant), 0.3);
padding: 8px 12px;
border-radius: 8px;
font-size: 14px;
line-height: 1.4;
margin-bottom: 8px;
white-space: pre-wrap;
word-break: break-word;
}
</style>
@@ -0,0 +1,77 @@
<template>
<BaseCreateFolderDialog v-model="showDialog" :parent-folder-id="parentFolderId" :labels="labels"
@create="handleCreate" ref="baseDialog" />
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
import { usePersonaStore } from '@/stores/personaStore';
import { mapActions } from 'pinia';
import BaseCreateFolderDialog from '@/components/folder/BaseCreateFolderDialog.vue';
import type { CreateFolderData } from '@/components/folder/types';
export default defineComponent({
name: 'CreateFolderDialog',
components: {
BaseCreateFolderDialog
},
props: {
modelValue: {
type: Boolean,
default: false
},
parentFolderId: {
type: String as PropType<string | null>,
default: null
}
},
emits: ['update:modelValue', 'created', 'error'],
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
computed: {
showDialog: {
get(): boolean {
return this.modelValue;
},
set(value: boolean) {
this.$emit('update:modelValue', value);
}
},
labels() {
return {
title: this.tm('folder.createDialog.title'),
nameLabel: this.tm('folder.form.name'),
descriptionLabel: this.tm('folder.form.description'),
nameRequired: this.tm('folder.validation.nameRequired'),
cancelButton: this.tm('buttons.cancel'),
createButton: this.tm('folder.createDialog.createButton')
};
}
},
methods: {
...mapActions(usePersonaStore, ['createFolder']),
async handleCreate(data: CreateFolderData) {
const baseDialog = this.$refs.baseDialog as InstanceType<typeof BaseCreateFolderDialog>;
baseDialog.setLoading(true);
try {
await this.createFolder({
name: data.name,
description: data.description,
parent_id: data.parent_id
});
this.$emit('created', this.tm('folder.messages.createSuccess'));
this.showDialog = false;
} catch (error: any) {
this.$emit('error', error.message || this.tm('folder.messages.createError'));
} finally {
baseDialog.setLoading(false);
}
}
}
});
</script>
@@ -0,0 +1,87 @@
<template>
<v-breadcrumbs :items="breadcrumbItems" class="folder-breadcrumb pa-0">
<template v-slot:prepend>
<v-icon size="small" class="mr-1">mdi-folder-outline</v-icon>
</template>
<template v-slot:item="{ item }">
<v-breadcrumbs-item :disabled="item.disabled" @click="!item.disabled && handleClick((item as any).folderId)"
:class="{ 'breadcrumb-link': !item.disabled }">
<v-icon v-if="(item as any).isRoot" size="small" class="mr-1">mdi-home</v-icon>
{{ item.title }}
</v-breadcrumbs-item>
</template>
<template v-slot:divider>
<v-icon size="small">mdi-chevron-right</v-icon>
</template>
</v-breadcrumbs>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
import { usePersonaStore } from '@/stores/personaStore';
import { mapState, mapActions } from 'pinia';
import type { FolderTreeNode } from '@/components/folder/types';
interface BreadcrumbItem {
title: string;
folderId: string | null;
disabled: boolean;
isRoot: boolean;
}
export default defineComponent({
name: 'FolderBreadcrumb',
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
computed: {
...mapState(usePersonaStore, ['breadcrumbPath', 'currentFolderId']),
breadcrumbItems(): BreadcrumbItem[] {
const items: BreadcrumbItem[] = [
{
title: this.tm('folder.rootFolder'),
folderId: null,
disabled: this.currentFolderId === null,
isRoot: true
}
];
(this.breadcrumbPath as FolderTreeNode[]).forEach((folder, index) => {
items.push({
title: folder.name,
folderId: folder.folder_id,
disabled: index === (this.breadcrumbPath as FolderTreeNode[]).length - 1,
isRoot: false
});
});
return items;
}
},
methods: {
...mapActions(usePersonaStore, ['navigateToFolder']),
handleClick(folderId: string | null) {
this.navigateToFolder(folderId);
}
}
});
</script>
<style scoped>
.folder-breadcrumb {
font-size: 14px;
}
.breadcrumb-link {
cursor: pointer;
transition: color 0.2s;
}
.breadcrumb-link:hover {
color: rgb(var(--v-theme-primary));
}
</style>
+120
View File
@@ -0,0 +1,120 @@
<template>
<v-card class="folder-card" :class="{ 'drag-over': isDragOver }" rounded="lg" @click="$emit('click')" @contextmenu.prevent="$emit('contextmenu', $event)"
elevation="1" hover @dragover.prevent="handleDragOver" @dragleave="handleDragLeave" @drop.prevent="handleDrop">
<v-card-text class="d-flex align-center pa-3">
<v-icon size="40" color="amber-darken-2" class="mr-3">mdi-folder</v-icon>
<div class="folder-info flex-grow-1 overflow-hidden">
<div class="text-subtitle-1 font-weight-medium text-truncate">{{ folder.name }}</div>
<div v-if="folder.description" class="text-body-2 text-medium-emphasis text-truncate">
{{ folder.description }}
</div>
</div>
<v-menu offset-y>
<template v-slot:activator="{ props }">
<v-btn icon="mdi-dots-vertical" variant="text" size="small" v-bind="props" @click.stop />
</template>
<v-list density="compact">
<v-list-item @click.stop="$emit('open')">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-open</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.open') }}</v-list-item-title>
</v-list-item>
<v-list-item @click.stop="$emit('rename')">
<template v-slot:prepend>
<v-icon size="small">mdi-pencil</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.rename') }}</v-list-item-title>
</v-list-item>
<v-list-item @click.stop="$emit('move')">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-move</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.moveTo') }}</v-list-item-title>
</v-list-item>
<v-divider class="my-1" />
<v-list-item @click.stop="$emit('delete')" class="text-error">
<template v-slot:prepend>
<v-icon size="small" color="error">mdi-delete</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.delete') }}</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
</v-card-text>
</v-card>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
import type { Folder } from '@/components/folder/types';
export default defineComponent({
name: 'FolderCard',
props: {
folder: {
type: Object as PropType<Folder>,
required: true
}
},
emits: ['click', 'contextmenu', 'open', 'rename', 'move', 'delete', 'persona-dropped'],
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
data() {
return {
isDragOver: false
};
},
methods: {
handleDragOver(event: DragEvent) {
if (event.dataTransfer) {
event.dataTransfer.dropEffect = 'move';
}
this.isDragOver = true;
},
handleDragLeave() {
this.isDragOver = false;
},
handleDrop(event: DragEvent) {
this.isDragOver = false;
if (!event.dataTransfer) return;
try {
const data = JSON.parse(event.dataTransfer.getData('application/json'));
if (data.type === 'persona') {
this.$emit('persona-dropped', {
persona_id: data.persona_id,
target_folder_id: this.folder.folder_id
});
}
} catch (e) {
console.error('Failed to parse drop data:', e);
}
}
}
});
</script>
<style scoped>
.folder-card {
cursor: pointer;
transition: all 0.2s ease;
}
.folder-card:hover {
transform: translateY(-2px);
}
.folder-card.drag-over {
background-color: rgba(var(--v-theme-primary), 0.15);
border: 2px dashed rgb(var(--v-theme-primary));
transform: scale(1.02);
}
.folder-info {
min-width: 0;
}
</style>
+320
View File
@@ -0,0 +1,320 @@
<template>
<div class="folder-tree">
<!-- 搜索框 -->
<v-text-field v-model="searchQuery" :placeholder="tm('folder.searchPlaceholder')" prepend-inner-icon="mdi-magnify"
variant="outlined" density="compact" hide-details clearable class="mb-3" />
<!-- 根目录节点 -->
<v-list density="compact" nav class="tree-list" bg-color="transparent">
<v-list-item :active="currentFolderId === null" @click="handleFolderClick(null)" rounded="lg"
:class="['root-item', { 'drag-over': isRootDragOver }]"
@dragover.prevent="handleRootDragOver" @dragleave="handleRootDragLeave" @drop.prevent="handleRootDrop">
<template v-slot:prepend>
<v-icon>mdi-home</v-icon>
</template>
<v-list-item-title>{{ tm('folder.rootFolder') }}</v-list-item-title>
</v-list-item>
<!-- 文件夹树 -->
<template v-if="!treeLoading">
<FolderTreeNode v-for="folder in filteredFolderTree" :key="folder.folder_id" :folder="folder"
:depth="0" :current-folder-id="currentFolderId" :search-query="searchQuery"
@folder-click="handleFolderClick" @folder-context-menu="handleContextMenu"
@persona-dropped="$emit('persona-dropped', $event)" />
</template>
<!-- 加载状态 -->
<div v-if="treeLoading" class="text-center pa-4">
<v-progress-circular indeterminate size="24" />
</div>
<!-- 空状态 -->
<div v-if="!treeLoading && folderTree.length === 0" class="text-center pa-4 text-medium-emphasis">
<v-icon size="32" class="mb-2">mdi-folder-outline</v-icon>
<div class="text-body-2">{{ tm('folder.noFolders') }}</div>
</div>
</v-list>
<!-- 右键菜单 -->
<v-menu v-model="contextMenu.show" :target="contextMenu.target as any" location="end" :close-on-content-click="true">
<v-list density="compact">
<v-list-item @click="openFolder">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-open</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.open') }}</v-list-item-title>
</v-list-item>
<v-list-item @click="renameFolder">
<template v-slot:prepend>
<v-icon size="small">mdi-pencil</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.rename') }}</v-list-item-title>
</v-list-item>
<v-list-item @click="$emit('move-folder', contextMenu.folder)">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-move</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.moveTo') }}</v-list-item-title>
</v-list-item>
<v-divider class="my-1" />
<v-list-item @click="confirmDeleteFolder" class="text-error">
<template v-slot:prepend>
<v-icon size="small" color="error">mdi-delete</v-icon>
</template>
<v-list-item-title>{{ tm('folder.contextMenu.delete') }}</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
<!-- 重命名对话框 -->
<v-dialog v-model="renameDialog.show" max-width="400px" persistent>
<v-card>
<v-card-title>{{ tm('folder.renameDialog.title') }}</v-card-title>
<v-card-text>
<v-text-field v-model="renameDialog.name" :label="tm('folder.form.name')"
:rules="[v => !!v || tm('folder.validation.nameRequired')]" variant="outlined"
density="comfortable" autofocus @keyup.enter="submitRename" />
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="renameDialog.show = false">
{{ tm('buttons.cancel') }}
</v-btn>
<v-btn color="primary" variant="flat" @click="submitRename" :loading="renameDialog.loading"
:disabled="!renameDialog.name">
{{ tm('buttons.save') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
<!-- 删除确认对话框 -->
<v-dialog v-model="deleteDialog.show" max-width="450px">
<v-card>
<v-card-title class="text-error">
<v-icon class="mr-2" color="error">mdi-alert</v-icon>
{{ tm('folder.deleteDialog.title') }}
</v-card-title>
<v-card-text>
<p>{{ tm('folder.deleteDialog.message', { name: deleteDialog.folder?.name ?? '' }) }}</p>
<p class="text-warning mt-2">
<v-icon size="small" class="mr-1">mdi-information</v-icon>
{{ tm('folder.deleteDialog.warning') }}
</p>
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="deleteDialog.show = false">
{{ tm('buttons.cancel') }}
</v-btn>
<v-btn color="error" variant="flat" @click="submitDelete" :loading="deleteDialog.loading">
{{ tm('buttons.delete') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
import { usePersonaStore } from '@/stores/personaStore';
import { mapState, mapActions } from 'pinia';
import FolderTreeNode from './FolderTreeNode.vue';
import type { FolderTreeNode as FolderTreeNodeType } from '@/components/folder/types';
interface ContextMenuState {
show: boolean;
target: [number, number] | null;
folder: FolderTreeNodeType | null;
}
interface RenameDialogState {
show: boolean;
folder: FolderTreeNodeType | null;
name: string;
loading: boolean;
}
interface DeleteDialogState {
show: boolean;
folder: FolderTreeNodeType | null;
loading: boolean;
}
export default defineComponent({
name: 'FolderTree',
components: {
FolderTreeNode
},
emits: ['move-folder', 'error', 'success', 'persona-dropped'],
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
data() {
return {
searchQuery: '',
isRootDragOver: false,
contextMenu: {
show: false,
target: null,
folder: null
} as ContextMenuState,
renameDialog: {
show: false,
folder: null,
name: '',
loading: false
} as RenameDialogState,
deleteDialog: {
show: false,
folder: null,
loading: false
} as DeleteDialogState
};
},
computed: {
...mapState(usePersonaStore, ['folderTree', 'currentFolderId', 'treeLoading']),
filteredFolderTree(): FolderTreeNodeType[] {
if (!this.searchQuery) {
return this.folderTree as FolderTreeNodeType[];
}
const query = this.searchQuery.toLowerCase();
return this.filterTreeBySearch(this.folderTree as FolderTreeNodeType[], query);
}
},
methods: {
...mapActions(usePersonaStore, ['navigateToFolder', 'updateFolder', 'deleteFolder']),
filterTreeBySearch(nodes: FolderTreeNodeType[], query: string): FolderTreeNodeType[] {
return nodes.filter(node => {
const matches = node.name.toLowerCase().includes(query);
const childMatches = this.filterTreeBySearch(node.children || [], query);
return matches || childMatches.length > 0;
}).map(node => ({
...node,
children: this.filterTreeBySearch(node.children || [], query)
}));
},
handleFolderClick(folderId: string | null) {
this.navigateToFolder(folderId);
},
handleRootDragOver(event: DragEvent) {
if (event.dataTransfer) {
event.dataTransfer.dropEffect = 'move';
}
this.isRootDragOver = true;
},
handleRootDragLeave() {
this.isRootDragOver = false;
},
handleRootDrop(event: DragEvent) {
this.isRootDragOver = false;
if (!event.dataTransfer) return;
try {
const data = JSON.parse(event.dataTransfer.getData('application/json'));
if (data.type === 'persona') {
this.$emit('persona-dropped', {
persona_id: data.persona_id,
target_folder_id: null
});
}
} catch (e) {
console.error('Failed to parse drop data:', e);
}
},
handleContextMenu(eventData: { event: MouseEvent; folder: FolderTreeNodeType }) {
this.contextMenu.target = [eventData.event.clientX, eventData.event.clientY];
this.contextMenu.folder = eventData.folder;
this.contextMenu.show = true;
},
openFolder() {
if (this.contextMenu.folder) {
this.navigateToFolder(this.contextMenu.folder.folder_id);
}
},
renameFolder() {
if (this.contextMenu.folder) {
this.renameDialog.folder = this.contextMenu.folder;
this.renameDialog.name = this.contextMenu.folder.name;
this.renameDialog.show = true;
}
},
async submitRename() {
if (!this.renameDialog.name || !this.renameDialog.folder) return;
this.renameDialog.loading = true;
try {
await this.updateFolder({
folder_id: this.renameDialog.folder.folder_id,
name: this.renameDialog.name
});
this.$emit('success', this.tm('folder.messages.renameSuccess'));
this.renameDialog.show = false;
} catch (error: any) {
this.$emit('error', error.message || this.tm('folder.messages.renameError'));
} finally {
this.renameDialog.loading = false;
}
},
confirmDeleteFolder() {
if (this.contextMenu.folder) {
this.deleteDialog.folder = this.contextMenu.folder;
this.deleteDialog.show = true;
}
},
async submitDelete() {
if (!this.deleteDialog.folder) return;
this.deleteDialog.loading = true;
try {
await this.deleteFolder(this.deleteDialog.folder.folder_id);
this.$emit('success', this.tm('folder.messages.deleteSuccess'));
this.deleteDialog.show = false;
} catch (error: any) {
this.$emit('error', error.message || this.tm('folder.messages.deleteError'));
} finally {
this.deleteDialog.loading = false;
}
}
}
});
</script>
<style scoped>
.folder-tree {
height: 100%;
display: flex;
flex-direction: column;
}
.tree-list {
flex: 1;
overflow-y: auto;
}
.root-item {
margin-bottom: 4px;
transition: all 0.2s ease;
}
.root-item.drag-over {
background-color: rgba(var(--v-theme-primary), 0.15);
border: 2px dashed rgb(var(--v-theme-primary));
border-radius: 8px;
}
</style>
@@ -0,0 +1,66 @@
<template>
<BaseFolderTreeNode :folder="folder" :depth="depth" :current-folder-id="currentFolderId"
:search-query="searchQuery" :expanded-folder-ids="expandedFolderIds" :accept-drop-types="['persona']"
@folder-click="$emit('folder-click', $event)"
@folder-context-menu="handleContextMenu"
@item-dropped="handleItemDropped"
@toggle-expansion="toggleFolderExpansion"
@set-expansion="handleSetExpansion" />
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import { usePersonaStore } from '@/stores/personaStore';
import { mapState, mapActions } from 'pinia';
import BaseFolderTreeNode from '@/components/folder/BaseFolderTreeNode.vue';
import type { FolderTreeNode as FolderTreeNodeType } from '@/components/folder/types';
export default defineComponent({
name: 'FolderTreeNode',
components: {
BaseFolderTreeNode
},
props: {
folder: {
type: Object as PropType<FolderTreeNodeType>,
required: true
},
depth: {
type: Number,
default: 0
},
currentFolderId: {
type: String as PropType<string | null>,
default: null
},
searchQuery: {
type: String,
default: ''
}
},
emits: ['folder-click', 'folder-context-menu', 'persona-dropped'],
computed: {
...mapState(usePersonaStore, ['expandedFolderIds'])
},
methods: {
...mapActions(usePersonaStore, ['toggleFolderExpansion', 'setFolderExpansion']),
handleContextMenu(event: { event: MouseEvent; folder: FolderTreeNodeType }) {
this.$emit('folder-context-menu', event);
},
handleItemDropped(data: { item_id: string; item_type: string; target_folder_id: string | null; source_data: any }) {
if (data.item_type === 'persona') {
this.$emit('persona-dropped', {
persona_id: data.item_id,
target_folder_id: data.target_folder_id
});
}
},
handleSetExpansion(data: { folderId: string; expanded: boolean }) {
this.setFolderExpansion(data.folderId, data.expanded);
}
}
});
</script>
@@ -0,0 +1,36 @@
<template>
<BaseMoveTargetNode :folder="folder" :depth="depth" :selected-folder-id="selectedFolderId"
:disabled-folder-ids="disabledFolderIds" @select="$emit('select', $event)" />
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import BaseMoveTargetNode from '@/components/folder/BaseMoveTargetNode.vue';
import type { FolderTreeNode } from '@/components/folder/types';
export default defineComponent({
name: 'MoveTargetNode',
components: {
BaseMoveTargetNode
},
props: {
folder: {
type: Object as PropType<FolderTreeNode>,
required: true
},
depth: {
type: Number,
default: 0
},
selectedFolderId: {
type: String as PropType<string | null>,
default: null
},
disabledFolderIds: {
type: Array as PropType<string[]>,
default: () => []
}
},
emits: ['select']
});
</script>
@@ -0,0 +1,201 @@
<template>
<v-dialog v-model="showDialog" max-width="500px" persistent>
<v-card>
<v-card-title>
<v-icon class="mr-2">mdi-folder-move</v-icon>
{{ tm('moveDialog.title') }}
</v-card-title>
<v-card-text>
<p class="text-body-2 text-medium-emphasis mb-4">
{{ tm('moveDialog.description', { name: itemName }) }}
</p>
<!-- 文件夹选择树 -->
<div class="folder-select-tree">
<v-list density="compact" nav class="tree-list">
<!-- 根目录选项 -->
<v-list-item :active="selectedFolderId === null" @click="selectFolder(null)" rounded="lg"
class="mb-1">
<template v-slot:prepend>
<v-icon>mdi-home</v-icon>
</template>
<v-list-item-title>{{ tm('folder.rootFolder') }}</v-list-item-title>
</v-list-item>
<!-- 文件夹树 -->
<template v-if="!treeLoading">
<MoveTargetNode v-for="folder in availableFolders" :key="folder.folder_id" :folder="folder"
:depth="0" :selected-folder-id="selectedFolderId" :disabled-folder-ids="disabledFolderIds"
@select="selectFolder" />
</template>
<!-- 加载状态 -->
<div v-if="treeLoading" class="text-center pa-4">
<v-progress-circular indeterminate size="24" />
</div>
</v-list>
</div>
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="closeDialog">
{{ tm('buttons.cancel') }}
</v-btn>
<v-btn color="primary" variant="flat" @click="submitMove" :loading="loading">
{{ tm('buttons.move') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
import { usePersonaStore } from '@/stores/personaStore';
import { mapState, mapActions } from 'pinia';
import MoveTargetNode from './MoveTargetNode.vue';
import { collectFolderAndChildrenIds } from '@/components/folder/useFolderManager';
import type { FolderTreeNode } from '@/components/folder/types';
interface PersonaItem {
persona_id: string;
folder_id?: string | null;
[key: string]: any;
}
interface FolderItem {
folder_id: string;
name: string;
parent_id?: string | null;
[key: string]: any;
}
export default defineComponent({
name: 'MoveToFolderDialog',
components: {
MoveTargetNode
},
props: {
modelValue: {
type: Boolean,
default: false
},
itemType: {
type: String as PropType<'persona' | 'folder'>,
required: true
},
item: {
type: Object as PropType<PersonaItem | FolderItem | null>,
default: null
}
},
emits: ['update:modelValue', 'moved', 'error'],
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
data() {
return {
selectedFolderId: null as string | null,
loading: false
};
},
computed: {
...mapState(usePersonaStore, ['folderTree', 'treeLoading']),
showDialog: {
get(): boolean {
return this.modelValue;
},
set(value: boolean) {
this.$emit('update:modelValue', value);
}
},
itemName(): string {
if (!this.item) return '';
return this.itemType === 'persona'
? (this.item as PersonaItem).persona_id
: (this.item as FolderItem).name;
},
// ID
disabledFolderIds(): string[] {
if (this.itemType !== 'folder' || !this.item) return [];
return collectFolderAndChildrenIds(
this.folderTree as FolderTreeNode[],
(this.item as FolderItem).folder_id
);
},
//
availableFolders(): FolderTreeNode[] {
return this.folderTree as FolderTreeNode[];
}
},
watch: {
modelValue(newValue: boolean) {
if (newValue) {
//
if (this.item) {
this.selectedFolderId = this.itemType === 'persona'
? (this.item as PersonaItem).folder_id ?? null
: (this.item as FolderItem).parent_id ?? null;
}
}
}
},
methods: {
...mapActions(usePersonaStore, ['movePersonaToFolder', 'moveFolderToFolder']),
selectFolder(folderId: string | null) {
//
if (folderId && this.disabledFolderIds.includes(folderId)) return;
this.selectedFolderId = folderId;
},
closeDialog() {
this.showDialog = false;
},
async submitMove() {
if (!this.item) return;
this.loading = true;
try {
if (this.itemType === 'persona') {
await this.movePersonaToFolder(
(this.item as PersonaItem).persona_id,
this.selectedFolderId
);
} else {
await this.moveFolderToFolder(
(this.item as FolderItem).folder_id,
this.selectedFolderId
);
}
this.$emit('moved', this.tm('moveDialog.success'));
this.closeDialog();
} catch (error: any) {
this.$emit('error', error.message || this.tm('moveDialog.error'));
} finally {
this.loading = false;
}
}
}
});
</script>
<style scoped>
.folder-select-tree {
max-height: 400px;
overflow-y: auto;
border: 1px solid rgba(var(--v-border-color), var(--v-border-opacity));
border-radius: 8px;
}
.tree-list {
padding: 8px;
}
</style>
+178
View File
@@ -0,0 +1,178 @@
<template>
<v-card class="persona-card" :class="{ 'dragging': isDragging }" rounded="lg" @click="$emit('view')" elevation="1" hover
draggable="true" @dragstart="handleDragStart" @dragend="handleDragEnd">
<v-card-title class="d-flex justify-space-between align-center">
<div class="text-truncate ml-2">{{ persona.persona_id }}</div>
<v-menu offset-y>
<template v-slot:activator="{ props }">
<v-btn icon="mdi-dots-vertical" variant="text" size="small" v-bind="props" @click.stop />
</template>
<v-list density="compact">
<v-list-item @click.stop="$emit('edit')">
<template v-slot:prepend>
<v-icon size="small">mdi-pencil</v-icon>
</template>
<v-list-item-title>{{ tm('buttons.edit') }}</v-list-item-title>
</v-list-item>
<v-list-item @click.stop="$emit('move')">
<template v-slot:prepend>
<v-icon size="small">mdi-folder-move</v-icon>
</template>
<v-list-item-title>{{ tm('persona.contextMenu.moveTo') }}</v-list-item-title>
</v-list-item>
<v-divider class="my-1" />
<v-list-item @click.stop="$emit('delete')" class="text-error">
<template v-slot:prepend>
<v-icon size="small" color="error">mdi-delete</v-icon>
</template>
<v-list-item-title>{{ tm('buttons.delete') }}</v-list-item-title>
</v-list-item>
</v-list>
</v-menu>
</v-card-title>
<v-card-text>
<div class="system-prompt-preview">
{{ truncateText(persona.system_prompt, 100) }}
</div>
<div class="mt-3 d-flex flex-wrap ga-1">
<v-chip v-if="persona.begin_dialogs && persona.begin_dialogs.length > 0" size="small" color="secondary"
variant="tonal" prepend-icon="mdi-chat">
{{ tm('labels.presetDialogs', { count: persona.begin_dialogs.length / 2 }) }}
</v-chip>
<v-chip v-if="persona.tools === null" size="small" color="success" variant="tonal"
prepend-icon="mdi-tools">
{{ tm('form.allToolsAvailable') }}
</v-chip>
<v-chip v-else-if="persona.tools && persona.tools.length > 0" size="small" color="primary" variant="tonal"
prepend-icon="mdi-tools">
{{ persona.tools.length }} {{ tm('persona.toolsCount') }}
</v-chip>
</div>
<div class="mt-3 text-caption text-medium-emphasis">
{{ tm('labels.createdAt') }}: {{ formatDate(persona.created_at) }}
</div>
</v-card-text>
</v-card>
<!-- Custom Drag Preview -->
<div ref="dragPreview" class="drag-preview">
<v-icon size="small" class="mr-2">mdi-account</v-icon>
<span class="text-subtitle-2">{{ persona.persona_id }}</span>
</div>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue';
import { useModuleI18n } from '@/i18n/composables';
interface Persona {
persona_id: string;
system_prompt: string;
begin_dialogs?: string[] | null;
tools?: string[] | null;
created_at?: string;
updated_at?: string;
folder_id?: string | null;
[key: string]: any;
}
export default defineComponent({
name: 'PersonaCard',
props: {
persona: {
type: Object as PropType<Persona>,
required: true
}
},
emits: ['view', 'edit', 'move', 'delete'],
setup() {
const { tm } = useModuleI18n('features/persona');
return { tm };
},
data() {
return {
isDragging: false
};
},
methods: {
handleDragStart(event: DragEvent) {
this.isDragging = true;
if (event.dataTransfer) {
event.dataTransfer.effectAllowed = 'move';
event.dataTransfer.setData('application/json', JSON.stringify({
type: 'persona',
persona_id: this.persona.persona_id,
persona: this.persona
}));
// Set custom drag image
const dragPreview = this.$refs.dragPreview as HTMLElement;
if (dragPreview) {
event.dataTransfer.setDragImage(dragPreview, 15, 15);
}
}
},
handleDragEnd() {
this.isDragging = false;
},
truncateText(text: string | undefined | null, maxLength: number): string {
if (!text) return '';
return text.length > maxLength ? text.substring(0, maxLength) + '...' : text;
},
formatDate(dateString: string | undefined | null): string {
if (!dateString) return '';
return new Date(dateString).toLocaleString();
}
}
});
</script>
<style scoped>
.persona-card {
height: 100%;
cursor: grab;
transition: all 0.2s ease;
}
.persona-card:active {
cursor: grabbing;
}
.persona-card.dragging {
opacity: 0.5;
transform: scale(0.95);
}
.persona-card:hover {
transform: translateY(-2px);
}
.system-prompt-preview {
font-size: 14px;
line-height: 1.4;
color: rgba(var(--v-theme-on-surface), 0.7);
overflow: hidden;
display: -webkit-box;
-webkit-line-clamp: 3;
line-clamp: 3;
-webkit-box-orient: vertical;
}
.drag-preview {
position: fixed;
top: -1000px;
left: -1000px;
background: rgb(var(--v-theme-surface));
padding: 12px 20px;
border-radius: 8px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
display: flex;
align-items: center;
border: 1px solid rgba(var(--v-border-color), var(--v-border-opacity));
z-index: 9999;
pointer-events: none;
}
</style>
@@ -0,0 +1,557 @@
<template>
<div class="persona-manager">
<!-- 移动端顶部导航 -->
<div class="mobile-nav d-md-none mb-4">
<FolderBreadcrumb />
</div>
<div class="manager-layout">
<!-- 左侧边栏 - 仅桌面端显示 -->
<div class="sidebar d-none d-md-block">
<div class="sidebar-header d-flex justify-space-between align-center mb-3">
<h3 class="text-h6">{{ tm('folder.sidebarTitle') }}</h3>
<v-btn icon="mdi-folder-plus" variant="text" size="small" @click="showCreateFolderDialog = true"
:title="tm('folder.createButton')" />
</div>
<FolderTree @move-folder="openMoveFolderDialog" @success="showSuccess" @error="showError"
@persona-dropped="handlePersonaDropped" />
</div>
<!-- 主内容区 -->
<div class="main-content">
<!-- 顶部工具栏 -->
<div class="toolbar d-flex flex-wrap justify-space-between align-center mb-4 ga-2">
<!-- 面包屑 - 仅桌面端显示 -->
<div class="d-none d-md-block">
<FolderBreadcrumb />
</div>
<!-- 操作按钮组 -->
<div class="d-flex ga-2">
<v-btn color="primary" variant="tonal" prepend-icon="mdi-plus" @click="openCreatePersonaDialog"
rounded="lg">
{{ tm('buttons.create') }}
</v-btn>
<v-btn variant="outlined" prepend-icon="mdi-folder-plus" @click="showCreateFolderDialog = true"
rounded="lg">
{{ tm('folder.createButton') }}
</v-btn>
</div>
</div>
<!-- 加载状态 - 只有加载超过阈值才显示骨架屏 -->
<v-fade-transition>
<div v-if="showSkeleton" class="loading-container">
<v-row>
<v-col v-for="n in 6" :key="n" cols="12" sm="6" lg="4" xl="3">
<v-skeleton-loader type="card" rounded="lg" />
</v-col>
</v-row>
</div>
</v-fade-transition>
<!-- 内容区域 -->
<div v-if="!loading">
<!-- 子文件夹区域 -->
<div v-if="currentFolders.length > 0" class="folders-section mb-6">
<h3 class="text-subtitle-1 font-weight-medium mb-3">
<v-icon size="small" class="mr-1">mdi-folder</v-icon>
{{ tm('folder.foldersTitle') }} ({{ currentFolders.length }})
</h3>
<v-row>
<v-col v-for="folder in currentFolders" :key="folder.folder_id" cols="12" sm="6" lg="4"
xl="3">
<FolderCard :folder="folder" @click="navigateToFolder(folder.folder_id)"
@open="navigateToFolder(folder.folder_id)" @rename="openRenameFolderDialog(folder)"
@move="openMoveFolderDialog(folder)" @delete="confirmDeleteFolder(folder)"
@persona-dropped="handlePersonaDropped" />
</v-col>
</v-row>
</div>
<!-- Persona 区域 -->
<div v-if="currentPersonas.length > 0" class="personas-section">
<h3 class="text-subtitle-1 font-weight-medium mb-3">
<v-icon size="small" class="mr-1">mdi-account-heart</v-icon>
{{ tm('persona.personasTitle') }} ({{ currentPersonas.length }})
</h3>
<v-row>
<v-col v-for="persona in currentPersonas" :key="persona.persona_id" cols="12" sm="6" lg="4"
xl="3">
<PersonaCard :persona="persona" @view="viewPersona(persona)"
@edit="editPersona(persona)" @move="openMovePersonaDialog(persona)"
@delete="confirmDeletePersona(persona)" />
</v-col>
</v-row>
</div>
<!-- 空状态 -->
<div v-if="currentFolders.length === 0 && currentPersonas.length === 0" class="empty-state">
<v-card class="text-center pa-8" elevation="0">
<v-icon size="64" color="grey-lighten-1" class="mb-4">mdi-folder-open-outline</v-icon>
<h3 class="text-h5 mb-2">{{ tm('empty.folderEmpty') }}</h3>
<p class="text-body-1 text-medium-emphasis mb-4">{{ tm('empty.folderEmptyDescription') }}</p>
<div class="d-flex justify-center ga-2">
<v-btn color="primary" variant="tonal" prepend-icon="mdi-plus"
@click="openCreatePersonaDialog">
{{ tm('buttons.create') }}
</v-btn>
<v-btn variant="outlined" prepend-icon="mdi-folder-plus"
@click="showCreateFolderDialog = true">
{{ tm('folder.createButton') }}
</v-btn>
</div>
</v-card>
</div>
</div>
</div>
</div>
<!-- 创建/编辑 Persona 对话框 -->
<PersonaForm v-model="showPersonaDialog" :editing-persona="editingPersona ?? undefined"
:current-folder-id="currentFolderId ?? undefined" :current-folder-name="currentFolderName ?? undefined"
@saved="handlePersonaSaved" @error="showError" />
<!-- 查看 Persona 详情对话框 -->
<v-dialog v-model="showViewDialog" max-width="700px">
<v-card v-if="viewingPersona">
<v-card-title class="d-flex justify-space-between align-center">
<span class="text-h5">{{ viewingPersona.persona_id }}</span>
<v-btn icon="mdi-close" variant="text" @click="showViewDialog = false" />
</v-card-title>
<v-card-text>
<div class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.systemPrompt') }}</h4>
<pre class="system-prompt-content">{{ viewingPersona.system_prompt }}</pre>
</div>
<div v-if="viewingPersona.begin_dialogs && viewingPersona.begin_dialogs.length > 0" class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.presetDialogs') }}</h4>
<div v-for="(dialog, index) in viewingPersona.begin_dialogs" :key="index" class="mb-2">
<v-chip :color="index % 2 === 0 ? 'primary' : 'secondary'" variant="tonal" size="small"
class="mb-1">
{{ index % 2 === 0 ? tm('form.userMessage') : tm('form.assistantMessage') }}
</v-chip>
<div class="dialog-content ml-2">{{ dialog }}</div>
</div>
</div>
<div class="mb-4">
<h4 class="text-h6 mb-2">{{ tm('form.tools') }}</h4>
<div v-if="viewingPersona.tools === null" class="text-body-2 text-medium-emphasis">
<v-chip size="small" color="success" variant="tonal" prepend-icon="mdi-check-all">
{{ tm('form.allToolsAvailable') }}
</v-chip>
</div>
<div v-else-if="viewingPersona.tools && viewingPersona.tools.length > 0"
class="d-flex flex-wrap ga-1">
<v-chip v-for="toolName in viewingPersona.tools" :key="toolName" size="small"
color="primary" variant="tonal">
{{ toolName }}
</v-chip>
</div>
<div v-else class="text-body-2 text-medium-emphasis">
{{ tm('form.noToolsSelected') }}
</div>
</div>
<div class="text-caption text-medium-emphasis">
<div>{{ tm('labels.createdAt') }}: {{ formatDate(viewingPersona.created_at) }}</div>
<div v-if="viewingPersona.updated_at">{{ tm('labels.updatedAt') }}:
{{ formatDate(viewingPersona.updated_at) }}</div>
</div>
</v-card-text>
</v-card>
</v-dialog>
<!-- 创建文件夹对话框 -->
<CreateFolderDialog v-model="showCreateFolderDialog" :parent-folder-id="currentFolderId"
@created="showSuccess" @error="showError" />
<!-- 重命名文件夹对话框 -->
<v-dialog v-model="showRenameFolderDialog" max-width="400px">
<v-card>
<v-card-title>{{ tm('folder.renameDialog.title') }}</v-card-title>
<v-card-text>
<v-text-field v-model="renameFolderData.name" :label="tm('folder.form.name')"
:rules="[v => !!v || tm('folder.validation.nameRequired')]" variant="outlined"
density="comfortable" autofocus @keyup.enter="submitRenameFolder" />
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="showRenameFolderDialog = false">
{{ tm('buttons.cancel') }}
</v-btn>
<v-btn color="primary" variant="flat" @click="submitRenameFolder" :loading="renameLoading"
:disabled="!renameFolderData.name">
{{ tm('buttons.save') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
<!-- 移动对话框 -->
<MoveToFolderDialog v-model="showMoveDialog" :item-type="moveDialogType" :item="moveDialogItem"
@moved="showSuccess" @error="showError" />
<!-- 删除文件夹确认对话框 -->
<v-dialog v-model="showDeleteFolderDialog" max-width="450px">
<v-card>
<v-card-title class="text-error">
<v-icon class="mr-2" color="error">mdi-alert</v-icon>
{{ tm('folder.deleteDialog.title') }}
</v-card-title>
<v-card-text>
<p>{{ tm('folder.deleteDialog.message', { name: deleteFolderData?.name ?? '' }) }}</p>
<p class="text-warning mt-2">
<v-icon size="small" class="mr-1">mdi-information</v-icon>
{{ tm('folder.deleteDialog.warning') }}
</p>
</v-card-text>
<v-card-actions>
<v-spacer />
<v-btn variant="text" @click="showDeleteFolderDialog = false">
{{ tm('buttons.cancel') }}
</v-btn>
<v-btn color="error" variant="flat" @click="submitDeleteFolder" :loading="deleteLoading">
{{ tm('buttons.delete') }}
</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
<!-- 消息提示 -->
<v-snackbar :timeout="3000" elevation="24" :color="messageType" v-model="showMessage" location="top">
{{ message }}
</v-snackbar>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useI18n, useModuleI18n } from '@/i18n/composables';
import { usePersonaStore } from '@/stores/personaStore';
import { mapState, mapActions } from 'pinia';
import FolderTree from './FolderTree.vue';
import FolderBreadcrumb from './FolderBreadcrumb.vue';
import FolderCard from './FolderCard.vue';
import PersonaCard from './PersonaCard.vue';
import PersonaForm from '@/components/shared/PersonaForm.vue';
import CreateFolderDialog from './CreateFolderDialog.vue';
import MoveToFolderDialog from './MoveToFolderDialog.vue';
import type { Folder, FolderTreeNode } from '@/components/folder/types';
interface Persona {
persona_id: string;
system_prompt: string;
begin_dialogs?: string[] | null;
tools?: string[] | null;
created_at?: string;
updated_at?: string;
folder_id?: string | null;
[key: string]: any;
}
interface RenameFolderData {
folder: Folder | null;
name: string;
}
export default defineComponent({
name: 'PersonaManager',
components: {
FolderTree,
FolderBreadcrumb,
FolderCard,
PersonaCard,
PersonaForm,
CreateFolderDialog,
MoveToFolderDialog
},
setup() {
const { t } = useI18n();
const { tm } = useModuleI18n('features/persona');
return { t, tm };
},
data() {
return {
// Persona
showPersonaDialog: false,
showViewDialog: false,
editingPersona: null as Persona | null,
viewingPersona: null as Persona | null,
//
showCreateFolderDialog: false,
showRenameFolderDialog: false,
showDeleteFolderDialog: false,
renameFolderData: { folder: null, name: '' } as RenameFolderData,
deleteFolderData: null as Folder | null,
renameLoading: false,
deleteLoading: false,
//
showMoveDialog: false,
moveDialogType: 'persona' as 'persona' | 'folder',
moveDialogItem: null as Persona | Folder | null,
//
showMessage: false,
message: '',
messageType: 'success' as 'success' | 'error',
//
showSkeleton: false,
skeletonTimer: null as ReturnType<typeof setTimeout> | null
};
},
computed: {
...mapState(usePersonaStore, ['folderTree', 'currentFolderId', 'currentFolders', 'currentPersonas', 'loading']),
currentFolderName(): string | null {
if (!this.currentFolderId) {
return null; // PersonaForm 使 tm('form.rootFolder')
}
//
const findName = (nodes: FolderTreeNode[], id: string): string | null => {
for (const node of nodes) {
if (node.folder_id === id) {
return node.name;
}
if (node.children && node.children.length > 0) {
const found = findName(node.children, id);
if (found) return found;
}
}
return null;
};
return findName(this.folderTree, this.currentFolderId);
}
},
watch: {
// loading
loading: {
handler(newVal: boolean) {
if (newVal) {
// 150ms
// 150ms
this.skeletonTimer = setTimeout(() => {
if (this.loading) {
this.showSkeleton = true;
}
}, 150);
} else {
//
if (this.skeletonTimer) {
clearTimeout(this.skeletonTimer);
this.skeletonTimer = null;
}
this.showSkeleton = false;
}
},
immediate: true
}
},
beforeUnmount() {
//
if (this.skeletonTimer) {
clearTimeout(this.skeletonTimer);
}
},
async mounted() {
await this.initialize();
},
methods: {
...mapActions(usePersonaStore, ['loadFolderTree', 'navigateToFolder', 'updateFolder', 'deleteFolder', 'deletePersona', 'refreshCurrentFolder', 'movePersonaToFolder']),
async initialize() {
await Promise.all([
this.loadFolderTree(),
this.navigateToFolder(null)
]);
},
// Persona
openCreatePersonaDialog() {
this.editingPersona = null;
this.showPersonaDialog = true;
},
editPersona(persona: Persona) {
this.editingPersona = persona;
this.showPersonaDialog = true;
},
viewPersona(persona: Persona) {
this.viewingPersona = persona;
this.showViewDialog = true;
},
handlePersonaSaved(message: string) {
this.showSuccess(message);
this.refreshCurrentFolder();
},
async confirmDeletePersona(persona: Persona) {
if (!confirm(this.tm('messages.deleteConfirm', { id: persona.persona_id }))) {
return;
}
try {
await this.deletePersona(persona.persona_id);
this.showSuccess(this.tm('messages.deleteSuccess'));
} catch (error: any) {
this.showError(error.message || this.tm('messages.deleteError'));
}
},
openMovePersonaDialog(persona: Persona) {
this.moveDialogType = 'persona';
this.moveDialogItem = persona;
this.showMoveDialog = true;
},
async handlePersonaDropped({ persona_id, target_folder_id }: { persona_id: string; target_folder_id: string | null }) {
try {
await this.movePersonaToFolder(persona_id, target_folder_id);
this.showSuccess(this.tm('persona.messages.moveSuccess'));
// Navigate to the target folder
await this.navigateToFolder(target_folder_id);
} catch (error: any) {
this.showError(error.message || this.tm('persona.messages.moveError'));
}
},
//
openRenameFolderDialog(folder: Folder) {
this.renameFolderData = { folder, name: folder.name };
this.showRenameFolderDialog = true;
},
async submitRenameFolder() {
if (!this.renameFolderData.name || !this.renameFolderData.folder) return;
this.renameLoading = true;
try {
await this.updateFolder({
folder_id: this.renameFolderData.folder.folder_id,
name: this.renameFolderData.name
});
this.showSuccess(this.tm('folder.messages.renameSuccess'));
this.showRenameFolderDialog = false;
} catch (error: any) {
this.showError(error.message || this.tm('folder.messages.renameError'));
} finally {
this.renameLoading = false;
}
},
openMoveFolderDialog(folder: Folder) {
this.moveDialogType = 'folder';
this.moveDialogItem = folder;
this.showMoveDialog = true;
},
confirmDeleteFolder(folder: Folder) {
this.deleteFolderData = folder;
this.showDeleteFolderDialog = true;
},
async submitDeleteFolder() {
if (!this.deleteFolderData) return;
this.deleteLoading = true;
try {
await this.deleteFolder(this.deleteFolderData.folder_id);
this.showSuccess(this.tm('folder.messages.deleteSuccess'));
this.showDeleteFolderDialog = false;
} catch (error: any) {
this.showError(error.message || this.tm('folder.messages.deleteError'));
} finally {
this.deleteLoading = false;
}
},
//
formatDate(dateString: string | undefined | null): string {
if (!dateString) return '';
return new Date(dateString).toLocaleString();
},
showSuccess(message: string) {
this.message = message;
this.messageType = 'success';
this.showMessage = true;
},
showError(message: string) {
this.message = message;
this.messageType = 'error';
this.showMessage = true;
}
}
});
</script>
<style scoped>
.persona-manager {
height: 100%;
}
.manager-layout {
display: flex;
gap: 24px;
height: 100%;
}
.sidebar {
width: 280px;
flex-shrink: 0;
padding-right: 16px;
height: fit-content;
max-height: calc(100vh - 200px);
overflow: hidden;
display: flex;
flex-direction: column;
}
.main-content {
flex: 1;
min-width: 0;
}
.system-prompt-content {
max-height: 400px;
overflow: auto;
padding: 12px;
border-radius: 8px;
font-size: 14px;
line-height: 1.5;
white-space: pre-wrap;
word-break: break-word;
background: rgba(var(--v-theme-surface-variant), 0.3);
}
.dialog-content {
background-color: rgba(var(--v-theme-surface-variant), 0.3);
padding: 8px 12px;
border-radius: 8px;
font-size: 14px;
line-height: 1.4;
margin-bottom: 8px;
white-space: pre-wrap;
word-break: break-word;
}
@media (max-width: 960px) {
.manager-layout {
flex-direction: column;
}
.sidebar {
display: none;
}
}
</style>
+23
View File
@@ -0,0 +1,23 @@
/**
* Persona
*
* 使 dashboard/src/components/folder
* personaStore
*/
// 主组件
export { default as PersonaManager } from './PersonaManager.vue';
// 文件夹相关组件
export { default as FolderTree } from './FolderTree.vue';
export { default as FolderTreeNode } from './FolderTreeNode.vue';
export { default as FolderBreadcrumb } from './FolderBreadcrumb.vue';
export { default as FolderCard } from './FolderCard.vue';
// 对话框组件
export { default as CreateFolderDialog } from './CreateFolderDialog.vue';
export { default as MoveToFolderDialog } from './MoveToFolderDialog.vue';
export { default as MoveTargetNode } from './MoveTargetNode.vue';
// Persona 相关组件
export { default as PersonaCard } from './PersonaCard.vue';