mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-12 07:20:45 +00:00
docs(ui): add animated underline for nav tabs (#21912)
Add a responsive, animated underline indicator for navigation tabs to improve visual focus and active-state feedback. - Introduce CSS for .nav-tabs, .nav-tabs-item and a .nav-tabs-underline element, including transitions, positioning, and dark mode color. - Hide default first h1 in #content to keep header layout consistent. - Add docs/nav-tabs-underline.js to create and manage the underline element, observe DOM mutations, and update underline position/width on changes, resize, and when fonts load. - Preserve last known underline position/width across re-initializations to avoid visual jumps. This change makes active tab state visible with smooth movement and ensures the underline stays synchronized with dynamic content.
This commit is contained in:
100
docs/nav-tabs-underline.js
Normal file
100
docs/nav-tabs-underline.js
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
(() => {
|
||||||
|
const NAV_TABS_SELECTOR = ".nav-tabs";
|
||||||
|
const ACTIVE_UNDERLINE_SELECTOR = ".nav-tabs-item > div.bg-primary";
|
||||||
|
const UNDERLINE_CLASS = "nav-tabs-underline";
|
||||||
|
const READY_CLASS = "nav-tabs-underline-ready";
|
||||||
|
|
||||||
|
let navTabs = null;
|
||||||
|
let navTabsObserver = null;
|
||||||
|
let lastX = null;
|
||||||
|
let lastWidth = null;
|
||||||
|
|
||||||
|
const ensureUnderline = (tabs) => {
|
||||||
|
let underline = tabs.querySelector(`.${UNDERLINE_CLASS}`);
|
||||||
|
if (!underline) {
|
||||||
|
underline = document.createElement("div");
|
||||||
|
underline.className = UNDERLINE_CLASS;
|
||||||
|
tabs.appendChild(underline);
|
||||||
|
}
|
||||||
|
return underline;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getActiveTab = (tabs) => {
|
||||||
|
const activeUnderline = tabs.querySelector(ACTIVE_UNDERLINE_SELECTOR);
|
||||||
|
return activeUnderline?.closest(".nav-tabs-item") ?? null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateUnderline = () => {
|
||||||
|
if (!navTabs) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ensureUnderline(navTabs);
|
||||||
|
|
||||||
|
const activeTab = getActiveTab(navTabs);
|
||||||
|
if (!activeTab) {
|
||||||
|
navTabs.classList.remove(READY_CLASS);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const navRect = navTabs.getBoundingClientRect();
|
||||||
|
const tabRect = activeTab.getBoundingClientRect();
|
||||||
|
const left = tabRect.left - navRect.left;
|
||||||
|
|
||||||
|
navTabs.style.setProperty("--nav-tab-underline-x", `${left}px`);
|
||||||
|
navTabs.style.setProperty("--nav-tab-underline-width", `${tabRect.width}px`);
|
||||||
|
navTabs.classList.add(READY_CLASS);
|
||||||
|
|
||||||
|
lastX = left;
|
||||||
|
lastWidth = tabRect.width;
|
||||||
|
};
|
||||||
|
|
||||||
|
const scheduleUpdate = () => {
|
||||||
|
requestAnimationFrame(updateUnderline);
|
||||||
|
};
|
||||||
|
|
||||||
|
const setupNavTabsObserver = (tabs) => {
|
||||||
|
if (!tabs || tabs === navTabs) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
navTabs = tabs;
|
||||||
|
ensureUnderline(navTabs);
|
||||||
|
if (lastX !== null && lastWidth !== null) {
|
||||||
|
navTabs.style.setProperty("--nav-tab-underline-x", `${lastX}px`);
|
||||||
|
navTabs.style.setProperty("--nav-tab-underline-width", `${lastWidth}px`);
|
||||||
|
navTabs.classList.add(READY_CLASS);
|
||||||
|
}
|
||||||
|
navTabsObserver?.disconnect();
|
||||||
|
navTabsObserver = new MutationObserver(scheduleUpdate);
|
||||||
|
navTabsObserver.observe(navTabs, {
|
||||||
|
subtree: true,
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
|
scheduleUpdate();
|
||||||
|
};
|
||||||
|
|
||||||
|
const setupObservers = () => {
|
||||||
|
const tabs = document.querySelector(NAV_TABS_SELECTOR);
|
||||||
|
if (tabs) {
|
||||||
|
setupNavTabsObserver(tabs);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const rootObserver = new MutationObserver(setupObservers);
|
||||||
|
|
||||||
|
if (document.readyState === "loading") {
|
||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
setupObservers();
|
||||||
|
rootObserver.observe(document.body, { childList: true, subtree: true });
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setupObservers();
|
||||||
|
rootObserver.observe(document.body, { childList: true, subtree: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener("resize", scheduleUpdate);
|
||||||
|
void document.fonts?.ready?.then(scheduleUpdate, () => {});
|
||||||
|
})();
|
||||||
@@ -1,3 +1,34 @@
|
|||||||
#content > h1:first-of-type {
|
#content > h1:first-of-type {
|
||||||
display: none !important;
|
display: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.nav-tabs {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-tabs-item > div {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-tabs-underline {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
bottom: 0;
|
||||||
|
height: 1.5px;
|
||||||
|
width: var(--nav-tab-underline-width, 0);
|
||||||
|
transform: translateX(var(--nav-tab-underline-x, 0));
|
||||||
|
background-color: rgb(var(--primary));
|
||||||
|
border-radius: 999px;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
transition: transform 260ms ease-in-out, width 260ms ease-in-out, opacity 160ms ease-in-out;
|
||||||
|
will-change: transform, width;
|
||||||
|
}
|
||||||
|
|
||||||
|
html.dark .nav-tabs-underline {
|
||||||
|
background-color: rgb(var(--primary-light));
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-tabs-underline-ready .nav-tabs-underline {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user