feat: folders as projects

Co-Authored-By: Classic298 <27028174+Classic298@users.noreply.github.com>
This commit is contained in:
Timothy Jaeryang Baek 2025-07-13 03:23:05 +04:00
parent 5abc03f4dd
commit 7607c53bd5
6 changed files with 238 additions and 69 deletions

View file

@ -212,13 +212,13 @@ class FolderTable:
.first() .first()
) )
if existing_folder: if existing_folder and existing_folder.id != id:
return None return None
folder.name = form_data.get("name", folder.name) folder.name = form_data.get("name", folder.name)
if "data" in form_data: if "data" in form_data:
folder.data = { folder.data = {
**folder.data, **(folder.data or {}),
**form_data["data"], **form_data["data"],
} }

View file

@ -120,7 +120,7 @@ async def update_folder_name_by_id(
existing_folder = Folders.get_folder_by_parent_id_and_user_id_and_name( existing_folder = Folders.get_folder_by_parent_id_and_user_id_and_name(
folder.parent_id, user.id, form_data.name folder.parent_id, user.id, form_data.name
) )
if existing_folder: if existing_folder and existing_folder.id != id:
raise HTTPException( raise HTTPException(
status_code=status.HTTP_400_BAD_REQUEST, status_code=status.HTTP_400_BAD_REQUEST,
detail=ERROR_MESSAGES.DEFAULT("Folder already exists"), detail=ERROR_MESSAGES.DEFAULT("Folder already exists"),

View file

@ -0,0 +1,136 @@
<script lang="ts">
import { getContext, createEventDispatcher, onMount } from 'svelte';
import Spinner from '$lib/components/common/Spinner.svelte';
import Modal from '$lib/components/common/Modal.svelte';
import XMark from '$lib/components/icons/XMark.svelte';
import { toast } from 'svelte-sonner';
import { page } from '$app/stores';
import { goto } from '$app/navigation';
import Textarea from '$lib/components/common/Textarea.svelte';
import Knowledge from '$lib/components/workspace/Models/Knowledge.svelte';
const i18n = getContext('i18n');
export let show = false;
export let onSubmit: Function = (e) => {};
export let folder = null;
let name = '';
let data = {
system_prompt: '',
files: []
};
let loading = false;
const submitHandler = async () => {
loading = true;
await onSubmit({
name,
data
});
show = false;
loading = false;
};
const init = () => {
name = folder.name;
data = folder.data || {
system_prompt: '',
files: []
};
};
$: if (folder) {
init();
}
</script>
<Modal size="md" bind:show>
<div>
<div class=" flex justify-between dark:text-gray-300 px-5 pt-4 pb-1">
<div class=" text-lg font-medium self-center">
{$i18n.t('Edit Folder')}
</div>
<button
class="self-center"
on:click={() => {
show = false;
}}
>
<XMark className={'size-5'} />
</button>
</div>
<div class="flex flex-col md:flex-row w-full px-5 pb-4 md:space-x-4 dark:text-gray-200">
<div class=" flex flex-col w-full sm:flex-row sm:justify-center sm:space-x-6">
<form
class="flex flex-col w-full"
on:submit|preventDefault={() => {
submitHandler();
}}
>
<div class="flex flex-col w-full mt-1">
<div class=" mb-1 text-xs text-gray-500">{$i18n.t('Folder Name')}</div>
<div class="flex-1">
<input
class="w-full text-sm bg-transparent placeholder:text-gray-300 dark:placeholder:text-gray-700 outline-hidden"
type="text"
bind:value={name}
placeholder={$i18n.t('Enter folder name')}
autocomplete="off"
/>
</div>
</div>
<hr class=" border-gray-100 dark:border-gray-700/10 my-2.5 w-full" />
<div class="my-1">
<div class="mb-2 text-xs text-gray-500">{$i18n.t('System Prompt')}</div>
<div>
<Textarea
className=" text-sm w-full bg-transparent outline-hidden resize-none overflow-y-hidden "
placeholder={`Write your model system prompt content here\ne.g.) You are Mario from Super Mario Bros, acting as an assistant.`}
rows={4}
bind:value={data.system_prompt}
/>
</div>
</div>
<div class="my-2">
<Knowledge bind:selectedItems={data.files}>
<div slot="label">
<div class="flex w-full justify-between">
<div class=" mb-2 text-xs text-gray-500">
{$i18n.t('Knowledge')}
</div>
</div>
</div>
</Knowledge>
</div>
<div class="flex justify-end pt-3 text-sm font-medium gap-1.5">
<button
class="px-3.5 py-1.5 text-sm font-medium bg-black hover:bg-gray-950 text-white dark:bg-white dark:text-black dark:hover:bg-gray-100 transition rounded-full flex flex-row space-x-1 items-center {loading
? ' cursor-not-allowed'
: ''}"
type="submit"
disabled={loading}
>
{$i18n.t('Save')}
{#if loading}
<div class="ml-2 self-center">
<Spinner />
</div>
{/if}
</button>
</div>
</form>
</div>
</div>
</div>
</Modal>

View file

@ -8,6 +8,23 @@
import fileSaver from 'file-saver'; import fileSaver from 'file-saver';
const { saveAs } = fileSaver; const { saveAs } = fileSaver;
import { toast } from 'svelte-sonner';
import { selectedFolder } from '$lib/stores';
import {
deleteFolderById,
updateFolderIsExpandedById,
updateFolderById,
updateFolderParentIdById
} from '$lib/apis/folders';
import {
getChatById,
getChatsByFolderId,
importChat,
updateChatFolderIdById
} from '$lib/apis/chats';
import ChevronDown from '../../icons/ChevronDown.svelte'; import ChevronDown from '../../icons/ChevronDown.svelte';
import ChevronRight from '../../icons/ChevronRight.svelte'; import ChevronRight from '../../icons/ChevronRight.svelte';
import Collapsible from '../../common/Collapsible.svelte'; import Collapsible from '../../common/Collapsible.svelte';
@ -15,23 +32,11 @@
import FolderOpen from '$lib/components/icons/FolderOpen.svelte'; import FolderOpen from '$lib/components/icons/FolderOpen.svelte';
import EllipsisHorizontal from '$lib/components/icons/EllipsisHorizontal.svelte'; import EllipsisHorizontal from '$lib/components/icons/EllipsisHorizontal.svelte';
import {
deleteFolderById,
updateFolderIsExpandedById,
updateFolderById,
updateFolderParentIdById
} from '$lib/apis/folders';
import { toast } from 'svelte-sonner';
import {
getChatById,
getChatsByFolderId,
importChat,
updateChatFolderIdById
} from '$lib/apis/chats';
import ChatItem from './ChatItem.svelte'; import ChatItem from './ChatItem.svelte';
import FolderMenu from './Folders/FolderMenu.svelte'; import FolderMenu from './Folders/FolderMenu.svelte';
import DeleteConfirmDialog from '$lib/components/common/ConfirmDialog.svelte'; import DeleteConfirmDialog from '$lib/components/common/ConfirmDialog.svelte';
import { selectedFolder } from '$lib/stores'; import EditFolderModal from './Folders/EditFolderModal.svelte';
export let open = false; export let open = false;
@ -39,14 +44,13 @@
export let folderId; export let folderId;
export let shiftKey = false; export let shiftKey = false;
export let onCreateChat = (e) => {};
export let className = ''; export let className = '';
export let parentDragged = false; export let parentDragged = false;
let folderElement; let folderElement;
let showEditFolderModal = false;
let edit = false; let edit = false;
let draggedOver = false; let draggedOver = false;
@ -235,7 +239,7 @@
delete folders[folderId].new; delete folders[folderId].new;
await tick(); await tick();
editHandler(); renameHandler();
} }
}); });
@ -265,23 +269,21 @@
} }
}; };
const nameUpdateHandler = async () => { const updateHandler = async ({ name, data }) => {
if (name === '') { if (name === '') {
toast.error($i18n.t('Folder name cannot be empty.')); toast.error($i18n.t('Folder name cannot be empty.'));
return; return;
} }
if (name === folders[folderId].name) {
edit = false;
return;
}
const currentName = folders[folderId].name; const currentName = folders[folderId].name;
name = name.trim(); name = name.trim();
folders[folderId].name = name; folders[folderId].name = name;
const res = await updateFolderById(localStorage.token, folderId, { name }).catch((error) => { const res = await updateFolderById(localStorage.token, folderId, {
name,
...(data ? { data } : {})
}).catch((error) => {
toast.error(`${error}`); toast.error(`${error}`);
folders[folderId].name = currentName; folders[folderId].name = currentName;
@ -290,7 +292,12 @@
if (res) { if (res) {
folders[folderId].name = name; folders[folderId].name = name;
toast.success($i18n.t('Folder name updated successfully')); if (data) {
folders[folderId].data = data;
}
// toast.success($i18n.t('Folder name updated successfully'));
toast.success($i18n.t('Folder updated successfully'));
if ($selectedFolder?.id === folderId) { if ($selectedFolder?.id === folderId) {
selectedFolder.set(folders[folderId]); selectedFolder.set(folders[folderId]);
@ -320,7 +327,7 @@
$: isExpandedUpdateDebounceHandler(open); $: isExpandedUpdateDebounceHandler(open);
const editHandler = async () => { const renameHandler = async () => {
console.log('Edit'); console.log('Edit');
await tick(); await tick();
name = folders[folderId].name; name = folders[folderId].name;
@ -368,6 +375,12 @@
</div> </div>
</DeleteConfirmDialog> </DeleteConfirmDialog>
<EditFolderModal
bind:show={showEditFolderModal}
folder={folders[folderId]}
onSubmit={updateHandler}
/>
{#if dragged && x && y} {#if dragged && x && y}
<DragGhost {x} {y}> <DragGhost {x} {y}>
<div class=" bg-black/80 backdrop-blur-2xl px-2 py-1 rounded-lg w-fit max-w-40"> <div class=" bg-black/80 backdrop-blur-2xl px-2 py-1 rounded-lg w-fit max-w-40">
@ -407,7 +420,7 @@
? 'bg-gray-100 dark:bg-gray-900' ? 'bg-gray-100 dark:bg-gray-900'
: ''}" : ''}"
on:dblclick={() => { on:dblclick={() => {
editHandler(); renameHandler();
}} }}
on:click={(e) => { on:click={(e) => {
selectedFolder.set(folders[folderId]); selectedFolder.set(folders[folderId]);
@ -431,7 +444,7 @@
e.target.select(); e.target.select();
}} }}
on:blur={() => { on:blur={() => {
nameUpdateHandler(); updateHandler({ name });
edit = false; edit = false;
}} }}
on:click={(e) => { on:click={(e) => {
@ -444,7 +457,7 @@
}} }}
on:keydown={(e) => { on:keydown={(e) => {
if (e.key === 'Enter') { if (e.key === 'Enter') {
nameUpdateHandler(); updateHandler({ name });
edit = false; edit = false;
} }
}} }}
@ -464,10 +477,7 @@
> >
<FolderMenu <FolderMenu
onEdit={() => { onEdit={() => {
// Requires a timeout to prevent the click event from closing the dropdown showEditFolderModal = true;
setTimeout(() => {
editHandler();
}, 200);
}} }}
onDelete={() => { onDelete={() => {
showDeleteConfirm = true; showDeleteConfirm = true;

View file

@ -1,24 +1,42 @@
<script lang="ts"> <script lang="ts">
import { getContext } from 'svelte'; import { getContext, onMount } from 'svelte';
import { knowledge } from '$lib/stores';
import Selector from './Knowledge/Selector.svelte'; import Selector from './Knowledge/Selector.svelte';
import FileItem from '$lib/components/common/FileItem.svelte'; import FileItem from '$lib/components/common/FileItem.svelte';
import { getKnowledgeBases } from '$lib/apis/knowledge';
export let selectedItems = []; export let selectedItems = [];
const i18n = getContext('i18n'); const i18n = getContext('i18n');
let loaded = false;
onMount(async () => {
if (!$knowledge) {
knowledge.set(await getKnowledgeBases(localStorage.token));
}
loaded = true;
});
</script> </script>
<div> <div>
<div class="flex w-full justify-between mb-1"> <slot name="label">
<div class=" self-center text-sm font-semibold">{$i18n.t('Knowledge')}</div> <div class="mb-2">
</div> <div class="flex w-full justify-between mb-1">
<div class=" self-center text-sm font-semibold">
{$i18n.t('Knowledge')}
</div>
</div>
<div class=" text-xs dark:text-gray-500"> <div class=" text-xs dark:text-gray-500">
{$i18n.t('To attach knowledge base here, add them to the "Knowledge" workspace first.')} {$i18n.t('To attach knowledge base here, add them to the "Knowledge" workspace first.')}
</div> </div>
</div>
</slot>
<div class="flex flex-col"> <div class="flex flex-col">
{#if selectedItems?.length > 0} {#if selectedItems?.length > 0}
<div class=" flex flex-wrap items-center gap-2 mt-2"> <div class=" flex flex-wrap items-center gap-2 mb-2.5">
{#each selectedItems as file, fileIdx} {#each selectedItems as file, fileIdx}
<FileItem <FileItem
{file} {file}
@ -38,27 +56,30 @@
</div> </div>
{/if} {/if}
<div class="flex flex-wrap text-sm font-medium gap-1.5 mt-2"> {#if loaded}
<Selector <div class="flex flex-wrap text-sm font-medium gap-1.5">
on:select={(e) => { <Selector
const item = e.detail; knowledgeItems={$knowledge || []}
on:select={(e) => {
const item = e.detail;
if (!selectedItems.find((k) => k.id === item.id)) { if (!selectedItems.find((k) => k.id === item.id)) {
selectedItems = [ selectedItems = [
...selectedItems, ...selectedItems,
{ {
...item ...item
} }
]; ];
} }
}} }}
>
<button
class=" px-3.5 py-1.5 font-medium hover:bg-black/5 dark:hover:bg-white/5 outline outline-1 outline-gray-100 dark:outline-gray-850 rounded-3xl"
type="button">{$i18n.t('Select Knowledge')}</button
> >
</Selector> <button
</div> class=" px-3.5 py-1.5 font-medium hover:bg-black/5 dark:hover:bg-white/5 outline outline-1 outline-gray-100 dark:outline-gray-850 rounded-3xl"
type="button">{$i18n.t('Select Knowledge')}</button
>
</Selector>
</div>
{/if}
<!-- {knowledge} --> <!-- {knowledge} -->
</div> </div>
</div> </div>

View file

@ -15,6 +15,8 @@
export let onClose: Function = () => {}; export let onClose: Function = () => {};
export let knowledgeItems = [];
let query = ''; let query = '';
let items = []; let items = [];
@ -51,7 +53,7 @@
}; };
}); });
let legacy_documents = $knowledge let legacy_documents = knowledgeItems
.filter((item) => item?.meta?.document) .filter((item) => item?.meta?.document)
.map((item) => ({ .map((item) => ({
...item, ...item,
@ -86,16 +88,16 @@
] ]
: []; : [];
let collections = $knowledge let collections = knowledgeItems
.filter((item) => !item?.meta?.document) .filter((item) => !item?.meta?.document)
.map((item) => ({ .map((item) => ({
...item, ...item,
type: 'collection' type: 'collection'
})); }));
let collection_files = let collection_files =
$knowledge.length > 0 knowledgeItems.length > 0
? [ ? [
...$knowledge ...knowledgeItems
.reduce((a, item) => { .reduce((a, item) => {
return [ return [
...new Set([ ...new Set([
@ -141,7 +143,7 @@
<div slot="content"> <div slot="content">
<DropdownMenu.Content <DropdownMenu.Content
class="w-full max-w-96 rounded-xl px-1 py-1.5 border border-gray-300/30 dark:border-gray-700/50 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg" class="w-full max-w-96 rounded-xl px-1 py-1.5 border border-gray-300/30 dark:border-gray-700/50 z-[99999999] bg-white dark:bg-gray-850 dark:text-white shadow-lg"
sideOffset={8} sideOffset={8}
side="bottom" side="bottom"
align="start" align="start"
@ -162,7 +164,7 @@
<div class="max-h-56 overflow-y-scroll"> <div class="max-h-56 overflow-y-scroll">
{#if filteredItems.length === 0} {#if filteredItems.length === 0}
<div class="text-center text-sm text-gray-500 dark:text-gray-400"> <div class="text-center text-xs text-gray-500 dark:text-gray-400 py-4">
{$i18n.t('No knowledge found')} {$i18n.t('No knowledge found')}
</div> </div>
{:else} {:else}