(() => { const btfFn = { debounce: (func, wait = 0, immediate = false) => { let timeout return (...args) => { const later = () => { timeout = null if (!immediate) func(...args) } const callNow = immediate && !timeout clearTimeout(timeout) timeout = setTimeout(later, wait) if (callNow) func(...args) } }, throttle: function (func, wait, options = {}) { let timeout, context, args let previous = 0 const later = () => { previous = options.leading === false ? 0 : new Date().getTime() timeout = null func.apply(context, args) if (!timeout) context = args = null } const throttled = (...params) => { const now = new Date().getTime() if (!previous && options.leading === false) previous = now const remaining = wait - (now - previous) context = this args = params if (remaining <= 0 || remaining > wait) { if (timeout) { clearTimeout(timeout) timeout = null } previous = now func.apply(context, args) if (!timeout) context = args = null } else if (!timeout && options.trailing !== false) { timeout = setTimeout(later, remaining) } } return throttled }, overflowPaddingR: { add: () => { const paddingRight = window.innerWidth - document.body.clientWidth if (paddingRight > 0) { document.body.style.paddingRight = `${paddingRight}px` document.body.style.overflow = 'hidden' const menuElement = document.querySelector('#page-header.nav-fixed #menus') if (menuElement) { menuElement.style.paddingRight = `${paddingRight}px` } } }, remove: () => { document.body.style.paddingRight = '' document.body.style.overflow = '' const menuElement = document.querySelector('#page-header.nav-fixed #menus') if (menuElement) { menuElement.style.paddingRight = '' } } }, snackbarShow: (text, showAction = false, duration = 2000) => { const { position, bgLight, bgDark } = GLOBAL_CONFIG.Snackbar const bg = document.documentElement.getAttribute('data-theme') === 'light' ? bgLight : bgDark Snackbar.show({ text, backgroundColor: bg, showAction, duration, pos: position, customClass: 'snackbar-css' }) }, diffDate: (inputDate, more = false) => { const dateNow = new Date() const datePost = new Date(inputDate) const diffMs = dateNow - datePost const diffSec = diffMs / 1000 const diffMin = diffSec / 60 const diffHour = diffMin / 60 const diffDay = diffHour / 24 const diffMonth = diffDay / 30 const { dateSuffix } = GLOBAL_CONFIG if (!more) return Math.floor(diffDay) if (diffMonth > 12) return datePost.toISOString().slice(0, 10) if (diffMonth >= 1) return `${Math.floor(diffMonth)} ${dateSuffix.month}` if (diffDay >= 1) return `${Math.floor(diffDay)} ${dateSuffix.day}` if (diffHour >= 1) return `${Math.floor(diffHour)} ${dateSuffix.hour}` if (diffMin >= 1) return `${Math.floor(diffMin)} ${dateSuffix.min}` return dateSuffix.just }, loadComment: (dom, callback) => { if ('IntersectionObserver' in window) { const observerItem = new IntersectionObserver((entries) => { if (entries[0].isIntersecting) { callback() observerItem.disconnect() } }, { threshold: [0] }) observerItem.observe(dom) } else { callback() } }, scrollToDest: (pos, time = 500) => { const currentPos = window.scrollY const isNavFixed = document.getElementById('page-header').classList.contains('fixed') if (currentPos > pos || isNavFixed) pos = pos - 70 if ('scrollBehavior' in document.documentElement.style) { window.scrollTo({ top: pos, behavior: 'smooth' }) return } const startTime = performance.now() const animate = currentTime => { const timeElapsed = currentTime - startTime const progress = Math.min(timeElapsed / time, 1) window.scrollTo(0, currentPos + (pos - currentPos) * progress) if (progress < 1) { requestAnimationFrame(animate) } } requestAnimationFrame(animate) }, animateIn: (ele, animation) => { ele.style.display = 'block' ele.style.animation = animation }, animateOut: (ele, animation) => { const handleAnimationEnd = () => { ele.style.display = '' ele.style.animation = '' ele.removeEventListener('animationend', handleAnimationEnd) } ele.addEventListener('animationend', handleAnimationEnd) ele.style.animation = animation }, wrap: (selector, eleType, options) => { const createEle = document.createElement(eleType) for (const [key, value] of Object.entries(options)) { createEle.setAttribute(key, value) } selector.parentNode.insertBefore(createEle, selector) createEle.appendChild(selector) }, isHidden: ele => ele.offsetHeight === 0 && ele.offsetWidth === 0, getEleTop: ele => { let actualTop = ele.offsetTop let current = ele.offsetParent while (current !== null) { actualTop += current.offsetTop current = current.offsetParent } return actualTop }, loadLightbox: ele => { const service = GLOBAL_CONFIG.lightbox if (service === 'medium_zoom') { mediumZoom(ele, { background: 'var(--zoom-bg)' }) } if (service === 'fancybox') { Array.from(ele).forEach(i => { if (i.parentNode.tagName !== 'A') { const dataSrc = i.dataset.lazySrc || i.src const dataCaption = i.title || i.alt || '' btf.wrap(i, 'a', { href: dataSrc, 'data-fancybox': 'gallery', 'data-caption': dataCaption, 'data-thumb': dataSrc }) } }) if (!window.fancyboxRun) { Fancybox.bind('[data-fancybox]', { Hash: false, Thumbs: { showOnStart: false }, Images: { Panzoom: { maxScale: 4 } }, Carousel: { transition: 'slide' }, Toolbar: { display: { left: ['infobar'], middle: [ 'zoomIn', 'zoomOut', 'toggle1to1', 'rotateCCW', 'rotateCW', 'flipX', 'flipY' ], right: ['slideshow', 'thumbs', 'close'] } } }) window.fancyboxRun = true } } }, setLoading: { add: ele => { const html = `
` ele.insertAdjacentHTML('afterend', html) }, remove: ele => { ele.nextElementSibling.remove() } }, updateAnchor: anchor => { if (anchor !== window.location.hash) { if (!anchor) anchor = location.pathname const title = GLOBAL_CONFIG_SITE.title window.history.replaceState({ url: location.href, title }, title, anchor) } }, getScrollPercent: (() => { let docHeight, winHeight, headerHeight, contentMath return (currentTop, ele) => { if (!docHeight || ele.clientHeight !== docHeight) { docHeight = ele.clientHeight winHeight = window.innerHeight headerHeight = ele.offsetTop contentMath = Math.max(docHeight - winHeight, document.documentElement.scrollHeight - winHeight) } const scrollPercent = (currentTop - headerHeight) / contentMath return Math.max(0, Math.min(100, Math.round(scrollPercent * 100))) } })(), addEventListenerPjax: (ele, event, fn, option = false) => { ele.addEventListener(event, fn, option) btf.addGlobalFn('pjaxSendOnce', () => { ele.removeEventListener(event, fn, option) }) }, removeGlobalFnEvent: (key, parent = window) => { const globalFn = parent.globalFn || {} const keyObj = globalFn[key] if (!keyObj) return Object.keys(keyObj).forEach(i => keyObj[i]()) delete globalFn[key] }, switchComments: (el = document, path) => { const switchBtn = el.querySelector('#switch-btn') if (!switchBtn) return let switchDone = false const postComment = el.querySelector('#post-comment') const handleSwitchBtn = () => { postComment.classList.toggle('move') if (!switchDone && typeof loadOtherComment === 'function') { switchDone = true loadOtherComment(el, path) } } btf.addEventListenerPjax(switchBtn, 'click', handleSwitchBtn) } } window.btf = { ...window.btf, ...btfFn } })()