feat(setup): add unresolved folders GUI for manual series resolution
- Add /setup/unresolved page for manual provider key resolution
- Integrate unresolved check into setup wizard flow
- Auto-redirect to unresolved page if folders need resolution
After initial setup scan, folders that couldn't be auto-resolved
are now tracked and can be resolved manually via the GUI.
Endpoints:
- GET /api/setup/unresolved - list unresolved folders
- POST /api/setup/unresolved/{folder}/resolve - resolve with provider key
- POST /api/setup/unresolved/{folder}/search - re-search for suggestions
- DELETE /api/setup/unresolved/{folder} - delete without adding
This commit is contained in:
@@ -59,3 +59,13 @@ async def loading_page(request: Request):
|
|||||||
request,
|
request,
|
||||||
title="Initializing - Aniworld"
|
title="Initializing - Aniworld"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@router.get("/setup/unresolved", response_class=HTMLResponse)
|
||||||
|
async def unresolved_page(request: Request):
|
||||||
|
"""Serve the unresolved folders resolution page."""
|
||||||
|
return render_template(
|
||||||
|
"unresolved.html",
|
||||||
|
request,
|
||||||
|
title="Resolve Series - Aniworld"
|
||||||
|
)
|
||||||
|
|||||||
@@ -790,17 +790,36 @@
|
|||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
if (response.ok && data.status === 'ok') {
|
if (response.ok && data.status === 'ok') {
|
||||||
// Redirect to loading page if provided, otherwise go to login
|
// Redirect to loading page if provided, otherwise check for unresolved folders
|
||||||
if (data.redirect) {
|
if (data.redirect) {
|
||||||
showMessage('Setup saved! Initializing your anime library...', 'success');
|
showMessage('Setup saved! Initializing your anime library...', 'success');
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.location.href = data.redirect;
|
window.location.href = data.redirect;
|
||||||
}, 500);
|
}, 500);
|
||||||
} else {
|
} else {
|
||||||
showMessage('Setup completed successfully! Redirecting to login...', 'success');
|
// Check for unresolved folders before redirecting
|
||||||
setTimeout(() => {
|
showMessage('Setup completed successfully! Checking for unresolved series...', 'success');
|
||||||
window.location.href = '/login';
|
setTimeout(async () => {
|
||||||
}, 2000);
|
try {
|
||||||
|
const token = localStorage.getItem('auth_token');
|
||||||
|
const res = await fetch('/api/setup/unresolved', {
|
||||||
|
headers: { 'Authorization': `Bearer ${token}` }
|
||||||
|
});
|
||||||
|
if (res.ok) {
|
||||||
|
const unresolved = await res.json();
|
||||||
|
if (unresolved && unresolved.length > 0) {
|
||||||
|
window.location.href = '/setup/unresolved';
|
||||||
|
} else {
|
||||||
|
window.location.href = '/login';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
window.location.href = '/login';
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Error checking unresolved folders:', e);
|
||||||
|
window.location.href = '/login';
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const errorMessage = data.detail || data.message || 'Setup failed';
|
const errorMessage = data.detail || data.message || 'Setup failed';
|
||||||
|
|||||||
707
src/server/web/templates/unresolved.html
Normal file
707
src/server/web/templates/unresolved.html
Normal file
@@ -0,0 +1,707 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" data-theme="light">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>AniWorld Manager - Resolve Series</title>
|
||||||
|
<link rel="stylesheet" href="/static/css/styles.css?v={{ static_v }}">
|
||||||
|
<link href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css" rel="stylesheet">
|
||||||
|
<style>
|
||||||
|
.unresolved-container {
|
||||||
|
min-height: 100vh;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
background: linear-gradient(135deg, var(--color-primary-light) 0%, var(--color-primary) 100%);
|
||||||
|
padding: 2rem 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unresolved-card {
|
||||||
|
background: var(--color-surface);
|
||||||
|
border-radius: 16px;
|
||||||
|
padding: 2rem;
|
||||||
|
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
|
||||||
|
width: 100%;
|
||||||
|
max-width: 700px;
|
||||||
|
border: 1px solid var(--color-border);
|
||||||
|
max-height: 90vh;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unresolved-header {
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unresolved-header .icon {
|
||||||
|
font-size: 3rem;
|
||||||
|
color: var(--color-warning);
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unresolved-header h1 {
|
||||||
|
margin: 0;
|
||||||
|
color: var(--color-text);
|
||||||
|
font-size: 1.8rem;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.unresolved-header p {
|
||||||
|
margin: 1rem 0 0 0;
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
font-size: 1rem;
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-item {
|
||||||
|
background: var(--color-background);
|
||||||
|
border: 1px solid var(--color-border);
|
||||||
|
border-radius: var(--border-radius-lg);
|
||||||
|
padding: 1.25rem;
|
||||||
|
transition: all var(--transition-duration) var(--transition-easing);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-item.resolving {
|
||||||
|
opacity: 0.7;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-item.resolved {
|
||||||
|
animation: fadeSlideOut 0.4s ease forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeSlideOut {
|
||||||
|
0% { opacity: 1; transform: translateY(0); max-height: 500px; }
|
||||||
|
100% { opacity: 0; transform: translateY(-10px); max-height: 0; padding: 0; margin: 0; border: 0; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-item-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: flex-start;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-title {
|
||||||
|
font-size: 1.1rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--color-text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-year {
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-attempts {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
color: var(--color-text-tertiary);
|
||||||
|
margin-top: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-delete-btn {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--color-text-tertiary);
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0.25rem;
|
||||||
|
border-radius: var(--border-radius-sm);
|
||||||
|
transition: color var(--transition-duration);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-delete-btn:hover {
|
||||||
|
color: var(--color-error);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-input-row {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.5rem;
|
||||||
|
margin-bottom: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-input {
|
||||||
|
flex: 1;
|
||||||
|
padding: 0.75rem 1rem;
|
||||||
|
border: 2px solid var(--color-border);
|
||||||
|
border-radius: var(--border-radius-md);
|
||||||
|
font-size: 0.95rem;
|
||||||
|
background: var(--color-surface);
|
||||||
|
color: var(--color-text);
|
||||||
|
transition: border-color var(--transition-duration);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-input:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--color-accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-input::placeholder {
|
||||||
|
color: var(--color-text-tertiary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.resolve-btn {
|
||||||
|
padding: 0.75rem 1.5rem;
|
||||||
|
background: var(--color-accent);
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
border-radius: var(--border-radius-md);
|
||||||
|
font-size: 0.95rem;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all var(--transition-duration);
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.resolve-btn:hover:not(:disabled) {
|
||||||
|
background: var(--color-accent-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.resolve-btn:disabled {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-suggestions {
|
||||||
|
margin-top: 0.75rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestions-label {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.35rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
font-size: 0.9rem;
|
||||||
|
color: var(--color-text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-item i {
|
||||||
|
color: var(--color-accent);
|
||||||
|
font-size: 0.8rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-link {
|
||||||
|
color: var(--color-accent);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-link:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-suggestions {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: var(--color-text-tertiary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-again-btn {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid var(--color-border);
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
padding: 0.4rem 0.8rem;
|
||||||
|
border-radius: var(--border-radius-md);
|
||||||
|
font-size: 0.8rem;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all var(--transition-duration);
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-again-btn:hover {
|
||||||
|
background: var(--color-surface-hover);
|
||||||
|
border-color: var(--color-accent);
|
||||||
|
color: var(--color-accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-again-btn.searching {
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0.7;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Empty state */
|
||||||
|
.empty-state {
|
||||||
|
text-align: center;
|
||||||
|
padding: 3rem 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-state .icon {
|
||||||
|
font-size: 4rem;
|
||||||
|
color: var(--color-success);
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-state h2 {
|
||||||
|
color: var(--color-success);
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-state p {
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Loading state */
|
||||||
|
.loading-state {
|
||||||
|
text-align: center;
|
||||||
|
padding: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner {
|
||||||
|
width: 2rem;
|
||||||
|
height: 2rem;
|
||||||
|
border: 3px solid var(--color-border);
|
||||||
|
border-top-color: var(--color-accent);
|
||||||
|
border-radius: 50%;
|
||||||
|
animation: spin 1s linear infinite;
|
||||||
|
margin: 0 auto 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
to { transform: rotate(360deg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Error state */
|
||||||
|
.folder-error {
|
||||||
|
color: var(--color-error);
|
||||||
|
font-size: 0.85rem;
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.folder-error.visible {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Toast container */
|
||||||
|
.toast-container {
|
||||||
|
position: fixed;
|
||||||
|
top: var(--spacing-xl);
|
||||||
|
right: var(--spacing-xl);
|
||||||
|
z-index: 1100;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: var(--spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toast {
|
||||||
|
background: var(--color-surface);
|
||||||
|
border: 1px solid var(--color-border);
|
||||||
|
border-radius: var(--border-radius-lg);
|
||||||
|
padding: var(--spacing-md) var(--spacing-lg);
|
||||||
|
box-shadow: var(--shadow-elevated);
|
||||||
|
min-width: 280px;
|
||||||
|
animation: slideIn 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toast.success { border-left: 4px solid var(--color-success); }
|
||||||
|
.toast.error { border-left: 4px solid var(--color-error); }
|
||||||
|
.toast.warning { border-left: 4px solid var(--color-warning); }
|
||||||
|
|
||||||
|
@keyframes slideIn {
|
||||||
|
from { transform: translateX(100%); opacity: 0; }
|
||||||
|
to { transform: translateX(0); opacity: 1; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-toggle {
|
||||||
|
position: absolute;
|
||||||
|
top: 1rem;
|
||||||
|
right: 1rem;
|
||||||
|
background: rgba(255, 255, 255, 0.1);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.2);
|
||||||
|
color: white;
|
||||||
|
padding: 0.5rem;
|
||||||
|
border-radius: 50%;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 2.5rem;
|
||||||
|
height: 2.5rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-toggle:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.2);
|
||||||
|
transform: scale(1.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.skip-link {
|
||||||
|
display: block;
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 1.5rem;
|
||||||
|
color: var(--color-text-secondary);
|
||||||
|
font-size: 0.9rem;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skip-link:hover {
|
||||||
|
color: var(--color-text-primary);
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
.folder-input-row {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
.resolve-btn {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="unresolved-container">
|
||||||
|
<button class="theme-toggle" id="theme-toggle" title="Toggle theme">
|
||||||
|
<i class="fas fa-moon"></i>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<div class="unresolved-card">
|
||||||
|
<div class="unresolved-header">
|
||||||
|
<div class="icon">
|
||||||
|
<i class="fas fa-folder-question"></i>
|
||||||
|
</div>
|
||||||
|
<h1>Resolve Unresolved Series</h1>
|
||||||
|
<p>Some series couldn't be found automatically. Enter the provider key for each folder to complete setup.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="loading-state" class="loading-state">
|
||||||
|
<div class="spinner"></div>
|
||||||
|
<p>Loading unresolved folders...</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="empty-state" class="empty-state" style="display: none;">
|
||||||
|
<div class="icon">
|
||||||
|
<i class="fas fa-check-circle"></i>
|
||||||
|
</div>
|
||||||
|
<h2>All Series Configured!</h2>
|
||||||
|
<p>Redirecting to your anime library...</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="folder-list" class="folder-list" style="display: none;"></div>
|
||||||
|
|
||||||
|
<a href="/" id="skip-link" class="skip-link" style="display: none;">
|
||||||
|
Skip and go to main app
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="toast-container" class="toast-container"></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Theme toggle
|
||||||
|
const themeToggle = document.getElementById('theme-toggle');
|
||||||
|
const htmlElement = document.documentElement;
|
||||||
|
const savedTheme = localStorage.getItem('theme') || 'light';
|
||||||
|
htmlElement.setAttribute('data-theme', savedTheme);
|
||||||
|
updateThemeIcon(savedTheme);
|
||||||
|
|
||||||
|
themeToggle.addEventListener('click', () => {
|
||||||
|
const currentTheme = htmlElement.getAttribute('data-theme');
|
||||||
|
const newTheme = currentTheme === 'dark' ? 'light' : 'dark';
|
||||||
|
htmlElement.setAttribute('data-theme', newTheme);
|
||||||
|
localStorage.setItem('theme', newTheme);
|
||||||
|
updateThemeIcon(newTheme);
|
||||||
|
});
|
||||||
|
|
||||||
|
function updateThemeIcon(theme) {
|
||||||
|
const icon = themeToggle.querySelector('i');
|
||||||
|
icon.className = theme === 'dark' ? 'fas fa-sun' : 'fas fa-moon';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Toast notification
|
||||||
|
function showToast(message, type = 'info') {
|
||||||
|
const container = document.getElementById('toast-container');
|
||||||
|
const toast = document.createElement('div');
|
||||||
|
toast.className = `toast ${type}`;
|
||||||
|
toast.textContent = message;
|
||||||
|
container.appendChild(toast);
|
||||||
|
setTimeout(() => {
|
||||||
|
toast.style.animation = 'slideIn 0.2s ease reverse';
|
||||||
|
setTimeout(() => toast.remove(), 200);
|
||||||
|
}, 3000);
|
||||||
|
}
|
||||||
|
|
||||||
|
// API client helpers
|
||||||
|
async function fetchUnresolved() {
|
||||||
|
const token = localStorage.getItem('auth_token');
|
||||||
|
if (!token) {
|
||||||
|
window.location.href = '/login';
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const res = await fetch('/api/setup/unresolved', {
|
||||||
|
headers: { 'Authorization': `Bearer ${token}` }
|
||||||
|
});
|
||||||
|
if (res.status === 401) {
|
||||||
|
localStorage.removeItem('auth_token');
|
||||||
|
window.location.href = '/login';
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return res.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function resolveFolder(folderName, providerKey) {
|
||||||
|
const token = localStorage.getItem('auth_token');
|
||||||
|
const encodedName = encodeURIComponent(folderName);
|
||||||
|
const res = await fetch(`/api/setup/unresolved/${encodedName}/resolve`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${token}`,
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ provider_key: providerKey })
|
||||||
|
});
|
||||||
|
return res.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function reSearchFolder(folderName) {
|
||||||
|
const token = localStorage.getItem('auth_token');
|
||||||
|
const encodedName = encodeURIComponent(folderName);
|
||||||
|
const res = await fetch(`/api/setup/unresolved/${encodedName}/search`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Authorization': `Bearer ${token}` }
|
||||||
|
});
|
||||||
|
return res.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteFolder(folderName) {
|
||||||
|
const token = localStorage.getItem('auth_token');
|
||||||
|
const encodedName = encodeURIComponent(folderName);
|
||||||
|
const res = await fetch(`/api/setup/unresolved/${encodedName}`, {
|
||||||
|
method: 'DELETE',
|
||||||
|
headers: { 'Authorization': `Bearer ${token}` }
|
||||||
|
});
|
||||||
|
return res.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Render functions
|
||||||
|
function renderFolderItem(folder) {
|
||||||
|
const suggestionsHtml = folder.search_suggestions && folder.search_suggestions.length > 0
|
||||||
|
? folder.search_suggestions.map(s => `
|
||||||
|
<div class="suggestion-item">
|
||||||
|
<i class="fas fa-link"></i>
|
||||||
|
<a href="${s.link}" class="suggestion-link" target="_blank">${s.title}</a>
|
||||||
|
</div>
|
||||||
|
`).join('')
|
||||||
|
: '<div class="no-suggestions"><i class="fas fa-info-circle"></i> No suggestions found</div>';
|
||||||
|
|
||||||
|
const searchAgainBtn = (folder.search_suggestions && folder.search_suggestions.length === 0)
|
||||||
|
? `<button class="search-again-btn" data-folder="${folder.folder_name}">
|
||||||
|
<i class="fas fa-search"></i> Search Again
|
||||||
|
</button>`
|
||||||
|
: '';
|
||||||
|
|
||||||
|
return `
|
||||||
|
<div class="folder-item" data-folder="${folder.folder_name}">
|
||||||
|
<div class="folder-item-header">
|
||||||
|
<div>
|
||||||
|
<div class="folder-title">${folder.title}${folder.year ? ` <span class="folder-year">(${folder.year})</span>` : ''}</div>
|
||||||
|
<div class="folder-attempts">${folder.search_attempts} search attempt${folder.search_attempts !== 1 ? 's' : ''}</div>
|
||||||
|
</div>
|
||||||
|
<button class="folder-delete-btn" data-folder="${folder.folder_name}" title="Remove without adding">
|
||||||
|
<i class="fas fa-times"></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="folder-input-row">
|
||||||
|
<input type="text" class="folder-input"
|
||||||
|
placeholder="Enter provider key (e.g., ooku-the-inner-chambers)"
|
||||||
|
data-folder="${folder.folder_name}">
|
||||||
|
<button class="resolve-btn" data-folder="${folder.folder_name}" disabled>
|
||||||
|
Resolve
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="folder-error" data-folder="${folder.folder_name}"></div>
|
||||||
|
<div class="folder-suggestions">
|
||||||
|
<div class="suggestions-label">Suggestions:</div>
|
||||||
|
<div class="suggestion-list">
|
||||||
|
${suggestionsHtml}
|
||||||
|
</div>
|
||||||
|
${searchAgainBtn}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderFolders(folders) {
|
||||||
|
const listEl = document.getElementById('folder-list');
|
||||||
|
const loadingEl = document.getElementById('loading-state');
|
||||||
|
const emptyEl = document.getElementById('empty-state');
|
||||||
|
|
||||||
|
loadingEl.style.display = 'none';
|
||||||
|
|
||||||
|
if (folders.length === 0) {
|
||||||
|
listEl.style.display = 'none';
|
||||||
|
emptyEl.style.display = 'block';
|
||||||
|
document.getElementById('skip-link').style.display = 'block';
|
||||||
|
setTimeout(() => { window.location.href = '/'; }, 2000);
|
||||||
|
} else {
|
||||||
|
listEl.style.display = 'flex';
|
||||||
|
emptyEl.style.display = 'none';
|
||||||
|
listEl.innerHTML = folders.map(renderFolderItem).join('');
|
||||||
|
attachFolderEvents();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function attachFolderEvents() {
|
||||||
|
// Input enable/disable resolve button
|
||||||
|
document.querySelectorAll('.folder-input').forEach(input => {
|
||||||
|
input.addEventListener('input', (e) => {
|
||||||
|
const folder = e.target.dataset.folder;
|
||||||
|
const btn = document.querySelector(`.resolve-btn[data-folder="${folder}"]`);
|
||||||
|
btn.disabled = !e.target.value.trim();
|
||||||
|
// Clear error
|
||||||
|
const errEl = document.querySelector(`.folder-error[data-folder="${folder}"]`);
|
||||||
|
errEl.classList.remove('visible');
|
||||||
|
errEl.textContent = '';
|
||||||
|
});
|
||||||
|
|
||||||
|
// Enter key triggers resolve
|
||||||
|
input.addEventListener('keydown', (e) => {
|
||||||
|
if (e.key === 'Enter') {
|
||||||
|
const folder = e.target.dataset.folder;
|
||||||
|
const btn = document.querySelector(`.resolve-btn[data-folder="${folder}"]`);
|
||||||
|
if (!btn.disabled) btn.click();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Resolve button
|
||||||
|
document.querySelectorAll('.resolve-btn').forEach(btn => {
|
||||||
|
btn.addEventListener('click', async (e) => {
|
||||||
|
const folder = e.target.dataset.folder;
|
||||||
|
const input = document.querySelector(`.folder-input[data-folder="${folder}"]`);
|
||||||
|
const item = document.querySelector(`.folder-item[data-folder="${folder}"]`);
|
||||||
|
const errEl = document.querySelector(`.folder-error[data-folder="${folder}"]`);
|
||||||
|
|
||||||
|
const providerKey = input.value.trim();
|
||||||
|
if (!providerKey) return;
|
||||||
|
|
||||||
|
item.classList.add('resolving');
|
||||||
|
btn.disabled = true;
|
||||||
|
btn.innerHTML = '<i class="fas fa-spinner fa-spin"></i>';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await resolveFolder(folder, providerKey);
|
||||||
|
|
||||||
|
if (result.status === 'success') {
|
||||||
|
showToast(`Added: ${result.message.replace('Successfully resolved and added series: ', '')}`, 'success');
|
||||||
|
item.classList.add('resolved');
|
||||||
|
setTimeout(() => {
|
||||||
|
item.remove();
|
||||||
|
checkEmptyList();
|
||||||
|
}, 400);
|
||||||
|
} else {
|
||||||
|
errEl.textContent = result.detail || result.message || 'Failed to resolve';
|
||||||
|
errEl.classList.add('visible');
|
||||||
|
btn.disabled = false;
|
||||||
|
btn.innerHTML = 'Resolve';
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
errEl.textContent = 'Server error. Please try again.';
|
||||||
|
errEl.classList.add('visible');
|
||||||
|
btn.disabled = false;
|
||||||
|
btn.innerHTML = 'Resolve';
|
||||||
|
} finally {
|
||||||
|
item.classList.remove('resolving');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Delete button
|
||||||
|
document.querySelectorAll('.folder-delete-btn').forEach(btn => {
|
||||||
|
btn.addEventListener('click', async (e) => {
|
||||||
|
const folder = e.currentTarget.dataset.folder;
|
||||||
|
const item = document.querySelector(`.folder-item[data-folder="${folder}"]`);
|
||||||
|
|
||||||
|
if (!confirm('Remove this unresolved folder? You can add the series manually later.')) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await deleteFolder(folder);
|
||||||
|
item.classList.add('resolved');
|
||||||
|
setTimeout(() => {
|
||||||
|
item.remove();
|
||||||
|
checkEmptyList();
|
||||||
|
}, 400);
|
||||||
|
} catch (err) {
|
||||||
|
showToast('Failed to remove folder', 'error');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Search again button
|
||||||
|
document.querySelectorAll('.search-again-btn').forEach(btn => {
|
||||||
|
btn.addEventListener('click', async (e) => {
|
||||||
|
const folder = e.target.dataset.folder || e.target.closest('button').dataset.folder;
|
||||||
|
const item = document.querySelector(`.folder-item[data-folder="${folder}"]`);
|
||||||
|
|
||||||
|
btn.classList.add('searching');
|
||||||
|
btn.innerHTML = '<i class="fas fa-spinner fa-spin"></i> Searching...';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await reSearchFolder(folder);
|
||||||
|
// Update suggestions in place
|
||||||
|
const suggestionsEl = item.querySelector('.suggestion-list');
|
||||||
|
if (result.search_suggestions && result.search_suggestions.length > 0) {
|
||||||
|
suggestionsEl.innerHTML = result.search_suggestions.map(s => `
|
||||||
|
<div class="suggestion-item">
|
||||||
|
<i class="fas fa-link"></i>
|
||||||
|
<a href="${s.link}" class="suggestion-link" target="_blank">${s.title}</a>
|
||||||
|
</div>
|
||||||
|
`).join('');
|
||||||
|
} else {
|
||||||
|
suggestionsEl.innerHTML = '<div class="no-suggestions"><i class="fas fa-info-circle"></i> No suggestions found</div>';
|
||||||
|
}
|
||||||
|
btn.remove();
|
||||||
|
} catch (err) {
|
||||||
|
showToast('Search failed', 'error');
|
||||||
|
} finally {
|
||||||
|
btn.classList.remove('searching');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkEmptyList() {
|
||||||
|
const listEl = document.getElementById('folder-list');
|
||||||
|
const emptyEl = document.getElementById('empty-state');
|
||||||
|
const skipLink = document.getElementById('skip-link');
|
||||||
|
|
||||||
|
if (listEl.children.length === 0) {
|
||||||
|
listEl.style.display = 'none';
|
||||||
|
emptyEl.style.display = 'block';
|
||||||
|
skipLink.style.display = 'block';
|
||||||
|
showToast('All series configured!', 'success');
|
||||||
|
setTimeout(() => { window.location.href = '/'; }, 2000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Init
|
||||||
|
(async function init() {
|
||||||
|
const folders = await fetchUnresolved();
|
||||||
|
if (folders !== null) {
|
||||||
|
renderFolders(folders);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
Reference in New Issue
Block a user