113 lines
4.4 KiB
Svelte
113 lines
4.4 KiB
Svelte
<!-- [DEF:FileList:Component] -->
|
|
<!--
|
|
@SEMANTICS: storage, files, list, table
|
|
@PURPOSE: Displays a table of files with metadata and actions.
|
|
@LAYER: Component
|
|
@RELATION: DEPENDS_ON -> storageService
|
|
|
|
@PROPS: files (Array) - List of StoredFile objects.
|
|
@EVENTS: delete (filename) - Dispatched when a file is deleted.
|
|
-->
|
|
|
|
<script lang="ts">
|
|
// [SECTION: IMPORTS]
|
|
import { createEventDispatcher } from 'svelte';
|
|
import { downloadFileUrl } from '../../services/storageService';
|
|
import { t } from '../../lib/i18n';
|
|
// [/SECTION: IMPORTS]
|
|
|
|
export let files = [];
|
|
const dispatch = createEventDispatcher();
|
|
|
|
function isDirectory(file) {
|
|
return file.mime_type === 'directory';
|
|
}
|
|
|
|
function formatSize(bytes) {
|
|
if (bytes === 0) return '0 B';
|
|
const k = 1024;
|
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
|
}
|
|
|
|
function formatDate(dateStr) {
|
|
return new Date(dateStr).toLocaleString();
|
|
}
|
|
</script>
|
|
|
|
<!-- [SECTION: TEMPLATE] -->
|
|
<div class="overflow-x-auto">
|
|
<table class="min-w-full bg-white border border-gray-200">
|
|
<thead class="bg-gray-50">
|
|
<tr>
|
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">{$t.storage.table.name}</th>
|
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">{$t.storage.table.category}</th>
|
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">{$t.storage.table.size}</th>
|
|
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">{$t.storage.table.created_at}</th>
|
|
<th class="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider">{$t.storage.table.actions}</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody class="divide-y divide-gray-200">
|
|
{#each files as file}
|
|
<tr class="hover:bg-gray-50">
|
|
<td class="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900">
|
|
{#if isDirectory(file)}
|
|
<button
|
|
on:click={() => dispatch('navigate', file.path)}
|
|
class="flex items-center text-indigo-600 hover:text-indigo-900"
|
|
>
|
|
<svg class="h-5 w-5 mr-2 text-yellow-400" fill="currentColor" viewBox="0 0 20 20">
|
|
<path d="M2 6a2 2 0 012-2h5l2 2h5a2 2 0 012 2v6a2 2 0 01-2 2H4a2 2 0 01-2-2V6z" />
|
|
</svg>
|
|
{file.name}
|
|
</button>
|
|
{:else}
|
|
<div class="flex items-center">
|
|
<svg class="h-5 w-5 mr-2 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 21h10a2 2 0 002-2V9.414a1 1 0 00-.293-.707l-5.414-5.414A1 1 0 0012.586 3H7a2 2 0 00-2 2v14a2 2 0 002 2z" />
|
|
</svg>
|
|
{file.name}
|
|
</div>
|
|
{/if}
|
|
</td>
|
|
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 capitalize">{file.category}</td>
|
|
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">
|
|
{isDirectory(file) ? '--' : formatSize(file.size)}
|
|
</td>
|
|
<td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500">{formatDate(file.created_at)}</td>
|
|
<td class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
|
|
{#if !isDirectory(file)}
|
|
<a
|
|
href={downloadFileUrl(file.category, file.path)}
|
|
download={file.name}
|
|
class="text-indigo-600 hover:text-indigo-900 mr-4"
|
|
>
|
|
{$t.storage.table.download}
|
|
</a>
|
|
{/if}
|
|
<button
|
|
on:click={() => dispatch('delete', { category: file.category, path: file.path, name: file.name })}
|
|
class="text-red-600 hover:text-red-900"
|
|
>
|
|
{$t.storage.table.delete}
|
|
</button>
|
|
</td>
|
|
</tr>
|
|
{:else}
|
|
<tr>
|
|
<td colspan="5" class="px-6 py-10 text-center text-sm text-gray-500">
|
|
{$t.storage.no_files}
|
|
</td>
|
|
</tr>
|
|
{/each}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<!-- [/SECTION: TEMPLATE] -->
|
|
|
|
<style>
|
|
/* ... */
|
|
</style>
|
|
|
|
<!-- [/DEF:FileList:Component] --> |