Marketplace: - Add skeleton loading states for stats grid and GPU offer cards - Show animated skeleton placeholders during data fetch - Add skeleton CSS with shimmer animation and dark mode support - Wrap stats section in #stats-grid container for skeleton injection Trade Exchange: - Replace inline header with data-global-header component - Switch GPU offers to production API (/api/miners/list) - Add fallback to demo
119 lines
3.9 KiB
JavaScript
119 lines
3.9 KiB
JavaScript
(function () {
|
|
const NAV_ITEMS = [
|
|
{ key: 'home', label: 'Home', href: '/' },
|
|
{ key: 'explorer', label: 'Explorer', href: '/explorer/' },
|
|
{ key: 'marketplace', label: 'Marketplace', href: '/marketplace/' },
|
|
{ key: 'exchange', label: 'Exchange', href: '/Exchange/' },
|
|
{ key: 'docs', label: 'Docs', href: '/docs/index.html' },
|
|
];
|
|
|
|
const CTA = { label: 'Launch Marketplace', href: '/marketplace/' };
|
|
|
|
function determineActiveKey(pathname) {
|
|
if (pathname.startsWith('/explorer')) return 'explorer';
|
|
if (pathname.startsWith('/marketplace')) return 'marketplace';
|
|
if (pathname.toLowerCase().startsWith('/exchange')) return 'exchange';
|
|
if (pathname.startsWith('/docs')) return 'docs';
|
|
return 'home';
|
|
}
|
|
|
|
function buildHeader(activeKey) {
|
|
const navLinks = NAV_ITEMS.map((item) => {
|
|
const active = item.key === activeKey ? 'active' : '';
|
|
return `<a href="${item.href}" class="global-nav__link ${active}">${item.label}</a>`;
|
|
}).join('');
|
|
|
|
return `
|
|
<header class="global-header">
|
|
<div class="global-header__inner">
|
|
<a class="global-brand" href="/">
|
|
<div class="global-brand__icon">
|
|
<i class="fas fa-cube"></i>
|
|
</div>
|
|
<div class="global-brand__text">
|
|
<span>AITBC Platform</span>
|
|
<small>AI Blockchain Network</small>
|
|
</div>
|
|
</a>
|
|
<nav class="global-nav">${navLinks}</nav>
|
|
<div class="global-header__actions">
|
|
<button type="button" class="global-dark-toggle" data-role="global-theme-toggle">
|
|
<span class="global-dark-toggle__emoji">🌙</span>
|
|
<span class="global-dark-toggle__text">Dark</span>
|
|
</button>
|
|
<a href="${CTA.href}" class="global-nav__cta">${CTA.label}</a>
|
|
</div>
|
|
</div>
|
|
</header>
|
|
`;
|
|
}
|
|
|
|
function getCurrentTheme() {
|
|
if (document.documentElement.hasAttribute('data-theme')) {
|
|
return document.documentElement.getAttribute('data-theme');
|
|
}
|
|
if (document.documentElement.classList.contains('dark')) return 'dark';
|
|
if (document.body && document.body.classList.contains('light')) return 'light';
|
|
return 'light';
|
|
}
|
|
|
|
function updateToggleLabel(theme) {
|
|
const emojiEl = document.querySelector('.global-dark-toggle__emoji');
|
|
const textEl = document.querySelector('.global-dark-toggle__text');
|
|
if (!emojiEl || !textEl) return;
|
|
if (theme === 'dark') {
|
|
emojiEl.textContent = '🌙';
|
|
textEl.textContent = 'Dark';
|
|
} else {
|
|
emojiEl.textContent = '☀️';
|
|
textEl.textContent = 'Light';
|
|
}
|
|
}
|
|
|
|
function bindThemeToggle() {
|
|
const toggle = document.querySelector('[data-role="global-theme-toggle"]');
|
|
if (!toggle) return;
|
|
|
|
toggle.addEventListener('click', () => {
|
|
if (typeof window.toggleDarkMode === 'function') {
|
|
window.toggleDarkMode();
|
|
} else if (typeof window.toggleTheme === 'function') {
|
|
window.toggleTheme();
|
|
} else {
|
|
const isDark = document.documentElement.classList.toggle('dark');
|
|
if (isDark) {
|
|
document.documentElement.setAttribute('data-theme', 'dark');
|
|
} else {
|
|
document.documentElement.removeAttribute('data-theme');
|
|
}
|
|
}
|
|
|
|
setTimeout(() => updateToggleLabel(getCurrentTheme()), 0);
|
|
});
|
|
|
|
updateToggleLabel(getCurrentTheme());
|
|
}
|
|
|
|
function initHeader() {
|
|
const activeKey = determineActiveKey(window.location.pathname);
|
|
const headerHTML = buildHeader(activeKey);
|
|
|
|
const placeholder = document.querySelector('[data-global-header]');
|
|
const existing = placeholder || document.querySelector('.global-header');
|
|
|
|
if (existing) {
|
|
existing.outerHTML = headerHTML;
|
|
} else {
|
|
document.body.insertAdjacentHTML('afterbegin', headerHTML);
|
|
}
|
|
|
|
bindThemeToggle();
|
|
}
|
|
|
|
if (document.readyState === 'loading') {
|
|
document.addEventListener('DOMContentLoaded', initHeader);
|
|
} else {
|
|
initHeader();
|
|
}
|
|
})();
|