open-webui/src/lib/components/common/FileItemModal.svelte

484 lines
14 KiB
Svelte
Raw Normal View History

2024-09-28 08:53:25 +00:00
<script lang="ts">
2025-12-20 14:12:03 +00:00
import * as XLSX from 'xlsx';
2025-08-06 15:50:51 +00:00
import { getContext, onMount, tick } from 'svelte';
2025-12-20 14:12:03 +00:00
2024-09-28 08:53:25 +00:00
import { formatFileSize, getLineCount } from '$lib/utils';
2025-02-20 07:44:11 +00:00
import { WEBUI_API_BASE_URL } from '$lib/constants';
2025-08-06 15:50:51 +00:00
import { getKnowledgeById } from '$lib/apis/knowledge';
2025-12-20 14:12:03 +00:00
import { getFileById } from '$lib/apis/files';
import CodeBlock from '$lib/components/chat/Messages/CodeBlock.svelte';
import Markdown from '$lib/components/chat/Messages/Markdown.svelte';
2024-09-28 08:53:25 +00:00
const i18n = getContext('i18n');
import Modal from './Modal.svelte';
import XMark from '../icons/XMark.svelte';
import Info from '../icons/Info.svelte';
2024-09-29 20:52:27 +00:00
import Switch from './Switch.svelte';
import Tooltip from './Tooltip.svelte';
import dayjs from 'dayjs';
2025-08-06 15:50:51 +00:00
import Spinner from './Spinner.svelte';
2024-09-28 08:53:25 +00:00
2024-10-04 05:22:22 +00:00
export let item;
2024-09-28 08:53:25 +00:00
export let show = false;
2024-09-29 20:52:27 +00:00
export let edit = false;
let enableFullContent = false;
2025-12-21 10:24:53 +00:00
let loading = false;
2025-05-04 13:29:59 +00:00
2025-12-21 10:24:53 +00:00
let isPDF = false;
2025-05-04 13:29:59 +00:00
let isAudio = false;
let isExcel = false;
2025-05-04 13:29:59 +00:00
2025-08-27 22:59:45 +00:00
let selectedTab = '';
let excelWorkbook: XLSX.WorkBook | null = null;
let excelSheetNames: string[] = [];
let selectedSheet = '';
let excelHtml = '';
let excelError = '';
let rowCount = 0;
2025-08-27 22:59:45 +00:00
2025-02-20 09:01:29 +00:00
$: isPDF =
item?.meta?.content_type === 'application/pdf' ||
(item?.name && item?.name.toLowerCase().endsWith('.pdf'));
2024-09-29 20:52:27 +00:00
$: isMarkdown =
item?.meta?.content_type === 'text/markdown' ||
(item?.name && item?.name.toLowerCase().endsWith('.md'));
$: isCode =
item?.name &&
(item.name.toLowerCase().endsWith('.py') ||
item.name.toLowerCase().endsWith('.js') ||
item.name.toLowerCase().endsWith('.ts') ||
item.name.toLowerCase().endsWith('.java') ||
item.name.toLowerCase().endsWith('.html') ||
item.name.toLowerCase().endsWith('.css') ||
item.name.toLowerCase().endsWith('.json') ||
item.name.toLowerCase().endsWith('.cpp') ||
item.name.toLowerCase().endsWith('.c') ||
item.name.toLowerCase().endsWith('.h') ||
item.name.toLowerCase().endsWith('.sh') ||
item.name.toLowerCase().endsWith('.bash') ||
item.name.toLowerCase().endsWith('.yaml') ||
item.name.toLowerCase().endsWith('.yml') ||
item.name.toLowerCase().endsWith('.xml') ||
item.name.toLowerCase().endsWith('.sql') ||
item.name.toLowerCase().endsWith('.go') ||
item.name.toLowerCase().endsWith('.rs') ||
item.name.toLowerCase().endsWith('.php') ||
item.name.toLowerCase().endsWith('.rb'));
2025-05-04 13:29:59 +00:00
$: isAudio =
2025-05-20 15:29:31 +00:00
(item?.meta?.content_type ?? '').startsWith('audio/') ||
2025-05-04 13:29:59 +00:00
(item?.name && item?.name.toLowerCase().endsWith('.mp3')) ||
(item?.name && item?.name.toLowerCase().endsWith('.wav')) ||
(item?.name && item?.name.toLowerCase().endsWith('.ogg')) ||
2025-05-04 14:17:35 +00:00
(item?.name && item?.name.toLowerCase().endsWith('.m4a')) ||
(item?.name && item?.name.toLowerCase().endsWith('.webm'));
2025-05-04 13:29:59 +00:00
$: isExcel =
item?.meta?.content_type === 'application/vnd.ms-excel' ||
item?.meta?.content_type ===
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' ||
item?.meta?.content_type === 'text/csv' ||
item?.meta?.content_type === 'application/csv' ||
(item?.name &&
(item.name.toLowerCase().endsWith('.xls') ||
item.name.toLowerCase().endsWith('.xlsx') ||
item.name.toLowerCase().endsWith('.csv')));
const loadExcelContent = async () => {
try {
excelError = '';
const response = await fetch(`${WEBUI_API_BASE_URL}/files/${item.id}/content`, {
headers: {
Authorization: `Bearer ${localStorage.token}`
}
});
if (!response.ok) {
throw new Error('Failed to fetch Excel file');
}
const arrayBuffer = await response.arrayBuffer();
excelWorkbook = XLSX.read(arrayBuffer, { type: 'array' });
excelSheetNames = excelWorkbook.SheetNames;
if (excelSheetNames.length > 0) {
selectedSheet = excelSheetNames[0];
renderExcelSheet();
}
} catch (error) {
console.error('Error loading Excel/CSV file:', error);
excelError = 'Failed to load Excel/CSV file. Please try downloading it instead.';
}
};
const renderExcelSheet = () => {
if (!excelWorkbook || !selectedSheet) return;
const worksheet = excelWorkbook.Sheets[selectedSheet];
// Calculate row count
const range = XLSX.utils.decode_range(worksheet['!ref'] || 'A1:A1');
rowCount = range.e.r - range.s.r + 1;
excelHtml = XLSX.utils.sheet_to_html(worksheet, {
id: 'excel-table',
editable: false,
header: ''
});
};
$: if (selectedSheet && excelWorkbook) {
renderExcelSheet();
}
2025-08-06 15:50:51 +00:00
const loadContent = async () => {
selectedTab = '';
2025-08-06 15:50:51 +00:00
if (item?.type === 'collection') {
loading = true;
const knowledge = await getKnowledgeById(localStorage.token, item.id).catch((e) => {
console.error('Error fetching knowledge base:', e);
return null;
});
if (knowledge) {
item.files = knowledge.files || [];
}
loading = false;
2025-08-25 14:22:28 +00:00
} else if (item?.type === 'file') {
loading = true;
const file = await getFileById(localStorage.token, item.id).catch((e) => {
console.error('Error fetching file:', e);
return null;
});
if (file) {
item.file = file || {};
}
// Load Excel content if it's an Excel file
if (isExcel) {
await loadExcelContent();
}
2025-08-25 14:22:28 +00:00
loading = false;
2025-08-06 15:50:51 +00:00
}
await tick();
};
$: if (show) {
loadContent();
}
2024-09-28 08:53:25 +00:00
onMount(() => {
2025-02-20 08:10:29 +00:00
console.log(item);
2024-10-04 05:22:22 +00:00
if (item?.context === 'full') {
2024-09-29 20:52:27 +00:00
enableFullContent = true;
}
2024-09-28 08:53:25 +00:00
});
</script>
2024-11-18 22:25:24 +00:00
<Modal bind:show size="lg">
2025-09-12 21:58:09 +00:00
<div class="font-primary px-4.5 py-3.5 w-full flex flex-col justify-center dark:text-gray-400">
2024-09-29 20:52:27 +00:00
<div class=" pb-2">
<div class="flex items-start justify-between">
<div>
<div class=" font-medium text-lg dark:text-gray-100">
<a
2025-02-20 07:44:11 +00:00
href="#"
2024-09-29 20:52:27 +00:00
class="hover:underline line-clamp-1"
2025-02-20 07:44:11 +00:00
on:click|preventDefault={() => {
if (!isPDF && item.url) {
2025-02-20 09:01:29 +00:00
window.open(
2025-12-21 19:17:53 +00:00
item.type === 'file'
? item?.url?.startsWith('http')
? item.url
: `${WEBUI_API_BASE_URL}/files/${item.url}/content`
: item.url,
2025-02-20 09:01:29 +00:00
'_blank'
);
2025-02-20 07:44:11 +00:00
}
}}
2024-09-29 20:52:27 +00:00
>
2024-10-04 05:22:22 +00:00
{item?.name ?? 'File'}
2024-09-29 20:52:27 +00:00
</a>
</div>
2024-09-28 08:53:25 +00:00
</div>
<div>
2024-09-29 20:52:27 +00:00
<button
on:click={() => {
show = false;
}}
>
<XMark />
</button>
</div>
</div>
<div>
<div class="flex flex-col items-center md:flex-row gap-1 justify-between w-full">
2025-08-27 22:59:45 +00:00
<div class=" flex flex-wrap text-xs gap-1 text-gray-500">
{#if item?.type === 'collection'}
{#if item?.type}
<div class="capitalize shrink-0">{item.type}</div>
{/if}
{#if item?.description}
<div class="line-clamp-1">{item.description}</div>
{/if}
{#if item?.created_at}
<div class="capitalize shrink-0">
{dayjs(item.created_at * 1000).format('LL')}
</div>
{/if}
{/if}
2024-10-04 05:22:22 +00:00
{#if item.size}
<div class="capitalize shrink-0">{formatFileSize(item.size)}</div>
2024-09-28 08:53:25 +00:00
{/if}
2024-10-04 05:22:22 +00:00
{#if item?.file?.data?.content}
2024-09-29 21:08:55 +00:00
<div class="capitalize shrink-0">
{#if isExcel && rowCount > 0 && selectedTab === 'preview'}
{$i18n.t('{{COUNT}} Rows', {
COUNT: rowCount
})}
{:else}
{$i18n.t('{{COUNT}} extracted lines', {
COUNT: getLineCount(item?.file?.data?.content ?? '')
})}
{/if}
2024-09-29 21:08:55 +00:00
</div>
2024-09-28 08:53:25 +00:00
2024-09-29 20:52:27 +00:00
<div class="flex items-center gap-1 shrink-0">
2025-08-27 23:01:53 +00:00
{$i18n.t('Formatting may be inconsistent from source.')}
2024-09-28 08:53:25 +00:00
</div>
{/if}
2025-07-14 13:47:21 +00:00
{#if item?.knowledge}
<div class="capitalize shrink-0">
{$i18n.t('Knowledge Base')}
</div>
{/if}
2024-09-28 08:53:25 +00:00
</div>
2024-09-29 20:52:27 +00:00
{#if edit}
2025-10-27 22:07:00 +00:00
<div class=" self-end">
2024-09-29 20:52:27 +00:00
<Tooltip
content={enableFullContent
2025-03-07 11:59:09 +00:00
? $i18n.t(
'Inject the entire content as context for comprehensive processing, this is recommended for complex queries.'
)
: $i18n.t(
'Default to segmented retrieval for focused and relevant content extraction, this is recommended for most cases.'
)}
2024-09-29 20:52:27 +00:00
>
<div class="flex items-center gap-1.5 text-xs">
{#if enableFullContent}
2025-07-17 12:25:26 +00:00
{$i18n.t('Using Entire Document')}
2024-09-29 20:52:27 +00:00
{:else}
2025-07-17 12:25:26 +00:00
{$i18n.t('Using Focused Retrieval')}
2024-09-29 20:52:27 +00:00
{/if}
<Switch
bind:state={enableFullContent}
on:change={(e) => {
2024-10-04 05:22:22 +00:00
item.context = e.detail ? 'full' : undefined;
2024-09-29 20:52:27 +00:00
}}
/>
</div>
</Tooltip>
</div>
{/if}
</div>
2024-09-28 08:53:25 +00:00
</div>
</div>
2025-02-20 07:44:11 +00:00
<div class="max-h-[75vh] overflow-auto">
2025-08-06 15:50:51 +00:00
{#if !loading}
{#if item?.type === 'collection'}
<div>
{#each item?.files as file}
<div class="flex items-center gap-2 mb-2">
<div class="flex-shrink-0 text-xs">
{file?.meta?.name}
</div>
</div>
2025-08-06 15:50:51 +00:00
{/each}
</div>
2025-12-21 10:24:53 +00:00
{/if}
{#if isAudio || isPDF || isExcel || isCode || isMarkdown}
2025-08-27 22:59:45 +00:00
<div
2025-11-30 08:56:12 +00:00
class="flex mb-2.5 scrollbar-none overflow-x-auto w-full border-b border-gray-50 dark:border-gray-850/30 text-center text-sm font-medium bg-transparent dark:text-gray-200"
2025-08-27 22:59:45 +00:00
>
<button
class="min-w-fit py-1.5 px-4 border-b {selectedTab === ''
? ' '
: ' border-transparent text-gray-300 dark:text-gray-600 hover:text-gray-700 dark:hover:text-white'} transition"
type="button"
on:click={() => {
selectedTab = '';
}}>{$i18n.t('Content')}</button
>
<button
class="min-w-fit py-1.5 px-4 border-b {selectedTab === 'preview'
? ' '
: ' border-transparent text-gray-300 dark:text-gray-600 hover:text-gray-700 dark:hover:text-white'} transition"
type="button"
on:click={() => {
selectedTab = 'preview';
}}>{$i18n.t('Preview')}</button
>
</div>
2025-12-21 10:24:53 +00:00
{/if}
2025-08-27 22:59:45 +00:00
2025-12-21 10:24:53 +00:00
{#if selectedTab === ''}
{#if item?.file?.data}
<div class="max-h-96 overflow-scroll scrollbar-hidden text-xs whitespace-pre-wrap">
{(item?.file?.data?.content ?? '').trim() || 'No content'}
</div>
{:else if item?.content}
<div class="max-h-96 overflow-scroll scrollbar-hidden text-xs whitespace-pre-wrap">
{(item?.content ?? '').trim() || 'No content'}
</div>
{/if}
{:else if selectedTab === 'preview'}
{#if isAudio}
<audio
src={`${WEBUI_API_BASE_URL}/files/${item.id}/content`}
class="w-full border-0 rounded-lg mb-2"
controls
playsinline
/>
{:else if isPDF}
2025-08-27 22:59:45 +00:00
<iframe
title={item?.name}
src={`${WEBUI_API_BASE_URL}/files/${item.id}/content`}
class="w-full h-[70vh] border-0 rounded-lg"
/>
2025-12-21 10:24:53 +00:00
{:else if isExcel}
2025-12-20 14:12:03 +00:00
{#if excelError}
<div class="text-red-500 text-sm p-4">
{excelError}
</div>
{:else}
2025-12-20 14:12:03 +00:00
{#if excelSheetNames.length > 1}
<div
class="flex mb-2.5 scrollbar-none overflow-x-auto w-full border-b border-gray-50 dark:border-gray-850/30 text-center text-sm font-medium bg-transparent dark:text-gray-200"
>
{#each excelSheetNames as sheetName}
<button
class="min-w-fit py-1.5 px-4 border-b {selectedSheet === sheetName
? ' '
: ' border-transparent text-gray-300 dark:text-gray-600 hover:text-gray-700 dark:hover:text-white'} transition"
type="button"
on:click={() => {
selectedSheet = sheetName;
}}>{sheetName}</button
>
{/each}
</div>
{/if}
{#if excelHtml}
<div class="excel-table-container overflow-auto max-h-[60vh]">
{@html excelHtml}
</div>
{:else}
<div class="text-gray-500 text-sm p-4">No content available</div>
{/if}
{/if}
2025-12-21 10:24:53 +00:00
{:else if isCode}
<div class="max-h-[60vh] overflow-scroll scrollbar-hidden text-sm relative">
<CodeBlock
code={item.file.data.content}
lang={item.name.split('.').pop()}
token={null}
edit={false}
run={false}
save={false}
/>
</div>
{:else if isMarkdown}
<div
class="max-h-[60vh] overflow-scroll scrollbar-hidden text-sm prose dark:prose-invert max-w-full"
>
<Markdown content={item.file.data.content} id="markdown-viewer" />
</div>
2025-12-20 14:12:03 +00:00
{:else}
<div class="max-h-96 overflow-scroll scrollbar-hidden text-xs whitespace-pre-wrap">
{(item?.file?.data?.content ?? '').trim() || 'No content'}
</div>
2025-08-06 15:50:51 +00:00
{/if}
2025-07-14 13:47:21 +00:00
{/if}
2025-08-06 15:50:51 +00:00
{:else}
<div class="flex items-center justify-center py-6">
<Spinner className="size-5" />
</div>
2025-02-20 07:44:11 +00:00
{/if}
2024-09-28 08:53:25 +00:00
</div>
</div>
</Modal>
<style>
:global(.excel-table-container table) {
width: 100%;
border-collapse: collapse;
font-size: 0.875rem;
line-height: 1.25rem;
}
:global(.excel-table-container table td),
:global(.excel-table-container table th) {
border-width: 1px;
border-style: solid;
border-color: var(--color-gray-300, #cdcdcd);
padding: 0.5rem 0.75rem;
text-align: left;
}
:global(.dark .excel-table-container table td),
:global(.dark .excel-table-container table th) {
border-color: var(--color-gray-600, #676767);
}
:global(.excel-table-container table th) {
background-color: var(--color-gray-100, #ececec);
font-weight: 600;
}
:global(.dark .excel-table-container table th) {
background-color: var(--color-gray-800, #333);
color: var(--color-gray-100, #ececec);
}
:global(.excel-table-container table tr:nth-child(even)) {
background-color: var(--color-gray-50, #f9f9f9);
}
:global(.dark .excel-table-container table tr:nth-child(even)) {
background-color: rgba(38, 38, 38, 0.5);
}
:global(.excel-table-container table tr:hover) {
background-color: var(--color-gray-100, #ececec);
}
:global(.dark .excel-table-container table tr:hover) {
background-color: rgba(51, 51, 51, 0.5);
}
</style>