| | <script lang="ts"> |
| | import { DropdownMenu } from 'bits-ui'; |
| | import { flyAndScale } from '$lib/utils/transitions'; |
| | import { getContext, createEventDispatcher, tick } from 'svelte'; |
| | |
| | import fileSaver from 'file-saver'; |
| | const { saveAs } = fileSaver; |
| | |
| | const dispatch = createEventDispatcher(); |
| | |
| | import Dropdown from '$lib/components/common/Dropdown.svelte'; |
| | import GarbageBin from '$lib/components/icons/GarbageBin.svelte'; |
| | import Pencil from '$lib/components/icons/Pencil.svelte'; |
| | import Tooltip from '$lib/components/common/Tooltip.svelte'; |
| | import Tags from '$lib/components/chat/Tags.svelte'; |
| | import Share from '$lib/components/icons/Share.svelte'; |
| | import ArchiveBox from '$lib/components/icons/ArchiveBox.svelte'; |
| | import DocumentDuplicate from '$lib/components/icons/DocumentDuplicate.svelte'; |
| | import Bookmark from '$lib/components/icons/Bookmark.svelte'; |
| | import BookmarkSlash from '$lib/components/icons/BookmarkSlash.svelte'; |
| | import { |
| | getChatById, |
| | getChatPinnedStatusById, |
| | toggleChatPinnedStatusById |
| | } from '$lib/apis/chats'; |
| | import { chats, folders, settings, theme, user } from '$lib/stores'; |
| | import { createMessagesList } from '$lib/utils'; |
| | import { downloadChatAsPDF } from '$lib/apis/utils'; |
| | import Download from '$lib/components/icons/Download.svelte'; |
| | import Folder from '$lib/components/icons/Folder.svelte'; |
| | import Messages from '$lib/components/chat/Messages.svelte'; |
| | |
| | const i18n = getContext('i18n'); |
| | |
| | export let shareHandler: Function; |
| | export let moveChatHandler: Function; |
| | |
| | export let cloneChatHandler: Function; |
| | export let archiveChatHandler: Function; |
| | export let renameHandler: Function; |
| | export let deleteHandler: Function; |
| | export let onClose: Function; |
| | |
| | export let chatId = ''; |
| | |
| | let show = false; |
| | let pinned = false; |
| | |
| | let chat = null; |
| | let showFullMessages = false; |
| | |
| | const pinHandler = async () => { |
| | await toggleChatPinnedStatusById(localStorage.token, chatId); |
| | dispatch('change'); |
| | }; |
| | |
| | const checkPinned = async () => { |
| | pinned = await getChatPinnedStatusById(localStorage.token, chatId); |
| | }; |
| | |
| | const getChatAsText = async (chat) => { |
| | const history = chat.chat.history; |
| | const messages = createMessagesList(history, history.currentId); |
| | const chatText = messages.reduce((a, message, i, arr) => { |
| | return `${a}### ${message.role.toUpperCase()}\n${message.content}\n\n`; |
| | }, ''); |
| | |
| | return chatText.trim(); |
| | }; |
| | |
| | const downloadTxt = async () => { |
| | const chat = await getChatById(localStorage.token, chatId); |
| | if (!chat) { |
| | return; |
| | } |
| | |
| | const chatText = await getChatAsText(chat); |
| | let blob = new Blob([chatText], { |
| | type: 'text/plain' |
| | }); |
| | |
| | saveAs(blob, `chat-${chat.chat.title}.txt`); |
| | }; |
| | |
| | const downloadPdf = async () => { |
| | chat = await getChatById(localStorage.token, chatId); |
| | if (!chat) { |
| | return; |
| | } |
| | |
| | const [{ default: jsPDF }, { default: html2canvas }] = await Promise.all([ |
| | import('jspdf'), |
| | import('html2canvas-pro') |
| | ]); |
| | |
| | if ($settings?.stylizedPdfExport ?? true) { |
| | showFullMessages = true; |
| | await tick(); |
| | |
| | const containerElement = document.getElementById('full-messages-container'); |
| | if (containerElement) { |
| | try { |
| | const isDarkMode = document.documentElement.classList.contains('dark'); |
| | const virtualWidth = 800; // px, fixed width for cloned element |
| | |
| | // Clone and style |
| | const clonedElement = containerElement.cloneNode(true); |
| | clonedElement.classList.add('text-black'); |
| | clonedElement.classList.add('dark:text-white'); |
| | clonedElement.style.width = `${virtualWidth}px`; |
| | clonedElement.style.position = 'absolute'; |
| | clonedElement.style.left = '-9999px'; |
| | clonedElement.style.height = 'auto'; |
| | document.body.appendChild(clonedElement); |
| | |
| | // Wait for DOM update/layout |
| | await new Promise((r) => setTimeout(r, 100)); |
| | |
| | // Render entire content once |
| | const canvas = await html2canvas(clonedElement, { |
| | backgroundColor: isDarkMode ? '#000' : '#fff', |
| | useCORS: true, |
| | scale: 2, // increase resolution |
| | width: virtualWidth |
| | }); |
| | |
| | document.body.removeChild(clonedElement); |
| | |
| | const pdf = new jsPDF('p', 'mm', 'a4'); |
| | const pageWidthMM = 210; |
| | const pageHeightMM = 297; |
| | |
| | |
| | |
| | const pxPerMM = canvas.width / virtualWidth; |
| | |
| | |
| | |
| | |
| | const pxPerPDFMM = canvas.width / pageWidthMM; |
| | |
| | |
| | const pagePixelHeight = Math.floor(pxPerPDFMM * pageHeightMM); |
| | |
| | let offsetY = 0; |
| | let page = 0; |
| | |
| | while (offsetY < canvas.height) { |
| | // Height of slice |
| | const sliceHeight = Math.min(pagePixelHeight, canvas.height - offsetY); |
| | |
| | // Create temp canvas for slice |
| | const pageCanvas = document.createElement('canvas'); |
| | pageCanvas.width = canvas.width; |
| | pageCanvas.height = sliceHeight; |
| | |
| | const ctx = pageCanvas.getContext('2d'); |
| | |
| | // Draw the slice of original canvas onto pageCanvas |
| | ctx.drawImage( |
| | canvas, |
| | 0, |
| | offsetY, |
| | canvas.width, |
| | sliceHeight, |
| | 0, |
| | 0, |
| | canvas.width, |
| | sliceHeight |
| | ); |
| | |
| | const imgData = pageCanvas.toDataURL('image/jpeg', 0.7); |
| | |
| | // Calculate image height in PDF units keeping aspect ratio |
| | const imgHeightMM = (sliceHeight * pageWidthMM) / canvas.width; |
| | |
| | if (page > 0) pdf.addPage(); |
| | |
| | if (isDarkMode) { |
| | pdf.setFillColor(0, 0, 0); |
| | pdf.rect(0, 0, pageWidthMM, pageHeightMM, 'F'); // black bg |
| | } |
| | |
| | pdf.addImage(imgData, 'JPEG', 0, 0, pageWidthMM, imgHeightMM); |
| | |
| | offsetY += sliceHeight; |
| | page++; |
| | } |
| | |
| | pdf.save(`chat-${chat.chat.title}.pdf`); |
| | |
| | showFullMessages = false; |
| | } catch (error) { |
| | console.error('Error generating PDF', error); |
| | } |
| | } |
| | } else { |
| | console.log('Downloading PDF'); |
| | |
| | const chatText = await getChatAsText(chat); |
| | |
| | const doc = new jsPDF(); |
| | |
| | // Margins |
| | const left = 15; |
| | const top = 20; |
| | const right = 15; |
| | const bottom = 20; |
| | |
| | const pageWidth = doc.internal.pageSize.getWidth(); |
| | const pageHeight = doc.internal.pageSize.getHeight(); |
| | const usableWidth = pageWidth - left - right; |
| | const usableHeight = pageHeight - top - bottom; |
| | |
| | // Font size and line height |
| | const fontSize = 8; |
| | doc.setFontSize(fontSize); |
| | const lineHeight = fontSize * 1; // adjust if needed |
| | |
| | // Split the markdown into lines (handles \n) |
| | const paragraphs = chatText.split('\n'); |
| | |
| | let y = top; |
| | |
| | for (let paragraph of paragraphs) { |
| | // Wrap each paragraph to fit the width |
| | const lines = doc.splitTextToSize(paragraph, usableWidth); |
| | |
| | for (let line of lines) { |
| | // If the line would overflow the bottom, add a new page |
| | if (y + lineHeight > pageHeight - bottom) { |
| | doc.addPage(); |
| | y = top; |
| | } |
| | doc.text(line, left, y); |
| | y += lineHeight * 0.5; |
| | } |
| | |
| | y += lineHeight * 0.1; |
| | } |
| | |
| | doc.save(`chat-${chat.chat.title}.pdf`); |
| | } |
| | }; |
| | |
| | const downloadJSONExport = async () => { |
| | const chat = await getChatById(localStorage.token, chatId); |
| | |
| | if (chat) { |
| | let blob = new Blob([JSON.stringify([chat])], { |
| | type: 'application/json' |
| | }); |
| | saveAs(blob, `chat-export-${Date.now()}.json`); |
| | } |
| | }; |
| | |
| | $: if (show) { |
| | checkPinned(); |
| | } |
| | </script> |
| |
|
| | {#if chat && showFullMessages} |
| | <div class="hidden w-full h-full flex-col"> |
| | <div id="full-messages-container"> |
| | <Messages |
| | className="h-full flex pt-4 pb-8 w-full" |
| | chatId={`chat-preview-${chat?.id ?? ''}`} |
| | user={$user} |
| | readOnly={true} |
| | history={chat.chat.history} |
| | messages={chat.chat.messages} |
| | autoScroll={true} |
| | sendMessage={() => {}} |
| | continueResponse={() => {}} |
| | regenerateResponse={() => {}} |
| | messagesCount={null} |
| | editCodeBlock={false} |
| | /> |
| | </div> |
| | </div> |
| | {/if} |
| |
|
| | <Dropdown |
| | bind:show |
| | on:change={(e) => { |
| | if (e.detail === false) { |
| | onClose(); |
| | } |
| | }} |
| | > |
| | <Tooltip content={$i18n.t('More')}> |
| | <slot /> |
| | </Tooltip> |
| |
|
| | <div slot="content"> |
| | <DropdownMenu.Content |
| | class="w-full max-w-[200px] rounded-2xl px-1 py-1 border border-gray-100 dark:border-gray-800 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg transition" |
| | sideOffset={-2} |
| | side="bottom" |
| | align="start" |
| | transition={flyAndScale} |
| | > |
| | {#if $user?.role === 'admin' || ($user.permissions?.chat?.share ?? true)} |
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | shareHandler(); |
| | }} |
| | > |
| | <Share strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Share')}</div> |
| | </DropdownMenu.Item> |
| | {/if} |
| |
|
| | <DropdownMenu.Sub> |
| | <DropdownMenu.SubTrigger |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | > |
| | <Download strokeWidth="1.5" /> |
| | |
| | <div class="flex items-center">{$i18n.t('Download')}</div> |
| | </DropdownMenu.SubTrigger> |
| | <DropdownMenu.SubContent |
| | class="w-full rounded-2xl p-1 z-50 bg-white dark:bg-gray-850 dark:text-white shadow-lg border border-gray-100 dark:border-gray-800" |
| | transition={flyAndScale} |
| | sideOffset={8} |
| | > |
| | {#if $user?.role === 'admin' || ($user.permissions?.chat?.export ?? true)} |
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | downloadJSONExport(); |
| | }} |
| | > |
| | <div class="flex items-center line-clamp-1">{$i18n.t('Export chat (.json)')}</div> |
| | </DropdownMenu.Item> |
| | {/if} |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | downloadTxt(); |
| | }} |
| | > |
| | <div class="flex items-center line-clamp-1">{$i18n.t('Plain text (.txt)')}</div> |
| | </DropdownMenu.Item> |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl select-none w-full" |
| | on:click={() => { |
| | downloadPdf(); |
| | }} |
| | > |
| | <div class="flex items-center line-clamp-1">{$i18n.t('PDF document (.pdf)')}</div> |
| | </DropdownMenu.Item> |
| | </DropdownMenu.SubContent> |
| | </DropdownMenu.Sub> |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | renameHandler(); |
| | }} |
| | > |
| | <Pencil strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Rename')}</div> |
| | </DropdownMenu.Item> |
| |
|
| | <hr class="border-gray-50/30 dark:border-gray-800/30 my-1" /> |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | pinHandler(); |
| | }} |
| | > |
| | {#if pinned} |
| | <BookmarkSlash strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Unpin')}</div> |
| | {:else} |
| | <Bookmark strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Pin')}</div> |
| | {/if} |
| | </DropdownMenu.Item> |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | cloneChatHandler(); |
| | }} |
| | > |
| | <DocumentDuplicate strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Clone')}</div> |
| | </DropdownMenu.Item> |
| |
|
| | {#if chatId && $folders.length > 0} |
| | <DropdownMenu.Sub> |
| | <DropdownMenu.SubTrigger |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl select-none w-full" |
| | > |
| | <Folder /> |
| | |
| | <div class="flex items-center">{$i18n.t('Move')}</div> |
| | </DropdownMenu.SubTrigger> |
| | <DropdownMenu.SubContent |
| | class="w-full rounded-2xl p-1 z-50 bg-white dark:bg-gray-850 dark:text-white border border-gray-100 dark:border-gray-800 shadow-lg max-h-52 overflow-y-auto scrollbar-hidden" |
| | transition={flyAndScale} |
| | sideOffset={8} |
| | > |
| | {#each $folders.sort((a, b) => b.updated_at - a.updated_at) as folder} |
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | moveChatHandler(chatId, folder.id); |
| | }} |
| | > |
| | <Folder /> |
| |
|
| | <div class="flex items-center">{folder?.name ?? 'Folder'}</div> |
| | </DropdownMenu.Item> |
| | {/each} |
| | </DropdownMenu.SubContent> |
| | </DropdownMenu.Sub> |
| | {/if} |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | archiveChatHandler(); |
| | }} |
| | > |
| | <ArchiveBox strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Archive')}</div> |
| | </DropdownMenu.Item> |
| |
|
| | <DropdownMenu.Item |
| | class="flex gap-2 items-center px-3 py-1.5 text-sm cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-800 rounded-xl" |
| | on:click={() => { |
| | deleteHandler(); |
| | }} |
| | > |
| | <GarbageBin strokeWidth="1.5" /> |
| | <div class="flex items-center">{$i18n.t('Delete')}</div> |
| | </DropdownMenu.Item> |
| | </DropdownMenu.Content> |
| | </div> |
| | </Dropdown> |
| |
|