add download service
This commit is contained in:
@@ -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
|
||||
|
||||
510
frontend/src/components/DownloadQueue.vue
Normal file
510
frontend/src/components/DownloadQueue.vue
Normal 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>
|
||||
799
frontend/src/components/OpeningsDownloader.vue
Normal file
799
frontend/src/components/OpeningsDownloader.vue
Normal 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>
|
||||
@@ -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,
|
||||
|
||||
Reference in New Issue
Block a user