Some checks failed
CI / docs-scope (push) Has been cancelled
CI / secrets (push) Has been cancelled
CI / ios (push) Has been cancelled
Docker Release / validate_manual_backfill (push) Has been cancelled
Install Smoke / docs-scope (push) Has been cancelled
Sandbox Common Smoke / sandbox-common-smoke (push) Has been cancelled
Workflow Sanity / no-tabs (push) Has been cancelled
Workflow Sanity / actionlint (push) Has been cancelled
Workflow Sanity / config-docs-drift (push) Has been cancelled
CI / changed-scope (push) Has been cancelled
CI / build-artifacts (push) Has been cancelled
CI / release-check (push) Has been cancelled
CI / checks (pnpm canvas:a2ui:bundle && bunx vitest run --config vitest.unit.config.ts, bun, test) (push) Has been cancelled
CI / checks (pnpm canvas:a2ui:bundle && pnpm test, node, 2, 1, test) (push) Has been cancelled
CI / checks (pnpm canvas:a2ui:bundle && pnpm test, node, 2, 2, test) (push) Has been cancelled
CI / checks (pnpm protocol:check, node, protocol) (push) Has been cancelled
CI / checks (pnpm test:channels, node, channels) (push) Has been cancelled
CI / checks (pnpm test:extensions, node, extensions) (push) Has been cancelled
CI / check (push) Has been cancelled
CI / startup-memory (push) Has been cancelled
CI / check-docs (push) Has been cancelled
CI / compat-node22 (push) Has been cancelled
CI / skills-python (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 1, test) (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 2, test) (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 3, test) (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 4, test) (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 5, test) (push) Has been cancelled
CI / checks-windows (pnpm test, node, 6, 6, test) (push) Has been cancelled
CI / macos (push) Has been cancelled
CI / android (./gradlew --no-daemon :app:assembleDebug, build) (push) Has been cancelled
CI / android (./gradlew --no-daemon :app:testDebugUnitTest, test) (push) Has been cancelled
Docker Release / approve_manual_backfill (push) Has been cancelled
Docker Release / build-amd64 (push) Has been cancelled
Docker Release / build-arm64 (push) Has been cancelled
Docker Release / create-manifest (push) Has been cancelled
Install Smoke / install-smoke (push) Has been cancelled
101 lines
2.8 KiB
JavaScript
101 lines
2.8 KiB
JavaScript
(() => {
|
|
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, () => {});
|
|
})();
|