Spaces:
Paused
Paused
| <script> | |
| import { getContext, createEventDispatcher, onMount, onDestroy, tick } from 'svelte'; | |
| const i18n = getContext('i18n'); | |
| const dispatch = createEventDispatcher(); | |
| import DOMPurify from 'dompurify'; | |
| import fileSaver from 'file-saver'; | |
| const { saveAs } = fileSaver; | |
| import ChevronDown from '../../icons/ChevronDown.svelte'; | |
| import ChevronRight from '../../icons/ChevronRight.svelte'; | |
| import Collapsible from '../../common/Collapsible.svelte'; | |
| import DragGhost from '$lib/components/common/DragGhost.svelte'; | |
| import FolderOpen from '$lib/components/icons/FolderOpen.svelte'; | |
| import EllipsisHorizontal from '$lib/components/icons/EllipsisHorizontal.svelte'; | |
| import { | |
| deleteFolderById, | |
| updateFolderIsExpandedById, | |
| updateFolderNameById, | |
| 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 FolderMenu from './Folders/FolderMenu.svelte'; | |
| import DeleteConfirmDialog from '$lib/components/common/ConfirmDialog.svelte'; | |
| export let open = false; | |
| export let folders; | |
| export let folderId; | |
| export let className = ''; | |
| export let parentDragged = false; | |
| let folderElement; | |
| let edit = false; | |
| let draggedOver = false; | |
| let dragged = false; | |
| let name = ''; | |
| const onDragOver = (e) => { | |
| e.preventDefault(); | |
| e.stopPropagation(); | |
| if (dragged || parentDragged) { | |
| return; | |
| } | |
| draggedOver = true; | |
| }; | |
| const onDrop = async (e) => { | |
| e.preventDefault(); | |
| e.stopPropagation(); | |
| if (dragged || parentDragged) { | |
| return; | |
| } | |
| if (folderElement.contains(e.target)) { | |
| console.log('Dropped on the Button'); | |
| if (e.dataTransfer.items && e.dataTransfer.items.length > 0) { | |
| // Iterate over all items in the DataTransferItemList use functional programming | |
| for (const item of Array.from(e.dataTransfer.items)) { | |
| // If dropped items aren't files, reject them | |
| if (item.kind === 'file') { | |
| const file = item.getAsFile(); | |
| if (file && file.type === 'application/json') { | |
| console.log('Dropped file is a JSON file!'); | |
| // Read the JSON file with FileReader | |
| const reader = new FileReader(); | |
| reader.onload = async function (event) { | |
| try { | |
| const fileContent = JSON.parse(event.target.result); | |
| open = true; | |
| dispatch('import', { | |
| folderId: folderId, | |
| items: fileContent | |
| }); | |
| } catch (error) { | |
| console.error('Error parsing JSON file:', error); | |
| } | |
| }; | |
| // Start reading the file | |
| reader.readAsText(file); | |
| } else { | |
| console.error('Only JSON file types are supported.'); | |
| } | |
| console.log(file); | |
| } else { | |
| // Handle the drag-and-drop data for folders or chats (same as before) | |
| const dataTransfer = e.dataTransfer.getData('text/plain'); | |
| const data = JSON.parse(dataTransfer); | |
| console.log(data); | |
| const { type, id, item } = data; | |
| if (type === 'folder') { | |
| open = true; | |
| if (id === folderId) { | |
| return; | |
| } | |
| // Move the folder | |
| const res = await updateFolderParentIdById(localStorage.token, id, folderId).catch( | |
| (error) => { | |
| toast.error(`${error}`); | |
| return null; | |
| } | |
| ); | |
| if (res) { | |
| dispatch('update'); | |
| } | |
| } else if (type === 'chat') { | |
| open = true; | |
| let chat = await getChatById(localStorage.token, id).catch((error) => { | |
| return null; | |
| }); | |
| if (!chat && item) { | |
| chat = await importChat(localStorage.token, item.chat, item?.meta ?? {}); | |
| } | |
| // Move the chat | |
| const res = await updateChatFolderIdById(localStorage.token, chat.id, folderId).catch( | |
| (error) => { | |
| toast.error(`${error}`); | |
| return null; | |
| } | |
| ); | |
| if (res) { | |
| dispatch('update'); | |
| } | |
| } | |
| } | |
| } | |
| } | |
| draggedOver = false; | |
| } | |
| }; | |
| const onDragLeave = (e) => { | |
| e.preventDefault(); | |
| if (dragged || parentDragged) { | |
| return; | |
| } | |
| draggedOver = false; | |
| }; | |
| const dragImage = new Image(); | |
| dragImage.src = | |
| ''; | |
| let x; | |
| let y; | |
| const onDragStart = (event) => { | |
| event.stopPropagation(); | |
| event.dataTransfer.setDragImage(dragImage, 0, 0); | |
| // Set the data to be transferred | |
| event.dataTransfer.setData( | |
| 'text/plain', | |
| JSON.stringify({ | |
| type: 'folder', | |
| id: folderId | |
| }) | |
| ); | |
| dragged = true; | |
| folderElement.style.opacity = '0.5'; // Optional: Visual cue to show it's being dragged | |
| }; | |
| const onDrag = (event) => { | |
| event.stopPropagation(); | |
| x = event.clientX; | |
| y = event.clientY; | |
| }; | |
| const onDragEnd = (event) => { | |
| event.stopPropagation(); | |
| folderElement.style.opacity = '1'; // Reset visual cue after drag | |
| dragged = false; | |
| }; | |
| onMount(async () => { | |
| open = folders[folderId].is_expanded; | |
| if (folderElement) { | |
| folderElement.addEventListener('dragover', onDragOver); | |
| folderElement.addEventListener('drop', onDrop); | |
| folderElement.addEventListener('dragleave', onDragLeave); | |
| // Event listener for when dragging starts | |
| folderElement.addEventListener('dragstart', onDragStart); | |
| // Event listener for when dragging occurs (optional) | |
| folderElement.addEventListener('drag', onDrag); | |
| // Event listener for when dragging ends | |
| folderElement.addEventListener('dragend', onDragEnd); | |
| } | |
| if (folders[folderId]?.new) { | |
| delete folders[folderId].new; | |
| await tick(); | |
| editHandler(); | |
| } | |
| }); | |
| onDestroy(() => { | |
| if (folderElement) { | |
| folderElement.addEventListener('dragover', onDragOver); | |
| folderElement.removeEventListener('drop', onDrop); | |
| folderElement.removeEventListener('dragleave', onDragLeave); | |
| folderElement.removeEventListener('dragstart', onDragStart); | |
| folderElement.removeEventListener('drag', onDrag); | |
| folderElement.removeEventListener('dragend', onDragEnd); | |
| } | |
| }); | |
| let showDeleteConfirm = false; | |
| const deleteHandler = async () => { | |
| const res = await deleteFolderById(localStorage.token, folderId).catch((error) => { | |
| toast.error(`${error}`); | |
| return null; | |
| }); | |
| if (res) { | |
| toast.success($i18n.t('Folder deleted successfully')); | |
| dispatch('update'); | |
| } | |
| }; | |
| const nameUpdateHandler = async () => { | |
| if (name === '') { | |
| toast.error($i18n.t('Folder name cannot be empty.')); | |
| return; | |
| } | |
| if (name === folders[folderId].name) { | |
| edit = false; | |
| return; | |
| } | |
| const currentName = folders[folderId].name; | |
| name = name.trim(); | |
| folders[folderId].name = name; | |
| const res = await updateFolderNameById(localStorage.token, folderId, name).catch((error) => { | |
| toast.error(`${error}`); | |
| folders[folderId].name = currentName; | |
| return null; | |
| }); | |
| if (res) { | |
| folders[folderId].name = name; | |
| toast.success($i18n.t('Folder name updated successfully')); | |
| dispatch('update'); | |
| } | |
| }; | |
| const isExpandedUpdateHandler = async () => { | |
| const res = await updateFolderIsExpandedById(localStorage.token, folderId, open).catch( | |
| (error) => { | |
| toast.error(`${error}`); | |
| return null; | |
| } | |
| ); | |
| }; | |
| let isExpandedUpdateTimeout; | |
| const isExpandedUpdateDebounceHandler = (open) => { | |
| clearTimeout(isExpandedUpdateTimeout); | |
| isExpandedUpdateTimeout = setTimeout(() => { | |
| isExpandedUpdateHandler(); | |
| }, 500); | |
| }; | |
| $: isExpandedUpdateDebounceHandler(open); | |
| const editHandler = async () => { | |
| console.log('Edit'); | |
| await tick(); | |
| name = folders[folderId].name; | |
| edit = true; | |
| await tick(); | |
| const input = document.getElementById(`folder-${folderId}-input`); | |
| if (input) { | |
| input.focus(); | |
| } | |
| }; | |
| const exportHandler = async () => { | |
| const chats = await getChatsByFolderId(localStorage.token, folderId).catch((error) => { | |
| toast.error(`${error}`); | |
| return null; | |
| }); | |
| if (!chats) { | |
| return; | |
| } | |
| const blob = new Blob([JSON.stringify(chats)], { | |
| type: 'application/json' | |
| }); | |
| saveAs(blob, `folder-${folders[folderId].name}-export-${Date.now()}.json`); | |
| }; | |
| </script> | |
| <DeleteConfirmDialog | |
| bind:show={showDeleteConfirm} | |
| title={$i18n.t('Delete folder?')} | |
| on:confirm={() => { | |
| deleteHandler(); | |
| }} | |
| > | |
| <div class=" text-sm text-gray-700 dark:text-gray-300 flex-1 line-clamp-3"> | |
| {@html DOMPurify.sanitize( | |
| $i18n.t('This will delete <strong>{{NAME}}</strong> and <strong>all its contents</strong>.', { | |
| NAME: folders[folderId].name | |
| }) | |
| )} | |
| </div> | |
| </DeleteConfirmDialog> | |
| {#if dragged && 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="flex items-center gap-1"> | |
| <FolderOpen className="size-3.5" strokeWidth="2" /> | |
| <div class=" text-xs text-white line-clamp-1"> | |
| {folders[folderId].name} | |
| </div> | |
| </div> | |
| </div> | |
| </DragGhost> | |
| {/if} | |
| <div bind:this={folderElement} class="relative {className}" draggable="true"> | |
| {#if draggedOver} | |
| <div | |
| class="absolute top-0 left-0 w-full h-full rounded-xs bg-gray-100/50 dark:bg-gray-700/20 bg-opacity-50 dark:bg-opacity-10 z-50 pointer-events-none touch-none" | |
| ></div> | |
| {/if} | |
| <Collapsible | |
| bind:open | |
| className="w-full" | |
| buttonClassName="w-full" | |
| hide={(folders[folderId]?.childrenIds ?? []).length === 0 && | |
| (folders[folderId].items?.chats ?? []).length === 0} | |
| onChange={(state) => { | |
| dispatch('open', state); | |
| }} | |
| > | |
| <!-- svelte-ignore a11y-no-static-element-interactions --> | |
| <div class="w-full group"> | |
| <button | |
| id="folder-{folderId}-button" | |
| class="relative w-full py-1.5 px-2 rounded-md flex items-center gap-1.5 text-xs text-gray-500 dark:text-gray-500 font-medium hover:bg-gray-100 dark:hover:bg-gray-900 transition" | |
| on:dblclick={() => { | |
| editHandler(); | |
| }} | |
| > | |
| <div class="text-gray-300 dark:text-gray-600"> | |
| {#if open} | |
| <ChevronDown className=" size-3" strokeWidth="2.5" /> | |
| {:else} | |
| <ChevronRight className=" size-3" strokeWidth="2.5" /> | |
| {/if} | |
| </div> | |
| <div class="translate-y-[0.5px] flex-1 justify-start text-start line-clamp-1"> | |
| {#if edit} | |
| <input | |
| id="folder-{folderId}-input" | |
| type="text" | |
| bind:value={name} | |
| on:focus={(e) => { | |
| e.target.select(); | |
| }} | |
| on:blur={() => { | |
| nameUpdateHandler(); | |
| edit = false; | |
| }} | |
| on:click={(e) => { | |
| // Prevent accidental collapse toggling when clicking inside input | |
| e.stopPropagation(); | |
| }} | |
| on:mousedown={(e) => { | |
| // Prevent accidental collapse toggling when clicking inside input | |
| e.stopPropagation(); | |
| }} | |
| on:keydown={(e) => { | |
| if (e.key === 'Enter') { | |
| nameUpdateHandler(); | |
| edit = false; | |
| } | |
| }} | |
| class="w-full h-full bg-transparent text-gray-500 dark:text-gray-500 outline-hidden" | |
| /> | |
| {:else} | |
| {folders[folderId].name} | |
| {/if} | |
| </div> | |
| <button | |
| class="absolute z-10 right-2 invisible group-hover:visible self-center flex items-center dark:text-gray-300" | |
| on:pointerup={(e) => { | |
| e.stopPropagation(); | |
| }} | |
| > | |
| <FolderMenu | |
| on:rename={() => { | |
| // Requires a timeout to prevent the click event from closing the dropdown | |
| setTimeout(() => { | |
| editHandler(); | |
| }, 200); | |
| }} | |
| on:delete={() => { | |
| showDeleteConfirm = true; | |
| }} | |
| on:export={() => { | |
| exportHandler(); | |
| }} | |
| > | |
| <button class="p-0.5 dark:hover:bg-gray-850 rounded-lg touch-auto" on:click={(e) => {}}> | |
| <EllipsisHorizontal className="size-4" strokeWidth="2.5" /> | |
| </button> | |
| </FolderMenu> | |
| </button> | |
| </button> | |
| </div> | |
| <div slot="content" class="w-full"> | |
| {#if (folders[folderId]?.childrenIds ?? []).length > 0 || (folders[folderId].items?.chats ?? []).length > 0} | |
| <div | |
| class="ml-3 pl-1 mt-[1px] flex flex-col overflow-y-auto scrollbar-hidden border-s border-gray-100 dark:border-gray-900" | |
| > | |
| {#if folders[folderId]?.childrenIds} | |
| {@const children = folders[folderId]?.childrenIds | |
| .map((id) => folders[id]) | |
| .sort((a, b) => | |
| a.name.localeCompare(b.name, undefined, { | |
| numeric: true, | |
| sensitivity: 'base' | |
| }) | |
| )} | |
| {#each children as childFolder (`${folderId}-${childFolder.id}`)} | |
| <svelte:self | |
| {folders} | |
| folderId={childFolder.id} | |
| parentDragged={dragged} | |
| on:import={(e) => { | |
| dispatch('import', e.detail); | |
| }} | |
| on:update={(e) => { | |
| dispatch('update', e.detail); | |
| }} | |
| on:change={(e) => { | |
| dispatch('change', e.detail); | |
| }} | |
| /> | |
| {/each} | |
| {/if} | |
| {#if folders[folderId].items?.chats} | |
| {#each folders[folderId].items.chats as chat (chat.id)} | |
| <ChatItem | |
| id={chat.id} | |
| title={chat.title} | |
| on:change={(e) => { | |
| dispatch('change', e.detail); | |
| }} | |
| /> | |
| {/each} | |
| {/if} | |
| </div> | |
| {/if} | |
| </div> | |
| </Collapsible> | |
| </div> | |