Files
librodrome/app/components/admin/AdminMediaBrowser.vue
2026-02-20 12:55:10 +01:00

212 lines
4.5 KiB
Vue

<template>
<div class="media-browser">
<!-- Toolbar -->
<div class="media-toolbar">
<div class="flex items-center gap-2">
<button
v-for="t in types"
:key="t.value"
class="filter-btn"
:class="{ 'filter-btn--active': filter === t.value }"
@click="filter = t.value"
>
{{ t.label }}
</button>
</div>
<span class="text-xs text-white/40">{{ filtered.length }} fichier(s)</span>
</div>
<!-- Grid -->
<div class="media-grid">
<div
v-for="file in filtered"
:key="file.path"
class="media-card"
:class="{ 'media-card--selected': selected === file.path }"
@click="selected = selected === file.path ? null : file.path"
>
<div v-if="file.type === 'image'" class="media-thumb">
<img :src="file.path" :alt="file.name" />
</div>
<div v-else class="media-icon">
<div
:class="file.type === 'audio' ? 'i-lucide-music' : file.type === 'document' ? 'i-lucide-file-text' : 'i-lucide-file'"
class="h-6 w-6"
/>
</div>
<div class="media-info">
<span class="media-name">{{ file.name }}</span>
<span class="media-size">{{ formatSize(file.size) }}</span>
</div>
</div>
</div>
<!-- Actions for selected -->
<div v-if="selected" class="media-actions">
<code class="text-xs text-accent">{{ selected }}</code>
<button class="delete-btn" @click="$emit('delete', selected)">
<div class="i-lucide-trash-2 h-4 w-4" />
Supprimer
</button>
</div>
</div>
</template>
<script setup lang="ts">
interface MediaFile {
name: string
path: string
size: number
type: string
modifiedAt: string
}
const props = defineProps<{
files: MediaFile[]
}>()
defineEmits<{
delete: [path: string]
}>()
const filter = ref('all')
const selected = ref<string | null>(null)
const types = [
{ value: 'all', label: 'Tous' },
{ value: 'image', label: 'Images' },
{ value: 'audio', label: 'Audio' },
{ value: 'document', label: 'Documents' },
]
const filtered = computed(() => {
if (filter.value === 'all') return props.files
return props.files.filter(f => f.type === filter.value)
})
function formatSize(bytes: number): string {
if (bytes < 1024) return bytes + ' B'
if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB'
return (bytes / (1024 * 1024)).toFixed(1) + ' MB'
}
</script>
<style scoped>
.media-toolbar {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 1rem;
}
.filter-btn {
padding: 0.25rem 0.625rem;
border-radius: 9999px;
border: 1px solid hsl(20 8% 18%);
background: none;
color: hsl(20 8% 55%);
font-size: 0.75rem;
cursor: pointer;
transition: all 0.2s;
}
.filter-btn--active {
background: hsl(12 76% 48% / 0.15);
border-color: hsl(12 76% 48% / 0.3);
color: hsl(12 76% 68%);
}
.media-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
gap: 0.75rem;
}
.media-card {
border: 1px solid hsl(20 8% 14%);
border-radius: 0.5rem;
overflow: hidden;
cursor: pointer;
transition: all 0.2s;
}
.media-card:hover {
border-color: hsl(20 8% 22%);
}
.media-card--selected {
border-color: hsl(12 76% 48%);
box-shadow: 0 0 0 1px hsl(12 76% 48% / 0.3);
}
.media-thumb {
aspect-ratio: 1;
overflow: hidden;
background: hsl(20 8% 6%);
}
.media-thumb img {
width: 100%;
height: 100%;
object-fit: cover;
}
.media-icon {
aspect-ratio: 1;
display: flex;
align-items: center;
justify-content: center;
background: hsl(20 8% 6%);
color: hsl(20 8% 40%);
}
.media-info {
padding: 0.5rem;
display: flex;
flex-direction: column;
gap: 0.125rem;
}
.media-name {
font-size: 0.72rem;
color: white;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.media-size {
font-size: 0.65rem;
color: hsl(20 8% 40%);
}
.media-actions {
display: flex;
align-items: center;
justify-content: space-between;
margin-top: 1rem;
padding: 0.75rem;
border: 1px solid hsl(20 8% 14%);
border-radius: 0.5rem;
background: hsl(20 8% 5%);
}
.delete-btn {
display: flex;
align-items: center;
gap: 0.375rem;
padding: 0.375rem 0.75rem;
border-radius: 0.375rem;
border: 1px solid hsl(0 60% 45% / 0.3);
background: hsl(0 60% 45% / 0.1);
color: hsl(0 60% 65%);
font-size: 0.8rem;
cursor: pointer;
transition: all 0.2s;
}
.delete-btn:hover {
background: hsl(0 60% 45% / 0.2);
}
</style>