document.addEventListener('DOMContentLoaded', () => { // Dropdowns const $metalinks = getAll('#meta a'); if ($metalinks.length > 0) { $metalinks.forEach($el => { $el.addEventListener('click', event => { event.preventDefault(); const target = $el.getAttribute('href'); const $target = document.getElementById(target.substring(1)); $target.scrollIntoView(true); // window.history.replaceState(null, document.title, `${window.location.origin}${window.location.pathname}${target}`); return false; }); }); } // Dropdowns const $dropdowns = getAll('.dropdown:not(.is-hoverable)'); if ($dropdowns.length > 0) { $dropdowns.forEach($el => { $el.addEventListener('click', event => { event.stopPropagation(); $el.classList.toggle('is-active'); }); }); document.addEventListener('click', event => { closeDropdowns(); }); } function closeDropdowns() { $dropdowns.forEach($el => { $el.classList.remove('is-active'); }); } // Toggles const $burgers = getAll('.burger'); if ($burgers.length > 0) { $burgers.forEach($el => { $el.addEventListener('click', () => { const target = $el.dataset.target; const $target = document.getElementById(target); $el.classList.toggle('is-active'); $target.classList.toggle('is-active'); }); }); } // Modals const rootEl = document.documentElement; const $modals = getAll('.modal'); const $modalButtons = getAll('.modal-button'); const $modalCloses = getAll('.modal-background, .modal-close, .modal-card-head .delete, .modal-card-foot .button'); if ($modalButtons.length > 0) { $modalButtons.forEach($el => { $el.addEventListener('click', () => { const target = $el.dataset.target; const $target = document.getElementById(target); rootEl.classList.add('is-clipped'); $target.classList.add('is-active'); }); }); } if ($modalCloses.length > 0) { $modalCloses.forEach($el => { $el.addEventListener('click', () => { closeModals(); }); }); } document.addEventListener('keydown', event => { const e = event || window.event; if (e.keyCode === 27) { closeModals(); closeDropdowns(); } }); function closeModals() { rootEl.classList.remove('is-clipped'); $modals.forEach($el => { $el.classList.remove('is-active'); }); } // Clipboard const $highlights = getAll('.highlight'); let itemsProcessed = 0; if ($highlights.length > 0) { $highlights.forEach($el => { const copyEl = ''; const expandEl = ''; $el.insertAdjacentHTML('beforeend', copyEl); const $parent = $el.parentNode; if ($parent && $parent.classList.contains('bd-is-more')) { const showEl = ''; $el.insertAdjacentHTML('beforeend', showEl); } else if ($el.firstElementChild.scrollHeight > 480 && $el.firstElementChild.clientHeight <= 480) { $el.insertAdjacentHTML('beforeend', expandEl); } itemsProcessed++; if (itemsProcessed === $highlights.length) { addHighlightControls(); } }); } function addHighlightControls() { const $highlightButtons = getAll('.highlight .bd-copy, .highlight .bd-expand'); $highlightButtons.forEach($el => { $el.addEventListener('mouseenter', () => { $el.parentNode.classList.add('bd-is-hovering'); }); $el.addEventListener('mouseleave', () => { $el.parentNode.classList.remove('bd-is-hovering'); }); }); const $highlightExpands = getAll('.highlight .bd-expand'); $highlightExpands.forEach($el => { $el.addEventListener('click', () => { $el.parentNode.firstElementChild.style.maxHeight = 'none'; }); }); const $highlightShows = getAll('.highlight .bd-show'); $highlightShows.forEach($el => { $el.addEventListener('click', () => { $el.parentNode.parentNode.classList.remove('bd-is-more-clipped'); }); }); } new Clipboard('.bd-copy', { target: function(trigger) { return trigger.previousSibling; } }); // Functions function getAll(selector) { return Array.prototype.slice.call(document.querySelectorAll(selector), 0); } // Scrolling const navbarEl = document.getElementById('navbar'); const navbarBurger = document.getElementById('navbarBurger'); const specialShadow = document.getElementById('specialShadow'); const NAVBAR_HEIGHT = 52; const THRESHOLD = 160; let navbarOpen = false; let horizon = NAVBAR_HEIGHT; let whereYouStoppedScrolling = 0; let scrollFactor = 0; let currentTranslate = 0; navbarBurger.addEventListener('click', el => { navbarOpen = !navbarOpen; if (navbarOpen) { rootEl.classList.add('bd-is-clipped-touch'); } else { rootEl.classList.remove('bd-is-clipped-touch'); } }); function upOrDown(lastY, currentY) { if (currentY >= lastY) { return goingDown(currentY); } return goingUp(currentY); } function goingDown(currentY) { const trigger = NAVBAR_HEIGHT; whereYouStoppedScrolling = currentY; if (currentY > horizon) { horizon = currentY; } translateHeader(currentY, false); } function goingUp(currentY) { const trigger = 0; if (currentY < (whereYouStoppedScrolling - NAVBAR_HEIGHT)) { horizon = currentY + NAVBAR_HEIGHT; } translateHeader(currentY, true); } function constrainDelta(delta) { return Math.max(0, Math.min(delta, NAVBAR_HEIGHT)); } function translateHeader(currentY, upwards) { // let topTranslateValue; let translateValue; if (upwards && currentTranslate == 0) { translateValue = 0; } else if (currentY <= NAVBAR_HEIGHT) { translateValue = currentY * -1; } else { const delta = constrainDelta(Math.abs(currentY - horizon)); translateValue = delta - NAVBAR_HEIGHT; } if (translateValue != currentTranslate) { const navbarStyle = ` transform: translateY(${translateValue}px); `; currentTranslate = translateValue; navbarEl.setAttribute('style', navbarStyle); } if (currentY > THRESHOLD * 2) { scrollFactor = 1; } else if (currentY > THRESHOLD) { scrollFactor = (currentY - THRESHOLD) / THRESHOLD; } else { scrollFactor = 0; } const translateFactor = 1 + translateValue / NAVBAR_HEIGHT; specialShadow.style.opacity = scrollFactor; specialShadow.style.transform = 'scaleY(' + translateFactor + ')'; } translateHeader(window.scrollY, false); let ticking = false; let lastY = 0; window.addEventListener('scroll', function() { const currentY = window.scrollY; if (!ticking) { window.requestAnimationFrame(function() { upOrDown(lastY, currentY); ticking = false; lastY = currentY; }); } ticking = true; }); });