* feat: implement light/dark mode theme system - Add semantic color CSS variables (success, warning, info, error) for both themes - Create ThemeProvider with React context and localStorage persistence - Add ThemeToggle component with sun/moon icons for header region - Add theme switcher in General Settings modal - Replace 200+ hardcoded Tailwind colors with CSS variables across 30+ components - Update layout.tsx to remove forced dark mode - Keep terminal colors unchanged as requested - Default to dark mode, with seamless light/dark switching Components updated: - High-priority: InstalledScriptsTab, ScriptInstallationCard, LXCSettingsModal, ScriptsGrid - All remaining component files with hardcoded colors - UI components: button, toggle, badge variants - Modal components: ErrorModal, ConfirmationModal, AuthModal, SetupModal - Form components: ServerForm, FilterBar, CategorySidebar - Display components: ScriptCard, ScriptCardList, DiffViewer, TextViewer Theme switchers: - Header: Small nuanced toggle in top-right - Settings: Detailed Light/Dark selection in General Settings * fix: resolve ESLint warnings - Fix missing dependencies in useCallback and useEffect hooks - Prefix unused parameter with underscore to satisfy ESLint rules - Build now completes without warnings * fix: improve toggle component styling for better visibility - Use explicit gray colors instead of CSS variables for toggle background - Ensure proper contrast in both light and dark modes - Toggle switches now display correctly with proper visual states * fix: improve toggle visual states for better UX - Use explicit conditional styling instead of peer classes - Active toggles now clearly show primary color background - Inactive toggles show gray background for clear distinction - Much easier to tell which toggles are on/off at a glance * fix: improve toggle contrast in dark mode - Change inactive toggle background from gray-700 to gray-600 for better visibility - Add darker border color (gray-500) for toggle handle in dark mode - Toggles now have proper contrast against dark backgrounds - Both light and dark modes now have clear visual distinction * fix: resolve dependency loop and improve dropdown styling - Fix circular dependency in InstalledScriptsTab status check - Remove fetchContainerStatuses function and inline logic in useEffect - Make all dropdown menu items grey with consistent hover effects - Update both ScriptInstallationCard and InstalledScriptsTab dropdowns - Remove unused useCallback import - Build now completes without warnings or errors * fix: restore proper button colors and eliminate dependency loop - Restore red color for Stop/Destroy buttons and green for Start buttons - Fix circular dependency by using ref for containerStatusMutation - Update both InstalledScriptsTab and ScriptInstallationCard dropdowns - Maintain grey color for other menu items (Update, Shell, Open UI, etc.) - Build now completes without warnings or dependency loops * feat: add missing hover utility classes for semantic colors - Add hover states for success, warning, info, error colors - Add hover:bg-success/20, hover:bg-error/20, etc. classes - Add hover:text-success-foreground, hover:text-error-foreground classes - Start/Stop and Destroy buttons now have proper hover effects - All dropdown menu items now have consistent hover behavior * feat: improve status cards with useful LXC container information - Replace useless 'Successful/Failed/In Progress' cards with meaningful data - Show 'Running LXC' count in green (actual running containers) - Show 'Stopped LXC' count in red (actual stopped containers) - Keep 'Total Installations' for overall count - Change layout from 4 columns to 3 columns for better spacing - Status cards now show real-time container states instead of installation status * style: center content in status cards - Add text-center class to each individual status card - Numbers and labels now centered within each card - Improves visual balance and readability - All three cards (Total, Running LXC, Stopped LXC) now have centered content
74 lines
2.1 KiB
TypeScript
74 lines
2.1 KiB
TypeScript
'use client';
|
|
|
|
import { useState, useEffect, type ReactNode } from 'react';
|
|
import { useAuth } from './AuthProvider';
|
|
import { AuthModal } from './AuthModal';
|
|
import { SetupModal } from './SetupModal';
|
|
|
|
interface AuthGuardProps {
|
|
children: ReactNode;
|
|
}
|
|
|
|
interface AuthConfig {
|
|
username: string | null;
|
|
enabled: boolean;
|
|
hasCredentials: boolean;
|
|
setupCompleted: boolean;
|
|
}
|
|
|
|
export function AuthGuard({ children }: AuthGuardProps) {
|
|
const { isAuthenticated, isLoading } = useAuth();
|
|
const [authConfig, setAuthConfig] = useState<AuthConfig | null>(null);
|
|
const [configLoading, setConfigLoading] = useState(true);
|
|
const [setupCompleted, setSetupCompleted] = useState(false);
|
|
|
|
const handleSetupComplete = async () => {
|
|
setSetupCompleted(true);
|
|
// Refresh auth config without reloading the page
|
|
await fetchAuthConfig();
|
|
};
|
|
|
|
const fetchAuthConfig = async () => {
|
|
try {
|
|
const response = await fetch('/api/settings/auth-credentials');
|
|
if (response.ok) {
|
|
const config = await response.json() as AuthConfig;
|
|
setAuthConfig(config);
|
|
}
|
|
} catch (error) {
|
|
console.error('Error fetching auth config:', error);
|
|
} finally {
|
|
setConfigLoading(false);
|
|
}
|
|
};
|
|
|
|
useEffect(() => {
|
|
void fetchAuthConfig();
|
|
}, []);
|
|
|
|
// Show loading while checking auth status
|
|
if (isLoading || configLoading) {
|
|
return (
|
|
<div className="min-h-screen bg-background flex items-center justify-center">
|
|
<div className="text-center">
|
|
<div className="inline-block animate-spin rounded-full h-8 w-8 border-b-2 border-primary mb-4"></div>
|
|
<p className="text-muted-foreground">Loading...</p>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
// Show setup modal if setup has not been completed yet
|
|
if (authConfig && !authConfig.setupCompleted && !setupCompleted) {
|
|
return <SetupModal isOpen={true} onComplete={handleSetupComplete} />;
|
|
}
|
|
|
|
// Show auth modal if auth is enabled but user is not authenticated
|
|
if (authConfig && authConfig.enabled && !isAuthenticated) {
|
|
return <AuthModal isOpen={true} />;
|
|
}
|
|
|
|
// Render children if authenticated or auth is disabled
|
|
return <>{children}</>;
|
|
}
|