mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
251 lines
8.5 KiB
JavaScript
251 lines
8.5 KiB
JavaScript
|
const getLogoSrc = (isDark) => (isDark ? DARK_LOGO_PATH : LIGHT_LOGO_PATH);
|
||
|
|
||
|
const getModeIconSrc = (isDark) => (isDark ? SUN_ICON_PATH : MOON_ICON_PATH);
|
||
|
|
||
|
const getMenuIconSrc = (isDark) =>
|
||
|
isDark ? DARK_HAMBURGER_PATH : LIGHT_HAMBURGER_PATH;
|
||
|
|
||
|
function addFooterNote() {
|
||
|
const contentInfo = document.querySelector("div[role=contentinfo]");
|
||
|
const footerNote = document.createElement("p");
|
||
|
footerNote.classList.add("footer-note");
|
||
|
footerNote.innerHTML =
|
||
|
'Customized with ❤️ by the <a href="https://ethereum.org/" target="_blank">ethereum.org</a> team.';
|
||
|
contentInfo.parentNode.insertBefore(footerNote, contentInfo.nextSibling);
|
||
|
}
|
||
|
|
||
|
function rearrangeDom() {
|
||
|
const bodyDivs = document.querySelectorAll("body>div");
|
||
|
bodyDivs.forEach((div) => div.remove());
|
||
|
const wrapperDiv = document.createElement("div");
|
||
|
wrapperDiv.classList.add(WRAPPER_CLASS);
|
||
|
bodyDivs.forEach((div) => wrapperDiv.appendChild(div));
|
||
|
document.body.prepend(wrapperDiv);
|
||
|
|
||
|
const rstVersions = document.querySelector(".rst-versions");
|
||
|
rstVersions.remove();
|
||
|
const wyNavSide = document.querySelector("nav.wy-nav-side");
|
||
|
wyNavSide.appendChild(rstVersions);
|
||
|
const backdrop = document.createElement("div");
|
||
|
backdrop.classList.add("backdrop");
|
||
|
wrapperDiv.appendChild(backdrop);
|
||
|
|
||
|
const content = document.querySelector(".wy-nav-content");
|
||
|
content.id = "content";
|
||
|
const oldWrap = document.querySelector("section.wy-nav-content-wrap");
|
||
|
oldWrap.remove();
|
||
|
document.querySelector(".wy-grid-for-nav").appendChild(content);
|
||
|
}
|
||
|
|
||
|
function buildHeader() {
|
||
|
const isDarkMode = localStorage.getItem(LS_COLOR_SCHEME) == DARK;
|
||
|
|
||
|
const header = document.createElement("div");
|
||
|
header.classList.add("unified-header");
|
||
|
document.querySelector(`.${WRAPPER_CLASS}`).prepend(header);
|
||
|
|
||
|
const innerHeader = document.createElement("div");
|
||
|
innerHeader.classList.add("inner-header");
|
||
|
header.appendChild(innerHeader);
|
||
|
|
||
|
const homeLink = document.createElement("a");
|
||
|
homeLink.classList.add("home-link");
|
||
|
homeLink.href = SOLIDITY_HOME_URL;
|
||
|
homeLink.ariaLabel = "Solidity home";
|
||
|
innerHeader.appendChild(homeLink);
|
||
|
|
||
|
const logo = document.createElement("img");
|
||
|
logo.classList.add(SOLIDITY_LOGO_CLASS);
|
||
|
logo.src = getLogoSrc(isDarkMode);
|
||
|
logo.alt = "Solidity logo";
|
||
|
homeLink.appendChild(logo);
|
||
|
|
||
|
const skipToContent = document.createElement("a");
|
||
|
skipToContent.classList.add("skip-to-content");
|
||
|
skipToContent.href = "#content";
|
||
|
skipToContent.innerText = "{ skip to content }";
|
||
|
innerHeader.appendChild(skipToContent);
|
||
|
|
||
|
const navBar = document.createElement("nav");
|
||
|
navBar.classList.add("nav-bar");
|
||
|
innerHeader.appendChild(navBar);
|
||
|
|
||
|
const linkElements = NAV_LINKS.map(({ name, href }) => {
|
||
|
const link = document.createElement("a");
|
||
|
link.classList.add("nav-link");
|
||
|
link.setAttribute("key", name);
|
||
|
link.setAttribute("href", href);
|
||
|
link.setAttribute("aria-label", name);
|
||
|
if (href === FORUM_URL) {
|
||
|
link.classList.add("forum-link");
|
||
|
link.setAttribute("target", "_blank");
|
||
|
link.setAttribute("rel", "noopener noreferrer");
|
||
|
}
|
||
|
link.innerText = name;
|
||
|
return link;
|
||
|
});
|
||
|
linkElements.forEach((link) => navBar.appendChild(link));
|
||
|
|
||
|
// Flex wrapper for color mode and mobile menu buttons
|
||
|
const navButtonContainer = document.createElement("div");
|
||
|
navButtonContainer.classList.add("nav-button-container");
|
||
|
navBar.appendChild(navButtonContainer);
|
||
|
|
||
|
// Build color toggle
|
||
|
const toggleIcon = document.createElement("img");
|
||
|
toggleIcon.classList.add(COLOR_TOGGLE_ICON_CLASS);
|
||
|
toggleIcon.src = getModeIconSrc(isDarkMode);
|
||
|
toggleIcon.alt = "Color mode toggle icon";
|
||
|
toggleIcon.setAttribute("aria-hidden", "true");
|
||
|
toggleIcon.setAttribute("key", "toggle icon");
|
||
|
const colorModeButton = document.createElement("button");
|
||
|
colorModeButton.classList.add("color-toggle");
|
||
|
colorModeButton.setAttribute("type", "button");
|
||
|
colorModeButton.setAttribute("aria-label", "Toggle light dark mode");
|
||
|
colorModeButton.setAttribute("key", "color mode button");
|
||
|
colorModeButton.addEventListener("click", toggleColorMode);
|
||
|
colorModeButton.appendChild(toggleIcon);
|
||
|
navButtonContainer.appendChild(colorModeButton);
|
||
|
|
||
|
// Build mobile hamburger menu
|
||
|
const menuIcon = document.createElement("img");
|
||
|
menuIcon.classList.add(COLOR_TOGGLE_ICON_CLASS);
|
||
|
menuIcon.src = getMenuIconSrc(isDarkMode);
|
||
|
menuIcon.alt = "Toggle menu";
|
||
|
menuIcon.setAttribute("aria-hidden", "true");
|
||
|
menuIcon.setAttribute("key", "menu icon");
|
||
|
const menuButton = document.createElement("button");
|
||
|
menuButton.classList.add("color-toggle");
|
||
|
menuButton.classList.add("mobile-menu-button");
|
||
|
menuButton.setAttribute("type", "button");
|
||
|
menuButton.setAttribute("aria-label", "Toggle menu");
|
||
|
menuButton.setAttribute("key", "menu button");
|
||
|
menuButton.addEventListener("click", toggleMenu);
|
||
|
menuButton.appendChild(menuIcon);
|
||
|
navButtonContainer.appendChild(menuButton);
|
||
|
}
|
||
|
|
||
|
const updateActiveNavLink = () => {
|
||
|
const navLinks = document.querySelectorAll(".unified-header .nav-link");
|
||
|
navLinks.forEach((link) => {
|
||
|
const href = link.getAttribute("href");
|
||
|
if (document.documentURI.includes("contributing.html")) {
|
||
|
link.classList[href.includes("contributing.html") ? "add" : "remove"](
|
||
|
"active"
|
||
|
);
|
||
|
} else {
|
||
|
link.classList[document.documentURI.includes(href) ? "add" : "remove"](
|
||
|
"active"
|
||
|
);
|
||
|
}
|
||
|
});
|
||
|
};
|
||
|
|
||
|
document.addEventListener("locationchange", updateActiveNavLink);
|
||
|
|
||
|
function initialize() {
|
||
|
// Preload fonts
|
||
|
const fonts = [
|
||
|
"overpass-regular.otf",
|
||
|
"overpass-bold.otf",
|
||
|
"overpass-mono-regular.otf",
|
||
|
"overpass-mono-bold.otf",
|
||
|
];
|
||
|
fonts.forEach((filename) => {
|
||
|
const link = document.createElement("link");
|
||
|
link.rel = "preload";
|
||
|
link.as = "font";
|
||
|
link.href = `https://solidity-docs-dev.readthedocs.io/en/latest/_static/fonts/${filename}`;
|
||
|
link.crossOrigin = "";
|
||
|
document.head.appendChild(link);
|
||
|
});
|
||
|
|
||
|
// Rearrange DOM elements for styling
|
||
|
rearrangeDom();
|
||
|
|
||
|
// Check localStorage for existing color scheme preference
|
||
|
var prefersDark = localStorage.getItem(LS_COLOR_SCHEME) == DARK;
|
||
|
// Check link for search param "color"... it may be "light" or "dark"
|
||
|
var urlParams = new URLSearchParams(window.location.search);
|
||
|
if (urlParams.size > 0) {
|
||
|
// This is used for color mode continuity between the main Solidity Lang site and the docs
|
||
|
var colorSchemeParam = urlParams.get("color");
|
||
|
// If present, overwrite prefersDark accordingly
|
||
|
if (colorSchemeParam) {
|
||
|
prefersDark = colorSchemeParam == DARK;
|
||
|
}
|
||
|
|
||
|
// Remove "color" search param from URL
|
||
|
const { location, title } = document;
|
||
|
const { pathname, origin, search, hash } = location;
|
||
|
const newSearchParams = new URLSearchParams(search);
|
||
|
newSearchParams.delete("color");
|
||
|
const sanitizedSearch =
|
||
|
newSearchParams.size < 1 ? "" : "?" + newSearchParams.toString();
|
||
|
window.history.replaceState(
|
||
|
origin,
|
||
|
title,
|
||
|
pathname + sanitizedSearch + hash
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// In case none existed, establish localStorage color scheme preference
|
||
|
var mode = prefersDark ? DARK : LIGHT;
|
||
|
localStorage.setItem(LS_COLOR_SCHEME, mode);
|
||
|
|
||
|
// Select the root element and set the style attribute to denote color-scheme attribute
|
||
|
document
|
||
|
.querySelector(":root")
|
||
|
.setAttribute("style", `--color-scheme: ${mode}`);
|
||
|
|
||
|
// Remove old input and RTD logo anchor element
|
||
|
document.querySelector("input[name=mode]").remove();
|
||
|
document.querySelector("label[for=switch]").remove();
|
||
|
document.querySelector(".wy-side-nav-search > a").remove();
|
||
|
|
||
|
// Add footer note
|
||
|
addFooterNote();
|
||
|
|
||
|
// Build header
|
||
|
buildHeader();
|
||
|
|
||
|
// Close menu
|
||
|
toggleMenu({ force: false });
|
||
|
|
||
|
// Update active nav link
|
||
|
updateActiveNavLink();
|
||
|
}
|
||
|
|
||
|
document.addEventListener("DOMContentLoaded", initialize);
|
||
|
|
||
|
const handleClick = (e) => {
|
||
|
if (e.target.closest(".backdrop")) {
|
||
|
toggleMenu({ force: false });
|
||
|
}
|
||
|
|
||
|
if (e.target.closest("a")) {
|
||
|
const target = e.target.closest("a");
|
||
|
const href = target.getAttribute("href");
|
||
|
if (href.includes(SOLIDITY_HOME_URL)) {
|
||
|
const url = new URL(href);
|
||
|
const params = new URLSearchParams(url.search);
|
||
|
params.set("color", localStorage.getItem(LS_COLOR_SCHEME));
|
||
|
url.search = params.toString();
|
||
|
target.setAttribute("href", url.toString());
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
document.addEventListener("click", handleClick);
|
||
|
|
||
|
const handleKeyDown = (e) => {
|
||
|
if (e.metaKey && e.key === "k") {
|
||
|
document.querySelector("#rtd-search-form input").focus();
|
||
|
} else if (e.key === "Escape") {
|
||
|
toggleMenu({ force: false });
|
||
|
}
|
||
|
if (e.metaKey && e.code === "Backslash") {
|
||
|
toggleColorMode();
|
||
|
}
|
||
|
};
|
||
|
document.addEventListener("keydown", handleKeyDown);
|