wip settings section
This commit is contained in:
@@ -65,27 +65,6 @@ async function resolveEntryMetadata(entry: any, type: string) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export async function getConfig(_request: FastifyRequest, reply: FastifyReply) {
|
|
||||||
try {
|
|
||||||
return loadConfig();
|
|
||||||
} catch {
|
|
||||||
return reply.status(500).send({ error: 'FAILED_TO_LOAD_CONFIG' });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function setConfig(request: FastifyRequest<{ Body: SetConfigBody }>, reply: FastifyReply) {
|
|
||||||
try {
|
|
||||||
const { body } = request;
|
|
||||||
if (!body || typeof body !== 'object') {
|
|
||||||
return reply.status(400).send({ error: 'INVALID_BODY' });
|
|
||||||
}
|
|
||||||
return saveConfig(body);
|
|
||||||
} catch {
|
|
||||||
return reply.status(500).send({ error: 'FAILED_TO_SAVE_CONFIG' });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function scanLibrary(request: FastifyRequest<{ Querystring: ScanQuery }>, reply: FastifyReply) {
|
export async function scanLibrary(request: FastifyRequest<{ Querystring: ScanQuery }>, reply: FastifyReply) {
|
||||||
try {
|
try {
|
||||||
const mode = request.query.mode || 'incremental';
|
const mode = request.query.mode || 'incremental';
|
||||||
|
|||||||
@@ -2,8 +2,6 @@ import { FastifyInstance } from 'fastify';
|
|||||||
import * as controller from './local.controller';
|
import * as controller from './local.controller';
|
||||||
|
|
||||||
async function localRoutes(fastify: FastifyInstance) {
|
async function localRoutes(fastify: FastifyInstance) {
|
||||||
fastify.get('/library/config', controller.getConfig);
|
|
||||||
fastify.post('/library/config', controller.setConfig);
|
|
||||||
fastify.post('/library/scan', controller.scanLibrary);
|
fastify.post('/library/scan', controller.scanLibrary);
|
||||||
fastify.get('/library/:type', controller.listEntries);
|
fastify.get('/library/:type', controller.listEntries);
|
||||||
fastify.get('/library/:type/:id', controller.getEntry);
|
fastify.get('/library/:type/:id', controller.getEntry);
|
||||||
|
|||||||
@@ -43,6 +43,43 @@ async function loadMeUI() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Variable para saber si el modal ya fue cargado
|
||||||
|
let settingsModalLoaded = false;
|
||||||
|
|
||||||
|
document.getElementById('nav-settings').addEventListener('click', openSettings)
|
||||||
|
|
||||||
|
async function openSettings() {
|
||||||
|
if (!settingsModalLoaded) {
|
||||||
|
try {
|
||||||
|
const res = await fetch('/views/components/settings-modal.html')
|
||||||
|
const html = await res.text()
|
||||||
|
document.body.insertAdjacentHTML('beforeend', html)
|
||||||
|
settingsModalLoaded = true;
|
||||||
|
|
||||||
|
// Esperar un momento para que el DOM se actualice
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 50));
|
||||||
|
|
||||||
|
// Ahora cargar los settings
|
||||||
|
if (window.toggleSettingsModal) {
|
||||||
|
await window.toggleSettingsModal(false);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error loading settings modal:', err);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (window.toggleSettingsModal) {
|
||||||
|
await window.toggleSettingsModal(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeSettings() {
|
||||||
|
const modal = document.getElementById('settings-modal');
|
||||||
|
if (modal) {
|
||||||
|
modal.classList.add('hidden');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function setupDropdown() {
|
function setupDropdown() {
|
||||||
const userAvatarBtn = document.querySelector(".user-avatar-btn")
|
const userAvatarBtn = document.querySelector(".user-avatar-btn")
|
||||||
const navDropdown = document.getElementById("nav-dropdown")
|
const navDropdown = document.getElementById("nav-dropdown")
|
||||||
|
|||||||
218
desktop/src/scripts/settings.js
Normal file
218
desktop/src/scripts/settings.js
Normal file
@@ -0,0 +1,218 @@
|
|||||||
|
const API_BASE = '/api/config';
|
||||||
|
let currentConfig = {};
|
||||||
|
let activeSection = '';
|
||||||
|
let modal, navContainer, formContent, form;
|
||||||
|
|
||||||
|
window.toggleSettingsModal = async (forceClose = false) => {
|
||||||
|
modal = document.getElementById('settings-modal');
|
||||||
|
navContainer = document.getElementById('config-nav');
|
||||||
|
formContent = document.getElementById('config-section-content');
|
||||||
|
form = document.getElementById('config-form');
|
||||||
|
|
||||||
|
if (!modal) {
|
||||||
|
console.error('Modal not found');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (forceClose) {
|
||||||
|
modal.classList.add('hidden');
|
||||||
|
} else {
|
||||||
|
const isHidden = modal.classList.contains('hidden');
|
||||||
|
|
||||||
|
if (isHidden) {
|
||||||
|
// Abrir modal
|
||||||
|
modal.classList.remove('hidden');
|
||||||
|
await loadSettings();
|
||||||
|
} else {
|
||||||
|
// Cerrar modal
|
||||||
|
modal.classList.add('hidden');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
async function loadSettings() {
|
||||||
|
if (!formContent) {
|
||||||
|
console.error('Form content not found');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mostrar loading
|
||||||
|
formContent.innerHTML = `
|
||||||
|
<div class="skeleton-loader">
|
||||||
|
<div class="skeleton title-skeleton"></div>
|
||||||
|
<div class="skeleton text-skeleton"></div>
|
||||||
|
<div class="skeleton text-skeleton"></div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await fetch(API_BASE);
|
||||||
|
|
||||||
|
if (!res.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${res.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await res.json();
|
||||||
|
|
||||||
|
if (data.error) throw new Error(data.error);
|
||||||
|
|
||||||
|
currentConfig = data;
|
||||||
|
renderNav();
|
||||||
|
|
||||||
|
// Seleccionar la primera sección si no hay ninguna activa
|
||||||
|
if (!activeSection || !currentConfig[activeSection]) {
|
||||||
|
activeSection = Object.keys(currentConfig)[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
switchSection(activeSection);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error loading settings:', err);
|
||||||
|
formContent.innerHTML = `
|
||||||
|
<div style="padding: 2rem; text-align: center;">
|
||||||
|
<p style="color: var(--color-danger); margin-bottom: 1rem;">Failed to load settings</p>
|
||||||
|
<p style="color: var(--color-text-muted); font-size: 0.9rem;">${err.message}</p>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderNav() {
|
||||||
|
if (!navContainer) return;
|
||||||
|
|
||||||
|
navContainer.innerHTML = '';
|
||||||
|
Object.keys(currentConfig).forEach(section => {
|
||||||
|
const btn = document.createElement('div');
|
||||||
|
btn.className = `nav-item ${section === activeSection ? 'active' : ''}`;
|
||||||
|
btn.textContent = section;
|
||||||
|
btn.onclick = () => switchSection(section);
|
||||||
|
navContainer.appendChild(btn);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function switchSection(section) {
|
||||||
|
if (!currentConfig[section]) return;
|
||||||
|
|
||||||
|
activeSection = section;
|
||||||
|
renderNav();
|
||||||
|
|
||||||
|
const sectionData = currentConfig[section];
|
||||||
|
|
||||||
|
formContent.innerHTML = `
|
||||||
|
<h2 class="section-title" style="margin-bottom: 2rem; text-transform: capitalize;">
|
||||||
|
${section.replace(/_/g, ' ')}
|
||||||
|
</h2>
|
||||||
|
`;
|
||||||
|
|
||||||
|
Object.entries(sectionData).forEach(([key, value]) => {
|
||||||
|
const group = document.createElement('div');
|
||||||
|
group.className = 'config-group';
|
||||||
|
|
||||||
|
const isBool = typeof value === 'boolean';
|
||||||
|
const inputId = `input-${section}-${key}`;
|
||||||
|
const label = key.replace(/_/g, ' ');
|
||||||
|
|
||||||
|
if (isBool) {
|
||||||
|
group.innerHTML = `
|
||||||
|
<div style="display: flex; align-items: center; gap: 0.5rem;">
|
||||||
|
<input type="checkbox" id="${inputId}" name="${key}" ${value ? 'checked' : ''}>
|
||||||
|
<label for="${inputId}" style="margin: 0; cursor: pointer;">${label}</label>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
} else {
|
||||||
|
group.innerHTML = `
|
||||||
|
<label for="${inputId}">${label}</label>
|
||||||
|
<input class="config-input" id="${inputId}" name="${key}"
|
||||||
|
type="${typeof value === 'number' ? 'number' : 'text'}"
|
||||||
|
value="${value}">
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
formContent.appendChild(group);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup form submit handler
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
// Usar delegación de eventos ya que el form se carga dinámicamente
|
||||||
|
document.addEventListener('submit', async (e) => {
|
||||||
|
if (e.target.id === 'config-form') {
|
||||||
|
e.preventDefault();
|
||||||
|
await saveSettings();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
async function saveSettings() {
|
||||||
|
if (!form || !activeSection) return;
|
||||||
|
|
||||||
|
const updatedData = {};
|
||||||
|
|
||||||
|
Object.keys(currentConfig[activeSection]).forEach(key => {
|
||||||
|
const input = form.elements[key];
|
||||||
|
if (!input) return;
|
||||||
|
|
||||||
|
if (input.type === 'checkbox') {
|
||||||
|
updatedData[key] = input.checked;
|
||||||
|
} else if (input.type === 'number') {
|
||||||
|
updatedData[key] = Number(input.value);
|
||||||
|
} else {
|
||||||
|
updatedData[key] = input.value;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await fetch(`${API_BASE}/${activeSection}`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify(updatedData)
|
||||||
|
});
|
||||||
|
|
||||||
|
if (res.ok) {
|
||||||
|
currentConfig[activeSection] = updatedData;
|
||||||
|
|
||||||
|
// Mostrar notificación de éxito
|
||||||
|
const notification = document.createElement('div');
|
||||||
|
notification.style.cssText = `
|
||||||
|
position: fixed;
|
||||||
|
top: 20px;
|
||||||
|
right: 20px;
|
||||||
|
background: var(--color-success, #10b981);
|
||||||
|
color: white;
|
||||||
|
padding: 1rem 1.5rem;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 4px 12px rgba(0,0,0,0.3);
|
||||||
|
z-index: 10000;
|
||||||
|
animation: slideIn 0.3s ease-out;
|
||||||
|
`;
|
||||||
|
notification.textContent = 'Settings saved successfully!';
|
||||||
|
document.body.appendChild(notification);
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
notification.style.animation = 'slideOut 0.3s ease-out';
|
||||||
|
setTimeout(() => notification.remove(), 300);
|
||||||
|
}, 2000);
|
||||||
|
} else {
|
||||||
|
throw new Error('Failed to save settings');
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error saving settings:', err);
|
||||||
|
alert('Error saving settings: ' + err.message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Añadir estilos para las animaciones (solo si no existen)
|
||||||
|
if (!document.getElementById('settings-animations')) {
|
||||||
|
const animationStyles = document.createElement('style');
|
||||||
|
animationStyles.id = 'settings-animations';
|
||||||
|
animationStyles.textContent = `
|
||||||
|
@keyframes slideIn {
|
||||||
|
from { transform: translateX(400px); opacity: 0; }
|
||||||
|
to { transform: translateX(0); opacity: 1; }
|
||||||
|
}
|
||||||
|
@keyframes slideOut {
|
||||||
|
from { transform: translateX(0); opacity: 1; }
|
||||||
|
to { transform: translateX(400px); opacity: 0; }
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
document.head.appendChild(animationStyles);
|
||||||
|
}
|
||||||
@@ -68,6 +68,14 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<button class="dropdown-item" id="nav-settings">
|
||||||
|
<svg width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
||||||
|
<circle cx="12" cy="12" r="3"/>
|
||||||
|
<path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 1 1-2.83 2.83l-.06-.06A1.65 1.65 0 0 0 15 19.4a1.65 1.65 0 0 0-1 .6 1.65 1.65 0 0 0-.33 1.82V22a2 2 0 1 1-4 0v-.18a1.65 1.65 0 0 0-.33-1.82 1.65 1.65 0 0 0-1-.6 1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 1 1-2.83-2.83l.06-.06A1.65 1.65 0 0 0 4.6 15a1.65 1.65 0 0 0-.6-1 1.65 1.65 0 0 0-1.82-.33H2a2 2 0 1 1 0-4h.18a1.65 1.65 0 0 0 1.82-.33 1.65 1.65 0 0 0 .6-1 1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 1 1 2.83-2.83l.06.06A1.65 1.65 0 0 0 9 4.6c.37 0 .72-.14 1-.6A1.65 1.65 0 0 0 10.33 2.18V2a2 2 0 1 1 4 0v.18a1.65 1.65 0 0 0 .33 1.82c.28.46.63.6 1 .6a1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 1 1 2.83 2.83l-.06.06A1.65 1.65 0 0 0 19.4 9c0 .37.14.72.6 1 .46.28.6.63.6 1z"/>
|
||||||
|
</svg>
|
||||||
|
<span>Settings</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
<a href="/my-list" class="dropdown-item">
|
<a href="/my-list" class="dropdown-item">
|
||||||
<svg width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
<svg width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
||||||
<path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/>
|
<path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/>
|
||||||
@@ -303,5 +311,6 @@
|
|||||||
<script src="/src/scripts/updateNotifier.js"></script>
|
<script src="/src/scripts/updateNotifier.js"></script>
|
||||||
<script src="/src/scripts/rpc-inapp.js"></script>
|
<script src="/src/scripts/rpc-inapp.js"></script>
|
||||||
<script src="/src/scripts/auth-guard.js"></script>
|
<script src="/src/scripts/auth-guard.js"></script>
|
||||||
|
<script src="/src/scripts/settings.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
62
desktop/views/components/navbar.html
Normal file
62
desktop/views/components/navbar.html
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
<nav class="navbar" id="navbar">
|
||||||
|
<a href="#" class="nav-brand">
|
||||||
|
<div class="brand-icon">
|
||||||
|
<img src="/public/assets/waifuboards.ico" alt="WF Logo">
|
||||||
|
</div>
|
||||||
|
WaifuBoard
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<nav class="navbar">
|
||||||
|
<div class="nav-center">
|
||||||
|
<a href="/anime" class="nav-button {{anime}}">Anime</a>
|
||||||
|
<a href="/books" class="nav-button {{books}}">Books</a>
|
||||||
|
<a href="/gallery" class="nav-button {{gallery}}">Gallery</a>
|
||||||
|
<a href="/schedule" class="nav-button {{schedule}}">Schedule</a>
|
||||||
|
<a href="/my-list" class="nav-button {{myList}}">My List</a>
|
||||||
|
<a href="/marketplace" class="nav-button {{marketplace}}">Marketplace</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="nav-right">
|
||||||
|
<div class="search-wrapper {{hideSearch}}">
|
||||||
|
<svg class="search-icon" width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
||||||
|
<circle cx="11" cy="11" r="8"/>
|
||||||
|
<path d="M21 21l-4.35-4.35"/>
|
||||||
|
</svg>
|
||||||
|
<input type="text" class="search-input" id="search-input" placeholder="Search anime..." autocomplete="off">
|
||||||
|
<div class="search-results" id="search-results"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="nav-user" id="nav-user" style="display:none;">
|
||||||
|
<div class="user-avatar-btn">
|
||||||
|
<img id="nav-avatar" src="/public/assets/waifuboards.ico" alt="avatar">
|
||||||
|
<div class="online-indicator"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="nav-dropdown" id="nav-dropdown">
|
||||||
|
<div class="dropdown-header">
|
||||||
|
<img id="dropdown-avatar" src="/public/assets/waifuboards.ico" alt="avatar" class="dropdown-avatar">
|
||||||
|
<div class="dropdown-user-info">
|
||||||
|
<div class="dropdown-username" id="nav-username"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<a href="/my-list" class="dropdown-item">
|
||||||
|
<svg width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
||||||
|
<path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/>
|
||||||
|
<polyline points="17 21 17 13 7 13 7 21"/>
|
||||||
|
<polyline points="7 3 7 8 15 8"/>
|
||||||
|
</svg>
|
||||||
|
<span>My List</span>
|
||||||
|
</a>
|
||||||
|
<button class="dropdown-item logout-item" id="nav-logout">
|
||||||
|
<svg width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" viewBox="0 0 24 24">
|
||||||
|
<path d="M9 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h4"/>
|
||||||
|
<polyline points="16 17 21 12 16 7"/>
|
||||||
|
<line x1="21" y1="12" x2="9" y2="12"/>
|
||||||
|
</svg>
|
||||||
|
<span>Logout</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
278
desktop/views/components/settings-modal.html
Normal file
278
desktop/views/components/settings-modal.html
Normal file
@@ -0,0 +1,278 @@
|
|||||||
|
<div id="settings-modal" class="modal hidden" onclick="if(event.target === this) window.toggleSettingsModal(true)">
|
||||||
|
<div class="modal-overlay"></div>
|
||||||
|
<div class="modal-content">
|
||||||
|
|
||||||
|
<aside class="modal-sidebar">
|
||||||
|
<div class="sidebar-header">
|
||||||
|
<h2 class="sidebar-title">Settings</h2>
|
||||||
|
</div>
|
||||||
|
<nav id="config-nav" class="nav-list">
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="sidebar-footer">
|
||||||
|
<button onclick="window.toggleSettingsModal(true)" class="btn-exit">
|
||||||
|
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M9 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h4"></path>
|
||||||
|
<polyline points="16 17 21 12 16 7"></polyline>
|
||||||
|
<line x1="21" y1="12" x2="9" y2="12"></line>
|
||||||
|
</svg>
|
||||||
|
Close
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</aside>
|
||||||
|
|
||||||
|
<main class="modal-main">
|
||||||
|
<form id="config-form" class="config-wrapper">
|
||||||
|
<div id="config-section-content" class="section-container">
|
||||||
|
<div class="skeleton-loader">
|
||||||
|
<div class="skeleton title-skeleton"></div>
|
||||||
|
<div class="skeleton field-skeleton"></div>
|
||||||
|
<div class="skeleton field-skeleton"></div>
|
||||||
|
<div class="skeleton field-skeleton"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="modal-footer-sticky">
|
||||||
|
<p class="footer-hint">Changes are applied immediately after saving.</p>
|
||||||
|
<button type="submit" class="btn-primary">Save Changes</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* --- AMOLED THEME VARIABLES --- */
|
||||||
|
:root {
|
||||||
|
--amoled-black: #000000;
|
||||||
|
--amoled-surface: #080808;
|
||||||
|
--amoled-field: #0e0e0e;
|
||||||
|
--amoled-border: rgba(255, 255, 255, 0.08);
|
||||||
|
--accent-purple: #8b5cf6;
|
||||||
|
--accent-glow: rgba(139, 92, 246, 0.15);
|
||||||
|
--text-main: #ffffff;
|
||||||
|
--text-dim: #a1a1aa;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- MODAL BASE --- */
|
||||||
|
.modal {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 10000;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal.hidden { display: none !important; }
|
||||||
|
|
||||||
|
.modal-overlay {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
background: rgba(0, 0, 0, 0.9);
|
||||||
|
backdrop-filter: blur(12px);
|
||||||
|
z-index: -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal-content {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row; /* Horizontal layout */
|
||||||
|
width: 95%;
|
||||||
|
max-width: 1200px; /* Increased size */
|
||||||
|
height: 85vh;
|
||||||
|
background: var(--amoled-black);
|
||||||
|
border: var(--amoled-border);
|
||||||
|
border-radius: 28px;
|
||||||
|
overflow: hidden;
|
||||||
|
box-shadow: 0 0 0 1px var(--amoled-border), 0 24px 60px rgba(0,0,0,0.8);
|
||||||
|
animation: modalScaleUp 0.4s cubic-bezier(0.16, 1, 0.3, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- SIDEBAR --- */
|
||||||
|
.modal-sidebar {
|
||||||
|
width: 280px;
|
||||||
|
background: var(--amoled-surface);
|
||||||
|
border-right: var(--amoled-border);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 2rem;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar-title {
|
||||||
|
font-size: 1.4rem;
|
||||||
|
font-weight: 800;
|
||||||
|
margin-bottom: 2.5rem;
|
||||||
|
color: var(--text-main);
|
||||||
|
letter-spacing: -0.02em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-list { flex: 1; }
|
||||||
|
|
||||||
|
.nav-item {
|
||||||
|
padding: 12px 16px;
|
||||||
|
border-radius: 14px;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--text-dim);
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
font-weight: 500;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
text-transform: capitalize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
color: var(--text-main);
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item.active {
|
||||||
|
background: var(--accent-glow);
|
||||||
|
color: var(--accent-purple);
|
||||||
|
box-shadow: inset 3px 0 0 var(--accent-purple);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- MAIN CONTENT & DYNAMIC INPUTS --- */
|
||||||
|
.modal-main {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background: var(--amoled-black);
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-wrapper {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-container {
|
||||||
|
flex: 1;
|
||||||
|
padding: 3.5rem;
|
||||||
|
overflow-y: auto;
|
||||||
|
scrollbar-width: thin;
|
||||||
|
scrollbar-color: #222 transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Styles for the injected section content */
|
||||||
|
.config-group {
|
||||||
|
margin-bottom: 2.5rem;
|
||||||
|
animation: fadeInSection 0.4s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-group label {
|
||||||
|
display: block;
|
||||||
|
font-size: 0.75rem;
|
||||||
|
color: var(--accent-purple);
|
||||||
|
margin-bottom: 0.8rem;
|
||||||
|
letter-spacing: 0.05em;
|
||||||
|
font-weight: 800;
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-input {
|
||||||
|
width: 100%;
|
||||||
|
padding: 1rem 1.2rem;
|
||||||
|
background: var(--amoled-field);
|
||||||
|
border: 1px solid #1a1a1a;
|
||||||
|
border-radius: 14px;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 1rem;
|
||||||
|
transition: all 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-input:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--accent-purple);
|
||||||
|
background: #121212;
|
||||||
|
box-shadow: 0 0 0 4px var(--accent-glow);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- FOOTER --- */
|
||||||
|
.modal-footer-sticky {
|
||||||
|
padding: 1.5rem 3.5rem;
|
||||||
|
background: rgba(0, 0, 0, 0.8);
|
||||||
|
backdrop-filter: blur(10px);
|
||||||
|
border-top: var(--amoled-border);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-hint {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: var(--text-dim);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- BUTTONS --- */
|
||||||
|
.btn-primary {
|
||||||
|
padding: 0.8rem 2.2rem;
|
||||||
|
background: #ffffff;
|
||||||
|
color: #000000;
|
||||||
|
border: none;
|
||||||
|
border-radius: 100px;
|
||||||
|
font-weight: 700;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-primary:hover {
|
||||||
|
transform: translateY(-2px);
|
||||||
|
background: #f0f0f0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-exit {
|
||||||
|
background: #111;
|
||||||
|
border: 1px solid #222;
|
||||||
|
color: #ef4444;
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: 12px;
|
||||||
|
width: 100%;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-top: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- ANIMATIONS & SKELETON --- */
|
||||||
|
@keyframes modalScaleUp {
|
||||||
|
from { opacity: 0; transform: scale(0.97) translateY(10px); }
|
||||||
|
to { opacity: 1; transform: scale(1) translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeInSection {
|
||||||
|
from { opacity: 0; transform: translateY(8px); }
|
||||||
|
to { opacity: 1; transform: translateY(0); }
|
||||||
|
}
|
||||||
|
|
||||||
|
.skeleton-loader { display: flex; flex-direction: column; gap: 2rem; }
|
||||||
|
.skeleton {
|
||||||
|
background: linear-gradient(90deg, #080808 25%, #121212 50%, #080808 75%);
|
||||||
|
background-size: 200% 100%;
|
||||||
|
animation: shimmer 1.5s infinite;
|
||||||
|
border-radius: 12px;
|
||||||
|
}
|
||||||
|
@keyframes shimmer {
|
||||||
|
0% { background-position: 200% 0; }
|
||||||
|
100% { background-position: -200% 0; }
|
||||||
|
}
|
||||||
|
.title-skeleton { height: 35px; width: 40%; }
|
||||||
|
.field-skeleton { height: 55px; width: 100%; }
|
||||||
|
|
||||||
|
/* Responsive Mobile View */
|
||||||
|
@media (max-width: 850px) {
|
||||||
|
.modal-content { flex-direction: column; height: 95vh; width: 100vw; border-radius: 0; }
|
||||||
|
.modal-sidebar { width: 100%; height: auto; border-right: none; border-bottom: var(--amoled-border); padding: 1rem; }
|
||||||
|
.sidebar-title { margin-bottom: 1rem; font-size: 1.2rem; }
|
||||||
|
.section-container { padding: 2rem; }
|
||||||
|
.modal-footer-sticky { padding: 1.5rem 2rem; }
|
||||||
|
}
|
||||||
|
</style>
|
||||||
Reference in New Issue
Block a user