Files
LocalAI/core/http/views/partials/head.html
Ettore Di Giacinto 2fabdc08e6 feat(ui): left navbar, dark/light theme (#8594)
* feat(ui): left navbar, dark/light theme

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>

* darker background

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>

---------

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>
2026-02-18 00:14:39 +01:00

140 lines
5.0 KiB
HTML

<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{{.Title}}</title>
<base href="{{.BaseURL}}" />
<link rel="shortcut icon" href="static/favicon.svg" type="image/svg">
<link rel="stylesheet" href="static/assets/highlightjs.css" />
<script defer src="static/assets/highlightjs.js"></script>
<script defer src="static/assets/alpine.js"></script>
<script defer src="static/assets/marked.js"></script>
<script defer src="static/assets/purify.js"></script>
<!-- LocalAI Design System CSS -->
<link href="static/theme.css" rel="stylesheet" />
<link href="static/typography.css" rel="stylesheet" />
<link href="static/animations.css" rel="stylesheet" />
<link href="static/components.css" rel="stylesheet" />
<link href="static/general.css" rel="stylesheet" />
<link rel="stylesheet" href="static/assets/tw-elements.css" />
<script src="static/assets/tailwindcss.js"></script>
<!-- Preload critical fonts -->
<link rel="preload" href="static/assets/playfair-display-bold.ttf" as="font" type="font/ttf" crossorigin>
<link rel="preload" href="static/assets/space-grotesk-regular.ttf" as="font" type="font/ttf" crossorigin>
<script>
tailwind.config = {
darkMode: "class",
theme: {
fontFamily: {
sans: ["Space Grotesk", "-apple-system", "BlinkMacSystemFont", "sans-serif"],
body: ["Space Grotesk", "-apple-system", "BlinkMacSystemFont", "sans-serif"],
display: ["Playfair Display", "serif"],
mono: ["JetBrains Mono", "Fira Code", "monospace"],
},
},
corePlugins: {
preflight: false,
},
};
// Theme Management
(function() {
const THEME_KEY = 'localai-theme';
const DARK = 'dark';
const LIGHT = 'light';
function getStoredTheme() {
return localStorage.getItem(THEME_KEY);
}
function getSystemTheme() {
return window.matchMedia('(prefers-color-scheme: dark)').matches ? DARK : LIGHT;
}
function applyTheme(theme) {
document.documentElement.setAttribute('data-theme', theme);
// Also set class for Tailwind compatibility
if (theme === DARK) {
document.documentElement.classList.add('dark');
} else {
document.documentElement.classList.remove('dark');
}
}
function toggleTheme() {
const current = document.documentElement.getAttribute('data-theme') || DARK;
const newTheme = current === DARK ? LIGHT : DARK;
localStorage.setItem(THEME_KEY, newTheme);
applyTheme(newTheme);
return newTheme;
}
// Initialize theme immediately to prevent flash
const stored = getStoredTheme();
const initialTheme = stored || getSystemTheme();
applyTheme(initialTheme);
// Expose toggle function globally
window.toggleTheme = toggleTheme;
window.getCurrentTheme = function() {
return document.documentElement.getAttribute('data-theme') || DARK;
};
// Listen for system theme changes
window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', function(e) {
if (!getStoredTheme()) {
applyTheme(e.matches ? DARK : LIGHT);
}
});
})();
function copyClipboard(token) {
// Try modern Clipboard API first (requires secure context)
if (navigator.clipboard && window.isSecureContext) {
navigator.clipboard.writeText(token)
.then(() => {
console.log('Text copied to clipboard:', token);
alert('Text copied to clipboard!');
})
.catch(err => {
console.error('Failed to copy token:', err);
fallbackCopy(token);
});
} else {
// Fallback for non-secure contexts
fallbackCopy(token);
}
}
function fallbackCopy(text) {
const textArea = document.createElement("textarea");
textArea.value = text;
textArea.style.position = "fixed";
textArea.style.left = "-999999px";
textArea.style.top = "-999999px";
document.body.appendChild(textArea);
textArea.focus();
textArea.select();
try {
const successful = document.execCommand('copy');
if (successful) {
console.log('Text copied to clipboard (fallback):', text);
alert('Text copied to clipboard!');
} else {
console.error('Fallback copy failed');
alert('Failed to copy text. Please copy manually.');
}
} catch (err) {
console.error('Fallback copy error:', err);
alert('Failed to copy text. Please copy manually.');
}
document.body.removeChild(textArea);
}
</script>
<link href="static/assets/fontawesome/css/fontawesome.css" rel="stylesheet" />
<link href="static/assets/fontawesome/css/brands.css" rel="stylesheet" />
<link href="static/assets/fontawesome/css/solid.css" rel="stylesheet" />
<script src="static/assets/flowbite.min.js"></script>
</head>