mirror of
https://github.com/Rapptz/discord.py.git
synced 2025-09-06 17:59:52 +00:00
[matrix] Refactor JS & add searchbar to mobile.
This commit is contained in:
204
docs/_static/custom.js
vendored
204
docs/_static/custom.js
vendored
@ -1,140 +1,69 @@
|
||||
'use-strict';
|
||||
|
||||
let activeModal = null;
|
||||
let activeLink = null;
|
||||
let bottomHeightThreshold, sections;
|
||||
let settingsModal;
|
||||
let hamburgerToggle;
|
||||
let mobileSearch;
|
||||
let sidebar;
|
||||
|
||||
function resizeSidebar() {
|
||||
let rect = sidebar.getBoundingClientRect();
|
||||
sidebar.style.height = `calc(100vh - 1em - ${rect.top + document.body.offsetTop}px)`;
|
||||
}
|
||||
|
||||
function closeModal(modal) {
|
||||
activeModal = null;
|
||||
modal.hidden = true;
|
||||
}
|
||||
|
||||
function openModal(modal) {
|
||||
if (activeModal) {
|
||||
closeModal(activeModal);
|
||||
class Modal {
|
||||
constructor(element) {
|
||||
this.element = element;
|
||||
}
|
||||
|
||||
activeModal = modal;
|
||||
modal.hidden = false;
|
||||
}
|
||||
|
||||
function changeDocumentation(element) {
|
||||
window.location = element.value;
|
||||
}
|
||||
|
||||
function updateSetting(element) {
|
||||
let value;
|
||||
switch (element.type) {
|
||||
case "checkbox":
|
||||
localStorage.setItem(element.name, element.checked);
|
||||
value = element.checked;
|
||||
break;
|
||||
case "radio":
|
||||
localStorage.setItem(element.name, `"${element.value}"`);
|
||||
value = element.value;
|
||||
break;
|
||||
close() {
|
||||
activeModal = null;
|
||||
this.element.hidden = true;
|
||||
}
|
||||
if (element.name in settings) {
|
||||
settings[element.name]["setter"](value);
|
||||
}
|
||||
}
|
||||
|
||||
function LoadSetting(name, defaultValue) {
|
||||
let value = JSON.parse(localStorage.getItem(name));
|
||||
return value === null ? defaultValue : value;
|
||||
}
|
||||
|
||||
function getRootAttributeToggle(attributeName, valueName) {
|
||||
function toggleRootAttribute(set) {
|
||||
if (set) {
|
||||
document.documentElement.setAttribute(`data-${attributeName}`, valueName);
|
||||
} else {
|
||||
document.documentElement.removeAttribute(`data-${attributeName}`);
|
||||
open() {
|
||||
if (activeModal) {
|
||||
activeModal.close();
|
||||
}
|
||||
}
|
||||
return toggleRootAttribute;
|
||||
}
|
||||
|
||||
function setTheme(value) {
|
||||
if (value === "automatic") {
|
||||
if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
|
||||
document.documentElement.setAttribute(`data-theme`, "dark");
|
||||
} else{
|
||||
document.documentElement.setAttribute(`data-theme`, "light");
|
||||
}
|
||||
}
|
||||
else {
|
||||
document.documentElement.setAttribute(`data-theme`, value);
|
||||
activeModal = this;
|
||||
this.element.hidden = false;
|
||||
}
|
||||
}
|
||||
|
||||
const settings = {
|
||||
useSerifFont: {
|
||||
settingType: "checkbox",
|
||||
defaultValue: false,
|
||||
setter: getRootAttributeToggle('font', 'serif')
|
||||
},
|
||||
setTheme: {
|
||||
settingType: "radio",
|
||||
defaultValue: "automatic",
|
||||
setter: setTheme
|
||||
}
|
||||
};
|
||||
class Search {
|
||||
|
||||
Object.entries(settings).forEach(([name, setting]) => {
|
||||
let { defaultValue, setter, ..._ } = setting;
|
||||
let value = LoadSetting(name, defaultValue);
|
||||
try {
|
||||
setter(value);
|
||||
} catch (error) {
|
||||
console.error(`Failed to apply setting "${name}" With value:`, value);
|
||||
console.error(error);
|
||||
constructor() {
|
||||
this.box = document.querySelector('nav.mobile-only');
|
||||
this.bar = document.querySelector('nav.mobile-only input[type="search"]');
|
||||
this.openButton = document.getElementById('open-search');
|
||||
this.closeButton = document.getElementById('close-search');
|
||||
}
|
||||
});
|
||||
|
||||
open() {
|
||||
this.openButton.hidden = true;
|
||||
this.closeButton.hidden = false;
|
||||
this.box.style.top = "100%";
|
||||
this.bar.focus();
|
||||
}
|
||||
|
||||
close() {
|
||||
this.openButton.hidden = false;
|
||||
this.closeButton.hidden = true;
|
||||
this.box.style.top = "0";
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
document.addEventListener('DOMContentLoaded', () => {
|
||||
mobileSearch = new Search();
|
||||
|
||||
bottomHeightThreshold = document.documentElement.scrollHeight - 30;
|
||||
sections = document.querySelectorAll('section');
|
||||
settingsModal = document.querySelector('div#settings.modal');
|
||||
hamburgerToggle = document.getElementById("hamburger-toggle");
|
||||
sidebar = document.getElementById("sidebar");
|
||||
hamburgerToggle = document.getElementById('hamburger-toggle');
|
||||
|
||||
resizeSidebar();
|
||||
|
||||
sidebar.addEventListener("click", (e) => {
|
||||
// If we click a navigation, close the hamburger menu
|
||||
if (e.target.tagName == "A" && sidebar.classList.contains("sidebar-toggle")) {
|
||||
sidebar.classList.remove("sidebar-toggle");
|
||||
let button = hamburgerToggle.firstElementChild;
|
||||
button.textContent = "menu";
|
||||
|
||||
// Scroll a little up to actually see the header
|
||||
// Note: this is generally around ~55px
|
||||
// A proper solution is getComputedStyle but it can be slow
|
||||
// Instead let's just rely on this quirk and call it a day
|
||||
// This has to be done after the browser actually processes
|
||||
// the section movement
|
||||
setTimeout(() => window.scrollBy(0, -100), 75);
|
||||
}
|
||||
})
|
||||
|
||||
hamburgerToggle.addEventListener("click", (e) => {
|
||||
sidebar.classList.toggle("sidebar-toggle");
|
||||
hamburgerToggle.addEventListener('click', (e) => {
|
||||
sidebar.element.classList.toggle('sidebar-toggle');
|
||||
let button = hamburgerToggle.firstElementChild;
|
||||
if (button.textContent == "menu") {
|
||||
button.textContent = "close";
|
||||
if (button.textContent == 'menu') {
|
||||
button.textContent = 'close';
|
||||
}
|
||||
else {
|
||||
button.textContent = "menu";
|
||||
button.textContent = 'menu';
|
||||
}
|
||||
});
|
||||
|
||||
@ -145,59 +74,10 @@ document.addEventListener('DOMContentLoaded', () => {
|
||||
// insert ourselves after the element
|
||||
parent.insertBefore(table, element.nextSibling);
|
||||
});
|
||||
|
||||
Object.entries(settings).forEach(([name, setting]) => {
|
||||
let { settingType, defaultValue, ..._ } = setting;
|
||||
let value = LoadSetting(name, defaultValue);
|
||||
if (settingType === "checkbox") {
|
||||
let element = document.querySelector(`input[name=${name}]`);
|
||||
element.checked = value;
|
||||
} else {
|
||||
let element = document.querySelector(`input[name=${name}][value=${value}]`);
|
||||
element.checked = true;
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
window.addEventListener('scroll', () => {
|
||||
let currentSection = null;
|
||||
|
||||
if (window.scrollY + window.innerHeight > bottomHeightThreshold) {
|
||||
currentSection = sections[sections.length - 1];
|
||||
}
|
||||
else {
|
||||
if (sections) {
|
||||
sections.forEach(section => {
|
||||
let rect = section.getBoundingClientRect();
|
||||
if (rect.top + document.body.offsetTop < 1) {
|
||||
currentSection = section;
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
if (activeLink) {
|
||||
activeLink.parentElement.classList.remove('active');
|
||||
}
|
||||
|
||||
if (currentSection) {
|
||||
activeLink = document.querySelector(`#sidebar a[href="#${currentSection.id}"]`);
|
||||
if (activeLink) {
|
||||
let headingChildren = activeLink.parentElement.parentElement;
|
||||
let heading = headingChildren.previousElementSibling.previousElementSibling;
|
||||
|
||||
if (heading && headingChildren.style.display === "none") {
|
||||
activeLink = heading;
|
||||
}
|
||||
activeLink.parentElement.classList.add('active');
|
||||
}
|
||||
}
|
||||
|
||||
resizeSidebar();
|
||||
});
|
||||
|
||||
document.addEventListener('keydown', (event) => {
|
||||
if (event.keyCode == 27 && activeModal) {
|
||||
closeModal(activeModal);
|
||||
if (event.code == "Escape" && activeModal) {
|
||||
activeModal.close();
|
||||
}
|
||||
});
|
||||
|
Reference in New Issue
Block a user