| | <script lang="ts"> |
| | import { marked } from 'marked'; |
| | import { toast } from 'svelte-sonner'; |
| | import fileSaver from 'file-saver'; |
| | |
| | const { saveAs } = fileSaver; |
| | |
| | import dayjs from '$lib/dayjs'; |
| | import duration from 'dayjs/plugin/duration'; |
| | import relativeTime from 'dayjs/plugin/relativeTime'; |
| | |
| | dayjs.extend(duration); |
| | dayjs.extend(relativeTime); |
| | |
| | async function loadLocale(locales) { |
| | for (const locale of locales) { |
| | try { |
| | dayjs.locale(locale); |
| | break; // Stop after successfully loading the first available locale |
| | } catch (error) { |
| | console.error(`Could not load locale '${locale}':`, error); |
| | } |
| | } |
| | } |
| | |
| | import { onMount, getContext, onDestroy } from 'svelte'; |
| | |
| | const i18n = getContext('i18n'); |
| | |
| | $: loadLocale($i18n.languages); |
| | |
| | import { goto } from '$app/navigation'; |
| | import { WEBUI_NAME, config, prompts as _prompts, user } from '$lib/stores'; |
| | import { createNewNote, deleteNoteById, getNoteList, searchNotes } from '$lib/apis/notes'; |
| | import { capitalizeFirstLetter, copyToClipboard, getTimeRange } from '$lib/utils'; |
| | import { downloadPdf, createNoteHandler } from './utils'; |
| | |
| | import EllipsisHorizontal from '../icons/EllipsisHorizontal.svelte'; |
| | import DeleteConfirmDialog from '$lib/components/common/ConfirmDialog.svelte'; |
| | import Search from '../icons/Search.svelte'; |
| | import Plus from '../icons/Plus.svelte'; |
| | import ChevronRight from '../icons/ChevronRight.svelte'; |
| | import Spinner from '../common/Spinner.svelte'; |
| | import Tooltip from '../common/Tooltip.svelte'; |
| | import NoteMenu from './Notes/NoteMenu.svelte'; |
| | import FilesOverlay from '../chat/MessageInput/FilesOverlay.svelte'; |
| | import XMark from '../icons/XMark.svelte'; |
| | import DropdownOptions from '../common/DropdownOptions.svelte'; |
| | import Loader from '../common/Loader.svelte'; |
| | |
| | let loaded = false; |
| | |
| | let importFiles = ''; |
| | let selectedNote = null; |
| | let showDeleteConfirm = false; |
| | |
| | let notes = {}; |
| | |
| | let items = null; |
| | let total = null; |
| | |
| | let query = ''; |
| | let searchDebounceTimer: ReturnType<typeof setTimeout>; |
| | |
| | let sortKey = null; |
| | let displayOption = null; |
| | let viewOption = null; |
| | let permission = null; |
| | |
| | let page = 1; |
| | |
| | let itemsLoading = false; |
| | let allItemsLoaded = false; |
| | |
| | const downloadHandler = async (type) => { |
| | if (type === 'txt') { |
| | const blob = new Blob([selectedNote.data.content.md], { type: 'text/plain' }); |
| | saveAs(blob, `${selectedNote.title}.txt`); |
| | } else if (type === 'md') { |
| | const blob = new Blob([selectedNote.data.content.md], { type: 'text/markdown' }); |
| | saveAs(blob, `${selectedNote.title}.md`); |
| | } else if (type === 'pdf') { |
| | try { |
| | await downloadPdf(selectedNote); |
| | } catch (error) { |
| | toast.error(`${error}`); |
| | } |
| | } |
| | }; |
| | |
| | const deleteNoteHandler = async (id) => { |
| | const res = await deleteNoteById(localStorage.token, id).catch((error) => { |
| | toast.error(`${error}`); |
| | return null; |
| | }); |
| | |
| | if (res) { |
| | init(); |
| | } |
| | }; |
| | |
| | const inputFilesHandler = async (inputFiles) => { |
| | // Check if all the file is a markdown file and extract name and content |
| | |
| | for (const file of inputFiles) { |
| | if (file.type !== 'text/markdown') { |
| | toast.error($i18n.t('Only markdown files are allowed')); |
| | return; |
| | } |
| | |
| | const reader = new FileReader(); |
| | reader.onload = async (event) => { |
| | const content = event.target.result; |
| | let name = file.name.replace(/\.md$/, ''); |
| | |
| | if (typeof content !== 'string') { |
| | toast.error($i18n.t('Invalid file content')); |
| | return; |
| | } |
| | |
| | |
| | const res = await createNewNote(localStorage.token, { |
| | title: name, |
| | data: { |
| | content: { |
| | json: null, |
| | html: marked.parse(content ?? ''), |
| | md: content |
| | } |
| | }, |
| | meta: null, |
| | access_grants: [] |
| | }).catch((error) => { |
| | toast.error(`${error}`); |
| | return null; |
| | }); |
| | |
| | if (res) { |
| | init(); |
| | } |
| | }; |
| | |
| | reader.readAsText(file); |
| | } |
| | }; |
| | |
| | const reset = () => { |
| | page = 1; |
| | items = null; |
| | total = null; |
| | allItemsLoaded = false; |
| | itemsLoading = false; |
| | notes = {}; |
| | }; |
| | |
| | const loadMoreItems = async () => { |
| | if (allItemsLoaded) return; |
| | page += 1; |
| | await getItemsPage(); |
| | }; |
| | |
| | const init = async () => { |
| | reset(); |
| | await getItemsPage(); |
| | }; |
| | |
| | $: if (query !== undefined) { |
| | clearTimeout(searchDebounceTimer); |
| | searchDebounceTimer = setTimeout(() => { |
| | if (loaded) { |
| | init(); |
| | } |
| | }, 300); |
| | } |
| | |
| | $: if (loaded && sortKey !== undefined && permission !== undefined && viewOption !== undefined) { |
| | init(); |
| | } |
| | |
| | const getItemsPage = async () => { |
| | itemsLoading = true; |
| | |
| | if (viewOption === 'created') { |
| | permission = null; |
| | } |
| | |
| | const res = await searchNotes( |
| | localStorage.token, |
| | query, |
| | viewOption, |
| | permission, |
| | sortKey, |
| | page |
| | ).catch(() => { |
| | return []; |
| | }); |
| | |
| | if (res) { |
| | console.log(res); |
| | total = res.total; |
| | const pageItems = res.items; |
| | |
| | if ((pageItems ?? []).length === 0) { |
| | allItemsLoaded = true; |
| | } else { |
| | allItemsLoaded = false; |
| | } |
| | |
| | if (items) { |
| | items = [...items, ...pageItems]; |
| | } else { |
| | items = pageItems; |
| | } |
| | } |
| | |
| | itemsLoading = false; |
| | return res; |
| | }; |
| | |
| | const groupNotes = (res) => { |
| | if (!Array.isArray(res)) { |
| | return []; // Return empty array for invalid input |
| | } |
| | |
| | |
| | const grouped: Record<string, any[]> = {}; |
| | const orderedKeys: string[] = []; |
| | |
| | for (const note of res) { |
| | const timeRange = getTimeRange(note.updated_at / 1000000000); |
| | if (!grouped[timeRange]) { |
| | grouped[timeRange] = []; |
| | orderedKeys.push(timeRange); |
| | } |
| | grouped[timeRange].push({ |
| | ...note, |
| | timeRange |
| | }); |
| | } |
| | |
| | |
| | return orderedKeys.map((key) => [key, grouped[key]] as [string, any[]]); |
| | }; |
| | |
| | let dragged = false; |
| | |
| | const onDragOver = (e) => { |
| | e.preventDefault(); |
| | |
| | // Check if a file is being dragged. |
| | if (e.dataTransfer?.types?.includes('Files')) { |
| | dragged = true; |
| | } else { |
| | dragged = false; |
| | } |
| | }; |
| | |
| | const onDragLeave = () => { |
| | dragged = false; |
| | }; |
| | |
| | const onDrop = async (e) => { |
| | e.preventDefault(); |
| | console.log(e); |
| | |
| | if (e.dataTransfer?.files) { |
| | const inputFiles = Array.from(e.dataTransfer?.files); |
| | if (inputFiles && inputFiles.length > 0) { |
| | console.log(inputFiles); |
| | inputFilesHandler(inputFiles); |
| | } |
| | } |
| | |
| | dragged = false; |
| | }; |
| | |
| | onMount(async () => { |
| | viewOption = localStorage?.noteViewOption ?? null; |
| | displayOption = localStorage?.noteDisplayOption ?? null; |
| | |
| | loaded = true; |
| | |
| | const dropzoneElement = document.getElementById('notes-container'); |
| | dropzoneElement?.addEventListener('dragover', onDragOver); |
| | dropzoneElement?.addEventListener('drop', onDrop); |
| | dropzoneElement?.addEventListener('dragleave', onDragLeave); |
| | }); |
| | |
| | onDestroy(() => { |
| | clearTimeout(searchDebounceTimer); |
| | console.log('destroy'); |
| | const dropzoneElement = document.getElementById('notes-container'); |
| | |
| | if (dropzoneElement) { |
| | dropzoneElement?.removeEventListener('dragover', onDragOver); |
| | dropzoneElement?.removeEventListener('drop', onDrop); |
| | dropzoneElement?.removeEventListener('dragleave', onDragLeave); |
| | } |
| | }); |
| | </script> |
| |
|
| | <svelte:head> |
| | <title> |
| | {$i18n.t('Notes')} • {$WEBUI_NAME} |
| | </title> |
| | </svelte:head> |
| |
|
| | <FilesOverlay show={dragged} /> |
| |
|
| | <div id="notes-container" class="w-full min-h-full h-full px-3 md:px-[18px]"> |
| | {#if loaded} |
| | <DeleteConfirmDialog |
| | bind:show={showDeleteConfirm} |
| | title={$i18n.t('Delete note?')} |
| | on:confirm={() => { |
| | deleteNoteHandler(selectedNote.id); |
| | showDeleteConfirm = false; |
| | }} |
| | > |
| | <div class=" text-sm text-gray-500 truncate"> |
| | {$i18n.t('This will delete')} <span class=" font-semibold">{selectedNote.title}</span>. |
| | </div> |
| | </DeleteConfirmDialog> |
| |
|
| | <div class="flex flex-col gap-1 px-1 mt-1.5 mb-3"> |
| | <div class="flex justify-between items-center"> |
| | <div class="flex items-center md:self-center text-xl font-medium px-0.5 gap-2 shrink-0"> |
| | <div> |
| | {$i18n.t('Notes')} |
| | </div> |
| | |
| | <div class="text-lg font-medium text-gray-500 dark:text-gray-500"> |
| | {total} |
| | </div> |
| | </div> |
| | |
| | <div class="flex w-full justify-end gap-1.5"> |
| | <button |
| | class=" px-2 py-1.5 rounded-xl bg-black text-white dark:bg-white dark:text-black transition font-medium text-sm flex items-center" |
| | on:click={async () => { |
| | const res = await createNoteHandler(dayjs().format('YYYY-MM-DD')); |
| | |
| | if (res) { |
| | goto(`/notes/${res.id}`); |
| | } |
| | }} |
| | > |
| | <Plus className="size-3" strokeWidth="2.5" /> |
| | |
| | <div class=" ml-1 text-xs">{$i18n.t('New Note')}</div> |
| | </button> |
| | </div> |
| | </div> |
| | </div> |
| |
|
| | <div |
| | class="py-2 bg-white dark:bg-gray-900 rounded-3xl border border-gray-100/30 dark:border-gray-850/30" |
| | > |
| | <div class="px-3.5 flex flex-1 items-center w-full space-x-2 py-0.5 pb-2"> |
| | <div class="flex flex-1 items-center"> |
| | <div class=" self-center ml-1 mr-3"> |
| | <Search className="size-3.5" /> |
| | </div> |
| | <input |
| | class=" w-full text-sm py-1 rounded-r-xl outline-hidden bg-transparent" |
| | bind:value={query} |
| | placeholder={$i18n.t('Search Notes')} |
| | /> |
| | |
| | {#if query} |
| | <div class="self-center pl-1.5 translate-y-[0.5px] rounded-l-xl bg-transparent"> |
| | <button |
| | class="p-0.5 rounded-full hover:bg-gray-100 dark:hover:bg-gray-900 transition" |
| | on:click={() => { |
| | query = ''; |
| | }} |
| | > |
| | <XMark className="size-3" strokeWidth="2" /> |
| | </button> |
| | </div> |
| | {/if} |
| | </div> |
| | </div> |
| |
|
| | <div class="px-3 flex justify-between"> |
| | <div |
| | class="flex w-full bg-transparent overflow-x-auto scrollbar-none" |
| | on:wheel={(e) => { |
| | if (e.deltaY !== 0) { |
| | e.preventDefault(); |
| | e.currentTarget.scrollLeft += e.deltaY; |
| | } |
| | }} |
| | > |
| | <div |
| | class="flex gap-3 w-fit text-center text-sm rounded-full bg-transparent px-0.5 whitespace-nowrap" |
| | > |
| | <DropdownOptions |
| | align="start" |
| | className="flex w-full items-center gap-2 truncate px-3 py-1.5 text-sm bg-gray-50 dark:bg-gray-850 rounded-xl placeholder-gray-400 outline-hidden focus:outline-hidden" |
| | bind:value={viewOption} |
| | items={[ |
| | { value: null, label: $i18n.t('All') }, |
| | { value: 'created', label: $i18n.t('Created by you') }, |
| | { value: 'shared', label: $i18n.t('Shared with you') } |
| | ]} |
| | onChange={(value) => { |
| | if (value) { |
| | localStorage.noteViewOption = value; |
| | } else { |
| | delete localStorage.noteViewOption; |
| | } |
| | }} |
| | /> |
| | |
| | {#if [null, 'shared'].includes(viewOption)} |
| | <DropdownOptions |
| | align="start" |
| | bind:value={permission} |
| | items={[ |
| | { value: null, label: $i18n.t('Write') }, |
| | { value: 'read_only', label: $i18n.t('Read Only') } |
| | ]} |
| | /> |
| | {/if} |
| | </div> |
| | </div> |
| |
|
| | <div> |
| | <DropdownOptions |
| | align="start" |
| | bind:value={displayOption} |
| | items={[ |
| | { value: null, label: $i18n.t('List') }, |
| | { value: 'grid', label: $i18n.t('Grid') } |
| | ]} |
| | onChange={() => { |
| | if (displayOption) { |
| | localStorage.noteDisplayOption = displayOption; |
| | } else { |
| | delete localStorage.noteDisplayOption; |
| | } |
| | }} |
| | /> |
| | </div> |
| | </div> |
| |
|
| | {#if items !== null && total !== null} |
| | {#if (items ?? []).length > 0} |
| | {@const groupedNotes = groupNotes(items)} |
| |
|
| | <div class="@container h-full py-2.5 px-2.5"> |
| | <div class=""> |
| | {#each groupedNotes as [timeRange, notesList], idx} |
| | <div |
| | class="w-full text-xs text-gray-500 dark:text-gray-500 font-medium px-2.5 pb-2.5" |
| | > |
| | {$i18n.t(timeRange)} |
| | </div> |
| | |
| | {#if displayOption === null} |
| | <div |
| | class="{groupedNotes.length - 1 !== idx ? 'mb-3' : ''} gap-1.5 flex flex-col" |
| | > |
| | {#each notesList as note, idx (note.id)} |
| | <div |
| | class=" flex cursor-pointer w-full px-3.5 py-1.5 border border-gray-50 dark:border-gray-850/30 bg-transparent dark:hover:bg-gray-850 hover:bg-white rounded-2xl transition" |
| | > |
| | <a href={`/notes/${note.id}`} class="w-full flex flex-col justify-between"> |
| | <div class="flex-1"> |
| | <div class=" flex items-center gap-2 self-center justify-between"> |
| | <Tooltip |
| | content={note.title} |
| | className="flex-1" |
| | placement="top-start" |
| | > |
| | <div |
| | class=" text-sm font-medium capitalize flex-1 w-full line-clamp-1" |
| | > |
| | {note.title} |
| | </div> |
| | </Tooltip> |
| | |
| | <div class="flex shrink-0 items-center text-xs gap-2.5"> |
| | <Tooltip content={dayjs(note.updated_at / 1000000).format('LLLL')}> |
| | <div> |
| | {dayjs(note.updated_at / 1000000).fromNow()} |
| | </div> |
| | </Tooltip> |
| | <Tooltip |
| | content={note?.user?.email ?? $i18n.t('Deleted User')} |
| | className="flex shrink-0" |
| | placement="top-start" |
| | > |
| | <div class="shrink-0 text-gray-500"> |
| | {$i18n.t('By {{name}}', { |
| | name: capitalizeFirstLetter( |
| | note?.user?.name ?? |
| | note?.user?.email ?? |
| | $i18n.t('Deleted User') |
| | ) |
| | })} |
| | </div> |
| | </Tooltip> |
| |
|
| | <div> |
| | <NoteMenu |
| | onDownload={(type) => { |
| | selectedNote = note; |
| | |
| | downloadHandler(type); |
| | }} |
| | onCopyLink={async () => { |
| | const baseUrl = window.location.origin; |
| | const res = await copyToClipboard( |
| | `${baseUrl}/notes/${note.id}` |
| | ); |
| | |
| | if (res) { |
| | toast.success($i18n.t('Copied link to clipboard')); |
| | } else { |
| | toast.error($i18n.t('Failed to copy link')); |
| | } |
| | }} |
| | onDelete={() => { |
| | selectedNote = note; |
| | showDeleteConfirm = true; |
| | }} |
| | > |
| | <button |
| | class="self-center w-fit text-sm p-1 dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-white/5 rounded-xl" |
| | type="button" |
| | > |
| | <EllipsisHorizontal className="size-5" /> |
| | </button> |
| | </NoteMenu> |
| | </div> |
| | </div> |
| | </div> |
| | </div> |
| | </a> |
| | </div> |
| | {/each} |
| | </div> |
| | {:else if displayOption === 'grid'} |
| | <div |
| | class="{groupedNotes.length - 1 !== idx |
| | ? 'mb-5' |
| | : ''} gap-2.5 grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 2xl:grid-cols-5" |
| | > |
| | {#each notesList as note, idx (note.id)} |
| | <div |
| | class=" flex space-x-4 cursor-pointer w-full px-4.5 py-4 border border-gray-50 dark:border-gray-850/30 bg-transparent dark:hover:bg-gray-850 hover:bg-white rounded-2xl transition" |
| | > |
| | <div class=" flex flex-1 space-x-4 cursor-pointer w-full"> |
| | <a |
| | href={`/notes/${note.id}`} |
| | class="w-full -translate-y-0.5 flex flex-col justify-between" |
| | > |
| | <div class="flex-1"> |
| | <div |
| | class=" flex items-center gap-2 self-center mb-1 justify-between" |
| | > |
| | <div class=" font-semibold line-clamp-1 capitalize"> |
| | {note.title} |
| | </div> |
| | |
| | <div> |
| | <NoteMenu |
| | onDownload={(type) => { |
| | selectedNote = note; |
| | |
| | downloadHandler(type); |
| | }} |
| | onCopyLink={async () => { |
| | const baseUrl = window.location.origin; |
| | const res = await copyToClipboard( |
| | `${baseUrl}/notes/${note.id}` |
| | ); |
| | |
| | if (res) { |
| | toast.success($i18n.t('Copied link to clipboard')); |
| | } else { |
| | toast.error($i18n.t('Failed to copy link')); |
| | } |
| | }} |
| | onDelete={() => { |
| | selectedNote = note; |
| | showDeleteConfirm = true; |
| | }} |
| | > |
| | <button |
| | class="self-center w-fit text-sm p-1 dark:text-gray-300 dark:hover:text-white hover:bg-black/5 dark:hover:bg-white/5 rounded-xl" |
| | type="button" |
| | > |
| | <EllipsisHorizontal className="size-5" /> |
| | </button> |
| | </NoteMenu> |
| | </div> |
| | </div> |
| |
|
| | <div |
| | class=" text-xs text-gray-500 dark:text-gray-500 mb-3 line-clamp-3 min-h-10" |
| | > |
| | {#if note.data?.content?.md} |
| | {note.data?.content?.md} |
| | {:else} |
| | {$i18n.t('No content')} |
| | {/if} |
| | </div> |
| | </div> |
| |
|
| | <div class=" text-xs px-0.5 w-full flex justify-between items-center"> |
| | <div> |
| | {dayjs(note.updated_at / 1000000).fromNow()} |
| | </div> |
| | <Tooltip |
| | content={note?.user?.email ?? $i18n.t('Deleted User')} |
| | className="flex shrink-0" |
| | placement="top-start" |
| | > |
| | <div class="shrink-0 text-gray-500"> |
| | {$i18n.t('By {{name}}', { |
| | name: capitalizeFirstLetter( |
| | note?.user?.name ?? |
| | note?.user?.email ?? |
| | $i18n.t('Deleted User') |
| | ) |
| | })} |
| | </div> |
| | </Tooltip> |
| | </div> |
| | </a> |
| | </div> |
| | </div> |
| | {/each} |
| | </div> |
| | {/if} |
| | {/each} |
| |
|
| | {#if !allItemsLoaded} |
| | <Loader |
| | on:visible={(e) => { |
| | if (!itemsLoading) { |
| | loadMoreItems(); |
| | } |
| | }} |
| | > |
| | <div |
| | class="w-full flex justify-center py-4 text-xs animate-pulse items-center gap-2" |
| | > |
| | <Spinner className=" size-4" /> |
| | <div class=" ">{$i18n.t('Loading...')}</div> |
| | </div> |
| | </Loader> |
| | {/if} |
| | </div> |
| | </div> |
| | {:else} |
| | <div class="w-full h-full flex flex-col items-center justify-center"> |
| | <div class="py-20 text-center"> |
| | <div class=" text-sm text-gray-400 dark:text-gray-600"> |
| | {$i18n.t('No Notes')} |
| | </div> |
| | |
| | <div class="mt-1 text-xs text-gray-300 dark:text-gray-700"> |
| | {$i18n.t('Create your first note by clicking on the plus button below.')} |
| | </div> |
| | </div> |
| | </div> |
| | {/if} |
| | {:else} |
| | <div class="w-full h-full flex justify-center items-center py-10"> |
| | <Spinner className="size-4" /> |
| | </div> |
| | {/if} |
| | </div> |
| | {:else} |
| | <div class="w-full h-full flex justify-center items-center"> |
| | <Spinner className="size-4" /> |
| | </div> |
| | {/if} |
| | </div> |
| |
|