add download service

This commit is contained in:
2026-01-10 11:06:45 +03:00
parent c33c5fd674
commit 266f3768ef
44 changed files with 2652 additions and 4 deletions

View File

@@ -23,6 +23,13 @@
>
Upload Files
</button>
<button
class="nav-btn"
:class="{ active: currentPage === 'downloader' }"
@click="currentPage = 'downloader'"
>
Openings Downloader
</button>
</nav>
<!-- Quiz Generator Page -->
@@ -306,6 +313,9 @@
<!-- Admin Page (Upload Files) -->
<AdminPage v-if="currentPage === 'admin'" />
<!-- Openings Downloader -->
<OpeningsDownloader v-if="currentPage === 'downloader'" />
</div>
</template>
@@ -313,6 +323,7 @@
import { ref, reactive, computed, onMounted, watch } from 'vue'
import AdminPage from './components/AdminPage.vue'
import MediaManager from './components/MediaManager.vue'
import OpeningsDownloader from './components/OpeningsDownloader.vue'
const STORAGE_KEY = 'animeQuizSettings'
@@ -320,7 +331,8 @@ export default {
name: 'App',
components: {
AdminPage,
MediaManager
MediaManager,
OpeningsDownloader
},
setup() {
// Navigation

View File

@@ -0,0 +1,510 @@
<template>
<div class="queue-panel">
<div class="queue-header">
<div class="queue-title">
<h3>Download Queue</h3>
<span v-if="queue.worker_running" class="worker-indicator active" title="Worker is processing">
<span class="pulse"></span> Active
</span>
<span v-else class="worker-indicator idle" title="Worker is idle">Idle</span>
</div>
<div class="queue-actions">
<button
v-if="queue.total_done > 0 || queue.total_failed > 0"
@click="$emit('clear')"
class="btn-clear"
title="Clear completed tasks"
>
Clear
</button>
<button @click="$emit('refresh')" class="btn-refresh" title="Refresh">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
<path d="M21.5 2v6h-6M2.5 22v-6h6M2 11.5a10 10 0 0 1 18.8-4.3M22 12.5a10 10 0 0 1-18.8 4.3"/>
</svg>
</button>
</div>
</div>
<div class="queue-stats">
<div class="stat">
<span class="stat-value queued">{{ queue.total_queued }}</span>
<span class="stat-label">Queued</span>
</div>
<div class="stat">
<span class="stat-value downloading">{{ queue.total_downloading }}</span>
<span class="stat-label">Active</span>
</div>
<div class="stat">
<span class="stat-value done">{{ queue.total_done }}</span>
<span class="stat-label">Done</span>
</div>
<div class="stat">
<span class="stat-value failed">{{ queue.total_failed }}</span>
<span class="stat-label">Failed</span>
</div>
</div>
<div v-if="queue.estimated_queue_size_bytes > 0" class="queue-size">
Est. queue size: {{ formatBytes(queue.estimated_queue_size_bytes) }}
</div>
<div class="queue-list" v-if="queue.tasks?.length">
<div
v-for="task in sortedTasks"
:key="task.id"
class="queue-item"
:class="task.status"
>
<div class="task-main">
<div class="task-info">
<div class="task-anime">{{ task.anime_title }}</div>
<div class="task-theme">
<span class="theme-badge">{{ task.theme_name }}</span>
<span v-if="task.song_title" class="song-name">{{ task.song_title }}</span>
</div>
</div>
<div class="task-status">
<span :class="'status-badge ' + task.status">{{ formatStatus(task.status) }}</span>
</div>
</div>
<div v-if="isActive(task.status)" class="task-progress">
<div class="progress-bar">
<div class="progress-fill" :style="{ width: task.progress_percent + '%' }"></div>
</div>
<span class="progress-text">{{ task.progress_percent }}%</span>
</div>
<div v-if="task.error_message" class="task-error">
{{ task.error_message }}
</div>
<div class="task-actions">
<button
v-if="task.status === 'queued'"
@click="$emit('cancel', task.id)"
class="btn-action btn-cancel"
title="Cancel"
>
Cancel
</button>
<button
v-if="task.status === 'failed'"
@click="$emit('retry', task.id)"
class="btn-action btn-retry"
title="Retry"
>
Retry
</button>
<span v-if="task.status === 'done'" class="task-size">
{{ formatBytes(task.estimated_size_bytes) }}
</span>
</div>
</div>
</div>
<div v-else class="queue-empty">
No tasks in queue
</div>
</div>
</template>
<script setup>
import { computed } from 'vue'
const props = defineProps({
queue: {
type: Object,
required: true,
},
})
defineEmits(['cancel', 'retry', 'refresh', 'clear'])
const sortedTasks = computed(() => {
if (!props.queue.tasks) return []
// Sort: active first, then queued, then done/failed
const order = {
downloading: 0,
converting: 0,
uploading: 0,
queued: 1,
failed: 2,
done: 3,
}
return [...props.queue.tasks].sort((a, b) => {
const orderDiff = (order[a.status] || 4) - (order[b.status] || 4)
if (orderDiff !== 0) return orderDiff
// Within same status, sort by created_at desc
return new Date(b.created_at) - new Date(a.created_at)
})
})
function formatBytes(bytes) {
if (!bytes) return '0 B'
const units = ['B', 'KB', 'MB', 'GB']
let i = 0
let value = bytes
while (value >= 1024 && i < units.length - 1) {
value /= 1024
i++
}
return `${value.toFixed(1)} ${units[i]}`
}
function formatStatus(status) {
const labels = {
queued: 'Queued',
downloading: 'Downloading',
converting: 'Converting',
uploading: 'Uploading',
done: 'Done',
failed: 'Failed',
}
return labels[status] || status
}
function isActive(status) {
return ['downloading', 'converting', 'uploading'].includes(status)
}
</script>
<style scoped>
.queue-panel {
background: rgba(255, 255, 255, 0.03);
border-radius: 8px;
padding: 1rem;
max-height: 500px;
display: flex;
flex-direction: column;
}
.queue-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 1rem;
}
.queue-title {
display: flex;
align-items: center;
gap: 0.75rem;
}
.queue-title h3 {
margin: 0;
color: #fff;
}
.queue-actions {
display: flex;
align-items: center;
gap: 0.5rem;
}
.worker-indicator {
font-size: 0.75rem;
padding: 0.25rem 0.5rem;
border-radius: 4px;
display: flex;
align-items: center;
gap: 0.25rem;
}
.worker-indicator.active {
background: rgba(0, 255, 136, 0.15);
color: #00ff88;
}
.worker-indicator.idle {
background: rgba(136, 136, 136, 0.15);
color: #888;
}
.pulse {
width: 8px;
height: 8px;
background: #00ff88;
border-radius: 50%;
animation: pulse 1.5s ease-in-out infinite;
}
@keyframes pulse {
0%, 100% { opacity: 1; transform: scale(1); }
50% { opacity: 0.5; transform: scale(0.8); }
}
.btn-clear {
background: rgba(255, 170, 0, 0.2);
border: 1px solid rgba(255, 170, 0, 0.3);
border-radius: 4px;
padding: 0.4rem 0.75rem;
color: #ffaa00;
font-size: 0.8rem;
cursor: pointer;
transition: background 0.2s, border-color 0.2s;
}
.btn-clear:hover {
background: rgba(255, 170, 0, 0.3);
border-color: #ffaa00;
}
.btn-refresh {
background: transparent;
border: 1px solid rgba(255, 255, 255, 0.2);
border-radius: 4px;
padding: 0.5rem;
color: #888;
cursor: pointer;
transition: color 0.2s, border-color 0.2s;
}
.btn-refresh:hover {
color: #00d4ff;
border-color: #00d4ff;
}
/* Stats */
.queue-stats {
display: flex;
gap: 1rem;
margin-bottom: 1rem;
padding-bottom: 1rem;
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
}
.stat {
display: flex;
flex-direction: column;
align-items: center;
flex: 1;
}
.stat-value {
font-size: 1.5rem;
font-weight: bold;
}
.stat-value.queued {
color: #ffaa00;
}
.stat-value.downloading {
color: #00d4ff;
}
.stat-value.done {
color: #00ff88;
}
.stat-value.failed {
color: #ff4444;
}
.stat-label {
font-size: 0.75rem;
color: #888;
text-transform: uppercase;
}
.queue-size {
text-align: center;
color: #888;
font-size: 0.85rem;
margin-bottom: 1rem;
}
/* Queue List */
.queue-list {
flex: 1;
overflow-y: auto;
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.queue-item {
background: rgba(0, 0, 0, 0.2);
border-radius: 8px;
padding: 0.75rem;
border-left: 3px solid transparent;
}
.queue-item.queued {
border-left-color: #ffaa00;
}
.queue-item.downloading,
.queue-item.converting,
.queue-item.uploading {
border-left-color: #00d4ff;
}
.queue-item.done {
border-left-color: #00ff88;
}
.queue-item.failed {
border-left-color: #ff4444;
}
.task-main {
display: flex;
justify-content: space-between;
align-items: flex-start;
gap: 1rem;
}
.task-info {
flex: 1;
min-width: 0;
}
.task-anime {
font-weight: 500;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.task-theme {
display: flex;
align-items: center;
gap: 0.5rem;
margin-top: 0.25rem;
}
.theme-badge {
font-size: 0.75rem;
padding: 0.125rem 0.5rem;
background: rgba(0, 212, 255, 0.2);
color: #00d4ff;
border-radius: 4px;
}
.song-name {
font-size: 0.85rem;
color: #888;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.task-status {
flex-shrink: 0;
}
.status-badge {
padding: 0.25rem 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
font-weight: 500;
}
.status-badge.queued {
background: rgba(255, 170, 0, 0.2);
color: #ffaa00;
}
.status-badge.downloading,
.status-badge.converting,
.status-badge.uploading {
background: rgba(0, 212, 255, 0.2);
color: #00d4ff;
}
.status-badge.done {
background: rgba(0, 255, 136, 0.2);
color: #00ff88;
}
.status-badge.failed {
background: rgba(255, 68, 68, 0.2);
color: #ff4444;
}
/* Progress */
.task-progress {
display: flex;
align-items: center;
gap: 0.5rem;
margin-top: 0.5rem;
}
.task-progress .progress-bar {
flex: 1;
height: 4px;
background: rgba(255, 255, 255, 0.1);
border-radius: 2px;
overflow: hidden;
}
.task-progress .progress-fill {
height: 100%;
background: linear-gradient(90deg, #00d4ff, #00ff88);
transition: width 0.3s ease;
}
.progress-text {
font-size: 0.75rem;
color: #00d4ff;
min-width: 36px;
text-align: right;
}
/* Error */
.task-error {
margin-top: 0.5rem;
padding: 0.5rem;
background: rgba(255, 68, 68, 0.1);
border-radius: 4px;
color: #ff4444;
font-size: 0.8rem;
word-break: break-word;
}
/* Actions */
.task-actions {
display: flex;
justify-content: flex-end;
gap: 0.5rem;
margin-top: 0.5rem;
}
.btn-action {
padding: 0.25rem 0.75rem;
border: none;
border-radius: 4px;
font-size: 0.8rem;
cursor: pointer;
transition: opacity 0.2s;
}
.btn-action:hover {
opacity: 0.8;
}
.btn-cancel {
background: rgba(255, 68, 68, 0.2);
color: #ff4444;
}
.btn-retry {
background: rgba(0, 212, 255, 0.2);
color: #00d4ff;
}
.task-size {
font-size: 0.8rem;
color: #00ff88;
}
/* Empty */
.queue-empty {
text-align: center;
padding: 2rem;
color: #666;
}
</style>

View File

@@ -0,0 +1,799 @@
<template>
<div class="downloader-page">
<h1 class="page-title">Openings Downloader</h1>
<!-- Storage Stats Bar -->
<div class="storage-bar">
<div class="storage-info">
<span class="storage-label">Storage:</span>
<span class="storage-values">{{ formatBytes(storageStats.used_bytes) }} / {{ formatBytes(storageStats.limit_bytes) }}</span>
<span class="storage-percent" :class="{ warning: storageStats.used_percent > 80, danger: storageStats.used_percent > 95 }">
({{ storageStats.used_percent.toFixed(1) }}%)
</span>
<span class="openings-count">{{ storageStats.openings_count }} openings</span>
</div>
<div class="progress-bar">
<div
class="progress-fill"
:class="{ warning: storageStats.used_percent > 80, danger: storageStats.used_percent > 95 }"
:style="{ width: Math.min(storageStats.used_percent, 100) + '%' }"
></div>
</div>
<div v-if="!storageStats.can_download" class="storage-warning">
Storage limit exceeded! Cannot download new files.
</div>
</div>
<div class="main-content">
<!-- Left Panel: Search & Results -->
<div class="search-panel">
<!-- Search Section -->
<div class="search-section">
<div class="search-input-group">
<input
v-model="searchQuery"
type="text"
placeholder="Search anime..."
@keyup.enter="searchAnime"
class="search-input"
/>
<button @click="searchAnime" class="btn btn-primary" :disabled="searching || !searchQuery.trim()">
{{ searching ? 'Searching...' : 'Search' }}
</button>
</div>
<div class="search-filters">
<input v-model.number="yearFilter" type="number" placeholder="Year" class="filter-input" />
<select v-model="statusFilter" class="filter-select">
<option value="">Any status</option>
<option value="ongoing">Ongoing</option>
<option value="released">Released</option>
<option value="announced">Announced</option>
</select>
</div>
</div>
<!-- Search Results -->
<div class="search-results" v-if="searchResults.length">
<h3>Search Results ({{ searchResults.length }})</h3>
<div class="results-grid">
<div
v-for="anime in searchResults"
:key="anime.shikimori_id"
class="anime-card"
:class="{ selected: selectedAnime?.shikimori_id === anime.shikimori_id }"
@click="selectAnime(anime.shikimori_id)"
>
<img
v-if="anime.poster_url"
:src="anime.poster_url"
:alt="anime.title_russian || anime.title_english"
class="anime-poster"
/>
<div v-else class="anime-poster placeholder">No Image</div>
<div class="anime-info">
<div class="anime-title">{{ anime.title_russian || anime.title_english }}</div>
<div class="anime-year" v-if="anime.year">{{ anime.year }}</div>
</div>
</div>
</div>
</div>
<div v-else-if="searched && !searching" class="no-results">
No anime found for "{{ lastSearchQuery }}"
</div>
</div>
<!-- Right Panel: Selected Anime & Queue -->
<div class="detail-panel">
<!-- Selected Anime Detail -->
<div class="anime-detail" v-if="selectedAnime">
<div class="detail-header">
<img
v-if="selectedAnime.poster_url"
:src="selectedAnime.poster_url"
class="detail-poster"
/>
<div class="detail-info">
<h2>{{ selectedAnime.title_russian || selectedAnime.title_english }}</h2>
<p v-if="selectedAnime.title_japanese" class="japanese-title">{{ selectedAnime.title_japanese }}</p>
<p v-if="selectedAnime.year" class="year-info">Year: {{ selectedAnime.year }}</p>
</div>
</div>
<div v-if="loadingDetail" class="loading">Loading themes...</div>
<div v-else-if="selectedAnime.themes?.length" class="themes-section">
<div class="themes-header">
<h3>Available Themes ({{ selectedAnime.themes.length }})</h3>
<button
@click="addAllToQueue"
class="btn btn-success"
:disabled="!canAddAll || !storageStats.can_download"
>
Add All to Queue
</button>
</div>
<div class="themes-list">
<div
v-for="theme in selectedAnime.themes"
:key="theme.id"
class="theme-item"
:class="{ downloaded: theme.is_downloaded, 'in-queue': theme.download_status }"
>
<div class="theme-info">
<span class="theme-name" :class="theme.theme_type.toLowerCase()">{{ theme.full_name }}</span>
<span class="song-title" v-if="theme.song_title">{{ theme.song_title }}</span>
<span class="artist" v-if="theme.artist">by {{ theme.artist }}</span>
</div>
<div class="theme-status">
<span v-if="theme.is_downloaded" class="status-badge done">
Downloaded ({{ formatBytes(theme.file_size_bytes) }})
</span>
<span v-else-if="theme.download_status" :class="'status-badge ' + theme.download_status">
{{ theme.download_status }}
</span>
<button
v-else-if="theme.video_url"
@click="addToQueue([theme.id])"
class="btn btn-small btn-primary"
:disabled="!storageStats.can_download"
>
Add
</button>
<span v-else class="status-badge unavailable">No source</span>
</div>
</div>
</div>
</div>
<div v-else class="no-themes">
No themes found for this anime on AnimeThemes
</div>
</div>
<!-- Download Queue -->
<DownloadQueue
:queue="queueStatus"
@cancel="cancelTask"
@retry="retryTask"
@refresh="refreshQueue"
@clear="clearQueue"
/>
</div>
</div>
</div>
</template>
<script setup>
import { ref, reactive, computed, onMounted, onUnmounted } from 'vue'
import DownloadQueue from './DownloadQueue.vue'
// State
const searchQuery = ref('')
const lastSearchQuery = ref('')
const yearFilter = ref(null)
const statusFilter = ref('')
const searching = ref(false)
const searched = ref(false)
const searchResults = ref([])
const selectedAnime = ref(null)
const loadingDetail = ref(false)
const storageStats = reactive({
used_bytes: 0,
limit_bytes: 107374182400, // 100 GB default
used_percent: 0,
available_bytes: 107374182400,
can_download: true,
openings_count: 0,
})
const queueStatus = reactive({
tasks: [],
total_queued: 0,
total_downloading: 0,
total_done: 0,
total_failed: 0,
estimated_queue_size_bytes: 0,
worker_running: false,
})
let refreshInterval = null
// Computed
const canAddAll = computed(() => {
if (!selectedAnime.value?.themes) return false
return selectedAnime.value.themes.some(
t => !t.is_downloaded && !t.download_status && t.video_url
)
})
// Methods
function formatBytes(bytes) {
if (!bytes) return '0 B'
const units = ['B', 'KB', 'MB', 'GB', 'TB']
let i = 0
let value = bytes
while (value >= 1024 && i < units.length - 1) {
value /= 1024
i++
}
return `${value.toFixed(1)} ${units[i]}`
}
async function searchAnime() {
if (!searchQuery.value.trim()) return
searching.value = true
searched.value = false
lastSearchQuery.value = searchQuery.value
searchResults.value = []
selectedAnime.value = null
try {
const params = new URLSearchParams({ query: searchQuery.value, limit: '30' })
if (yearFilter.value) params.append('year', yearFilter.value)
if (statusFilter.value) params.append('status', statusFilter.value)
const response = await fetch(`/api/downloader/search?${params}`)
const data = await response.json()
searchResults.value = data.results || []
} catch (error) {
console.error('Search error:', error)
} finally {
searching.value = false
searched.value = true
}
}
async function selectAnime(shikimoriId) {
loadingDetail.value = true
selectedAnime.value = null
try {
const response = await fetch(`/api/downloader/anime/${shikimoriId}`)
const data = await response.json()
selectedAnime.value = data
} catch (error) {
console.error('Error loading anime detail:', error)
} finally {
loadingDetail.value = false
}
}
async function addToQueue(themeIds) {
try {
const response = await fetch('/api/downloader/queue/add', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ theme_ids: themeIds }),
})
const data = await response.json()
if (response.ok) {
Object.assign(queueStatus, data)
// Refresh selected anime to update statuses
if (selectedAnime.value) {
await selectAnime(selectedAnime.value.shikimori_id)
}
} else {
alert(data.detail || 'Failed to add to queue')
}
} catch (error) {
console.error('Error adding to queue:', error)
}
}
async function addAllToQueue() {
if (!selectedAnime.value?.id) return
try {
const response = await fetch('/api/downloader/queue/add-all', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ anime_id: selectedAnime.value.id }),
})
const data = await response.json()
if (response.ok) {
Object.assign(queueStatus, data)
await selectAnime(selectedAnime.value.shikimori_id)
} else {
alert(data.detail || 'Failed to add all to queue')
}
} catch (error) {
console.error('Error adding all to queue:', error)
}
}
async function cancelTask(taskId) {
try {
const response = await fetch(`/api/downloader/queue/${taskId}`, { method: 'DELETE' })
if (response.ok) {
await refreshQueue()
}
} catch (error) {
console.error('Error cancelling task:', error)
}
}
async function retryTask(taskId) {
try {
const response = await fetch(`/api/downloader/queue/${taskId}/retry`, { method: 'POST' })
if (response.ok) {
await refreshQueue()
}
} catch (error) {
console.error('Error retrying task:', error)
}
}
async function refreshQueue() {
try {
const response = await fetch('/api/downloader/queue')
const data = await response.json()
Object.assign(queueStatus, data)
} catch (error) {
console.error('Error refreshing queue:', error)
}
}
async function clearQueue() {
try {
const response = await fetch('/api/downloader/queue/clear?include_failed=true', { method: 'DELETE' })
if (response.ok) {
await refreshQueue()
}
} catch (error) {
console.error('Error clearing queue:', error)
}
}
async function refreshStorage() {
try {
const response = await fetch('/api/downloader/storage')
const data = await response.json()
Object.assign(storageStats, data)
} catch (error) {
console.error('Error refreshing storage:', error)
}
}
async function refreshAll() {
await Promise.all([refreshQueue(), refreshStorage()])
// Also refresh selected anime if any
if (selectedAnime.value?.shikimori_id) {
const response = await fetch(`/api/downloader/anime/${selectedAnime.value.shikimori_id}`)
if (response.ok) {
selectedAnime.value = await response.json()
}
}
}
// Lifecycle
onMounted(async () => {
await refreshAll()
// Auto-refresh every 3 seconds
refreshInterval = setInterval(refreshAll, 3000)
})
onUnmounted(() => {
if (refreshInterval) {
clearInterval(refreshInterval)
}
})
</script>
<style scoped>
.downloader-page {
padding: 1rem;
max-width: 1600px;
margin: 0 auto;
}
.page-title {
background: linear-gradient(90deg, #00d4ff, #7b2cbf);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
margin-bottom: 1rem;
}
/* Storage Bar */
.storage-bar {
background: rgba(255, 255, 255, 0.05);
border-radius: 8px;
padding: 1rem;
margin-bottom: 1.5rem;
}
.storage-info {
display: flex;
align-items: center;
gap: 1rem;
margin-bottom: 0.5rem;
flex-wrap: wrap;
}
.storage-label {
color: #888;
}
.storage-values {
font-weight: bold;
color: #fff;
}
.storage-percent {
color: #00ff88;
}
.storage-percent.warning {
color: #ffaa00;
}
.storage-percent.danger {
color: #ff4444;
}
.openings-count {
color: #00d4ff;
margin-left: auto;
}
.progress-bar {
height: 8px;
background: rgba(255, 255, 255, 0.1);
border-radius: 4px;
overflow: hidden;
}
.progress-fill {
height: 100%;
background: linear-gradient(90deg, #00d4ff, #00ff88);
transition: width 0.3s ease;
}
.progress-fill.warning {
background: linear-gradient(90deg, #ffaa00, #ff8800);
}
.progress-fill.danger {
background: linear-gradient(90deg, #ff4444, #ff0000);
}
.storage-warning {
color: #ff4444;
font-weight: bold;
margin-top: 0.5rem;
padding: 0.5rem;
background: rgba(255, 68, 68, 0.1);
border-radius: 4px;
}
/* Main Content */
.main-content {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1.5rem;
}
@media (max-width: 1200px) {
.main-content {
grid-template-columns: 1fr;
}
}
/* Search Panel */
.search-panel {
background: rgba(255, 255, 255, 0.03);
border-radius: 8px;
padding: 1rem;
}
.search-section {
margin-bottom: 1rem;
}
.search-input-group {
display: flex;
gap: 0.5rem;
margin-bottom: 0.5rem;
}
.search-input {
flex: 1;
padding: 0.75rem 1rem;
border: 1px solid rgba(255, 255, 255, 0.1);
border-radius: 8px;
background: rgba(0, 0, 0, 0.3);
color: #fff;
font-size: 1rem;
}
.search-input:focus {
outline: none;
border-color: #00d4ff;
}
.search-filters {
display: flex;
gap: 0.5rem;
}
.filter-input,
.filter-select {
padding: 0.5rem;
border: 1px solid rgba(255, 255, 255, 0.1);
border-radius: 4px;
background: rgba(0, 0, 0, 0.3);
color: #fff;
font-size: 0.9rem;
}
.filter-input {
width: 100px;
}
/* Results Grid */
.results-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
gap: 1rem;
max-height: 600px;
overflow-y: auto;
}
.anime-card {
background: rgba(255, 255, 255, 0.05);
border-radius: 8px;
overflow: hidden;
cursor: pointer;
transition: transform 0.2s, box-shadow 0.2s;
}
.anime-card:hover {
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(0, 212, 255, 0.2);
}
.anime-card.selected {
border: 2px solid #00d4ff;
}
.anime-poster {
width: 100%;
aspect-ratio: 2/3;
object-fit: cover;
}
.anime-poster.placeholder {
display: flex;
align-items: center;
justify-content: center;
background: rgba(255, 255, 255, 0.05);
color: #666;
}
.anime-info {
padding: 0.5rem;
}
.anime-title {
font-size: 0.85rem;
font-weight: 500;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.anime-year {
font-size: 0.75rem;
color: #888;
}
.no-results {
text-align: center;
padding: 2rem;
color: #888;
}
/* Detail Panel */
.detail-panel {
display: flex;
flex-direction: column;
gap: 1rem;
}
.anime-detail {
background: rgba(255, 255, 255, 0.03);
border-radius: 8px;
padding: 1rem;
}
.detail-header {
display: flex;
gap: 1rem;
margin-bottom: 1rem;
}
.detail-poster {
width: 120px;
height: 180px;
object-fit: cover;
border-radius: 8px;
}
.detail-info h2 {
margin: 0 0 0.5rem 0;
font-size: 1.2rem;
}
.japanese-title {
color: #888;
font-size: 0.9rem;
}
.year-info {
color: #00d4ff;
}
/* Themes Section */
.themes-section {
margin-top: 1rem;
}
.themes-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 1rem;
}
.themes-header h3 {
margin: 0;
}
.themes-list {
display: flex;
flex-direction: column;
gap: 0.5rem;
max-height: 400px;
overflow-y: auto;
}
.theme-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.75rem;
background: rgba(0, 0, 0, 0.2);
border-radius: 8px;
}
.theme-item.downloaded {
border-left: 3px solid #00ff88;
}
.theme-item.in-queue {
border-left: 3px solid #00d4ff;
}
.theme-info {
display: flex;
flex-direction: column;
gap: 0.25rem;
}
.theme-name {
font-weight: bold;
padding: 0.125rem 0.5rem;
border-radius: 4px;
display: inline-block;
width: fit-content;
}
.theme-name.op {
background: rgba(0, 212, 255, 0.2);
color: #00d4ff;
}
.theme-name.ed {
background: rgba(123, 44, 191, 0.2);
color: #b57edc;
}
.song-title {
color: #fff;
font-size: 0.9rem;
}
.artist {
color: #888;
font-size: 0.8rem;
}
.theme-status {
display: flex;
align-items: center;
}
.status-badge {
padding: 0.25rem 0.75rem;
border-radius: 12px;
font-size: 0.8rem;
font-weight: 500;
}
.status-badge.done {
background: rgba(0, 255, 136, 0.2);
color: #00ff88;
}
.status-badge.queued {
background: rgba(255, 170, 0, 0.2);
color: #ffaa00;
}
.status-badge.downloading,
.status-badge.converting,
.status-badge.uploading {
background: rgba(0, 212, 255, 0.2);
color: #00d4ff;
}
.status-badge.failed {
background: rgba(255, 68, 68, 0.2);
color: #ff4444;
}
.status-badge.unavailable {
background: rgba(136, 136, 136, 0.2);
color: #888;
}
.no-themes {
text-align: center;
padding: 2rem;
color: #888;
}
.loading {
text-align: center;
padding: 2rem;
color: #00d4ff;
}
/* Buttons */
.btn {
padding: 0.75rem 1.5rem;
border: none;
border-radius: 8px;
font-weight: 500;
cursor: pointer;
transition: opacity 0.2s, transform 0.2s;
}
.btn:disabled {
opacity: 0.5;
cursor: not-allowed;
}
.btn:not(:disabled):hover {
transform: translateY(-1px);
}
.btn-primary {
background: linear-gradient(90deg, #00d4ff, #0099cc);
color: #fff;
}
.btn-success {
background: linear-gradient(90deg, #00ff88, #00cc6a);
color: #000;
}
.btn-small {
padding: 0.375rem 0.75rem;
font-size: 0.85rem;
}
</style>

View File

@@ -7,6 +7,22 @@ export default defineConfig({
host: '0.0.0.0',
port: 5173,
proxy: {
// Downloader API - already has /api prefix on backend
'/api/downloader': {
target: 'http://backend:8000',
changeOrigin: true
},
// Openings API - already has /api prefix on backend
'/api/openings': {
target: 'http://backend:8000',
changeOrigin: true
},
// Backgrounds API - already has /api prefix on backend
'/api/backgrounds': {
target: 'http://backend:8000',
changeOrigin: true
},
// Core API endpoints - strip /api prefix
'/api': {
target: 'http://backend:8000',
changeOrigin: true,