diff --git a/assets/js/scripts.js b/assets/js/scripts.js index 45e6ec76..d92b3afc 100644 --- a/assets/js/scripts.js +++ b/assets/js/scripts.js @@ -1,459 +1,464 @@ -const $ = (selector) => document.querySelector(selector) -const $$ = (selector) => document.querySelectorAll(selector) - -const loadedScripts = [] - -function loadScript(src) { - if (loadedScripts.includes(src)) return Promise.resolve() - - return new Promise(function (resolve, reject) { - const s = document.createElement('script') - let r = false - s.type = 'text/javascript' - s.src = src - s.async = true - s.onerror = function (err) { - reject(err, s) - } - s.onload = s.onreadystatechange = function () { - // console.log(this.readyState); // uncomment this line to see which ready states are called. - if (!r && (!this.readyState || this.readyState === 'complete')) { - r = true - loadedScripts.push(src) - resolve() +function fnInit() { + const $ = (selector) => document.querySelector(selector) + const $$ = (selector) => document.querySelectorAll(selector) + + const loadedScripts = [] + + function loadScript(src) { + if (loadedScripts.includes(src)) return Promise.resolve() + + return new Promise(function (resolve, reject) { + const s = document.createElement('script') + let r = false + s.type = 'text/javascript' + s.src = src + s.async = true + s.onerror = function (err) { + reject(err, s) } - } - const t = document.getElementsByTagName('script')[0] - t.parentElement.insertBefore(s, t) - }) -} + s.onload = s.onreadystatechange = function () { + // console.log(this.readyState); // uncomment this line to see which ready states are called. + if (!r && (!this.readyState || this.readyState === 'complete')) { + r = true + loadedScripts.push(src) + resolve() + } + } + const t = document.getElementsByTagName('script')[0] + t.parentElement.insertBefore(s, t) + }) + } -// youtube functionality -function createYoutubeFrame(id) { - const html = - "" - const fragment = document.createRange().createContextualFragment(html) - document.body.appendChild(fragment) - - $('#lightbox a').addEventListener( - 'click', - function (e) { - e.preventDefault() - const lightbox = $('#lightbox') - lightbox.parentNode.removeChild(lightbox) - }, - { once: true } - ) -} + // youtube functionality + function createYoutubeFrame(id) { + const html = + "" + const fragment = document.createRange().createContextualFragment(html) + document.body.appendChild(fragment) + + $('#lightbox a').addEventListener( + 'click', + function (e) { + e.preventDefault() + const lightbox = $('#lightbox') + lightbox.parentNode.removeChild(lightbox) + }, + { once: true } + ) + } -function initScheme() { - const isBrowserSchemeDark = window.matchMedia( - '(prefers-color-scheme: dark)' - ).matches + function initScheme() { + const isBrowserSchemeDark = window.matchMedia( + '(prefers-color-scheme: dark)' + ).matches - let scheme = isBrowserSchemeDark ? 'dark-mode' : 'light-mode' + let scheme = isBrowserSchemeDark ? 'dark-mode' : 'light-mode' - if (localStorage.getItem('scheme')) { - scheme = localStorage.getItem('scheme') + if (localStorage.getItem('scheme')) { + scheme = localStorage.getItem('scheme') + } + document.documentElement.setAttribute('scheme', scheme) } - document.documentElement.setAttribute('scheme', scheme) -} -initScheme() + initScheme() -$('#emoticon-mode').addEventListener('click', function (e) { - const scheme = document.documentElement.getAttribute('scheme') - const isLightMode = scheme === 'light-mode' + $('#emoticon-mode').addEventListener('click', function (e) { + const scheme = document.documentElement.getAttribute('scheme') + const isLightMode = scheme === 'light-mode' - document.documentElement.setAttribute( - 'scheme', - isLightMode ? 'dark-mode' : 'light-mode' - ) + document.documentElement.setAttribute( + 'scheme', + isLightMode ? 'dark-mode' : 'light-mode' + ) - localStorage.setItem('scheme', isLightMode ? 'dark-mode' : 'light-mode') -}) - -$$('.youtube-link').forEach(function (link) { - link.addEventListener('click', function (e) { - e.preventDefault() - const id = this.getAttribute('data-id') - createYoutubeFrame(id) + localStorage.setItem('scheme', isLightMode ? 'dark-mode' : 'light-mode') }) -}) -class LiteYTEmbed extends window.HTMLElement { - async connectedCallback() { - this.videoId = this.getAttribute('videoid') + $$('.youtube-link').forEach(function (link) { + link.addEventListener('click', function (e) { + e.preventDefault() + const id = this.getAttribute('data-id') + createYoutubeFrame(id) + }) + }) - let playBtnEl = this.querySelector('.lty-playbtn') - // A label for the button takes priority over a [playlabel] attribute on the custom-element - this.playLabel = - (playBtnEl && playBtnEl.textContent.trim()) || - this.getAttribute('playlabel') || - 'Play' + class LiteYTEmbed extends window.HTMLElement { + async connectedCallback() { + this.videoId = this.getAttribute('videoid') - const isWebpSupported = await LiteYTEmbed.checkWebPSupport() + let playBtnEl = this.querySelector('.lty-playbtn') + // A label for the button takes priority over a [playlabel] attribute on the custom-element + this.playLabel = + (playBtnEl && playBtnEl.textContent.trim()) || + this.getAttribute('playlabel') || + 'Play' - this.posterUrl = isWebpSupported - ? `https://i.ytimg.com/vi_webp/${this.videoId}/hqdefault.webp` - : `https://i.ytimg.com/vi/${this.videoId}/hqdefault.jpg` + const isWebpSupported = await LiteYTEmbed.checkWebPSupport() - // Warm the connection for the poster image - LiteYTEmbed.addPrefetch('preload', this.posterUrl, 'image') + this.posterUrl = isWebpSupported + ? `https://i.ytimg.com/vi_webp/${this.videoId}/hqdefault.webp` + : `https://i.ytimg.com/vi/${this.videoId}/hqdefault.jpg` - this.style.backgroundImage = `url("${this.posterUrl}")` + // Warm the connection for the poster image + LiteYTEmbed.addPrefetch('preload', this.posterUrl, 'image') - // Set up play button, and its visually hidden label - if (!playBtnEl) { - playBtnEl = document.createElement('button') - playBtnEl.type = 'button' - playBtnEl.classList.add('lty-playbtn') - this.append(playBtnEl) - } - if (!playBtnEl.textContent) { - const playBtnLabelEl = document.createElement('span') - playBtnLabelEl.className = 'lyt-visually-hidden' - playBtnLabelEl.textContent = this.playLabel - playBtnEl.append(playBtnLabelEl) - } + this.style.backgroundImage = `url("${this.posterUrl}")` - // On hover (or tap), warm up the TCP connections we're (likely) about to use. - this.addEventListener('pointerover', LiteYTEmbed.warmConnections, { - once: true - }) + // Set up play button, and its visually hidden label + if (!playBtnEl) { + playBtnEl = document.createElement('button') + playBtnEl.type = 'button' + playBtnEl.classList.add('lty-playbtn') + this.append(playBtnEl) + } + if (!playBtnEl.textContent) { + const playBtnLabelEl = document.createElement('span') + playBtnLabelEl.className = 'lyt-visually-hidden' + playBtnLabelEl.textContent = this.playLabel + playBtnEl.append(playBtnLabelEl) + } - // Once the user clicks, add the real iframe and drop our play button - // TODO: In the future we could be like amp-youtube and silently swap in the iframe during idle time - // We'd want to only do this for in-viewport or near-viewport ones: https://github.com/ampproject/amphtml/pull/5003 - this.addEventListener('click', (e) => this.addIframe()) - } + // On hover (or tap), warm up the TCP connections we're (likely) about to use. + this.addEventListener('pointerover', LiteYTEmbed.warmConnections, { + once: true + }) - static addPrefetch(kind, url, as) { - const linkEl = document.createElement('link') - linkEl.rel = kind - linkEl.href = url - if (as) { - linkEl.as = as + // Once the user clicks, add the real iframe and drop our play button + // TODO: In the future we could be like amp-youtube and silently swap in the iframe during idle time + // We'd want to only do this for in-viewport or near-viewport ones: https://github.com/ampproject/amphtml/pull/5003 + this.addEventListener('click', (e) => this.addIframe()) } - document.head.append(linkEl) - } - /** - * Check WebP support for the user - */ - static checkWebPSupport() { - if (typeof LiteYTEmbed.hasWebPSupport !== 'undefined') { - return Promise.resolve(LiteYTEmbed.hasWebPSupport) + static addPrefetch(kind, url, as) { + const linkEl = document.createElement('link') + linkEl.rel = kind + linkEl.href = url + if (as) { + linkEl.as = as + } + document.head.append(linkEl) } - return new Promise((resolve) => { - const resolveAndSaveValue = (value) => { - LiteYTEmbed.hasWebPSupport = value - resolve(value) + /** + * Check WebP support for the user + */ + static checkWebPSupport() { + if (typeof LiteYTEmbed.hasWebPSupport !== 'undefined') { + return Promise.resolve(LiteYTEmbed.hasWebPSupport) } - const img = new window.Image() - img.onload = () => resolveAndSaveValue(true) - img.onerror = () => resolveAndSaveValue(false) - img.src = - '' - }) - } + return new Promise((resolve) => { + const resolveAndSaveValue = (value) => { + LiteYTEmbed.hasWebPSupport = value + resolve(value) + } + + const img = new window.Image() + img.onload = () => resolveAndSaveValue(true) + img.onerror = () => resolveAndSaveValue(false) + img.src = + '' + }) + } - /** - * Begin pre-connecting to warm up the iframe load - * Since the embed's network requests load within its iframe, - * preload/prefetch'ing them outside the iframe will only cause double-downloads. - * So, the best we can do is warm up a few connections to origins that are in the critical path. - * - * Maybe `` would work, but it's unsupported: http://crbug.com/593267 - * But TBH, I don't think it'll happen soon with Site Isolation and split caches adding serious complexity. - */ - static warmConnections() { - if (LiteYTEmbed.preconnected) return - - // The iframe document and most of its subresources come right off youtube.com - LiteYTEmbed.addPrefetch('preconnect', 'https://www.youtube-nocookie.com') - // The botguard script is fetched off from google.com - LiteYTEmbed.addPrefetch('preconnect', 'https://www.google.com') - - // Not certain if these ad related domains are in the critical path. Could verify with domain-specific throttling. - LiteYTEmbed.addPrefetch('preconnect', 'https://googleads.g.doubleclick.net') - LiteYTEmbed.addPrefetch('preconnect', 'https://static.doubleclick.net') - - LiteYTEmbed.preconnected = true + /** + * Begin pre-connecting to warm up the iframe load + * Since the embed's network requests load within its iframe, + * preload/prefetch'ing them outside the iframe will only cause double-downloads. + * So, the best we can do is warm up a few connections to origins that are in the critical path. + * + * Maybe `` would work, but it's unsupported: http://crbug.com/593267 + * But TBH, I don't think it'll happen soon with Site Isolation and split caches adding serious complexity. + */ + static warmConnections() { + if (LiteYTEmbed.preconnected) return + + // The iframe document and most of its subresources come right off youtube.com + LiteYTEmbed.addPrefetch('preconnect', 'https://www.youtube-nocookie.com') + // The botguard script is fetched off from google.com + LiteYTEmbed.addPrefetch('preconnect', 'https://www.google.com') + + // Not certain if these ad related domains are in the critical path. Could verify with domain-specific throttling. + LiteYTEmbed.addPrefetch('preconnect', 'https://googleads.g.doubleclick.net') + LiteYTEmbed.addPrefetch('preconnect', 'https://static.doubleclick.net') + + LiteYTEmbed.preconnected = true + } + + addIframe() { + const params = new URLSearchParams(this.getAttribute('params') || []) + params.append('autoplay', '1') + + const iframeEl = document.createElement('iframe') + iframeEl.width = 560 + iframeEl.height = 315 + // No encoding necessary as [title] is safe. https://cheatsheetseries.owasp.org/cheatsheets/Cross_Site_Scripting_Prevention_Cheat_Sheet.html#:~:text=Safe%20HTML%20Attributes%20include + iframeEl.title = this.playLabel + iframeEl.allow = + 'accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture' + iframeEl.allowFullscreen = true + // AFAIK, the encoding here isn't necessary for XSS, but we'll do it only because this is a URL + // https://stackoverflow.com/q/64959723/89484 + iframeEl.src = `https://www.youtube-nocookie.com/embed/${encodeURIComponent( + this.videoId + )}?${params.toString()}` + this.append(iframeEl) + + this.classList.add('lyt-activated') + + // Set focus for a11y + this.querySelector('iframe').focus() + } } + // Register custome element + window.customElements.define('lite-youtube', LiteYTEmbed) - addIframe() { - const params = new URLSearchParams(this.getAttribute('params') || []) - params.append('autoplay', '1') - - const iframeEl = document.createElement('iframe') - iframeEl.width = 560 - iframeEl.height = 315 - // No encoding necessary as [title] is safe. https://cheatsheetseries.owasp.org/cheatsheets/Cross_Site_Scripting_Prevention_Cheat_Sheet.html#:~:text=Safe%20HTML%20Attributes%20include - iframeEl.title = this.playLabel - iframeEl.allow = - 'accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture' - iframeEl.allowFullscreen = true - // AFAIK, the encoding here isn't necessary for XSS, but we'll do it only because this is a URL - // https://stackoverflow.com/q/64959723/89484 - iframeEl.src = `https://www.youtube-nocookie.com/embed/${encodeURIComponent( - this.videoId - )}?${params.toString()}` - this.append(iframeEl) - - this.classList.add('lyt-activated') - - // Set focus for a11y - this.querySelector('iframe').focus() + // Show share only when needed + const intersectionObserverOptions = { + rootMargin: '0px', + threshold: 1.0 } -} -// Register custome element -window.customElements.define('lite-youtube', LiteYTEmbed) -// Show share only when needed -const intersectionObserverOptions = { - rootMargin: '0px', - threshold: 1.0 -} + const $share = document.getElementById('share') -const $share = document.getElementById('share') + if ($share) { + const $articlePagination = document.getElementById('article-pagination') + const $footer = $('footer') + const elementToObserve = $articlePagination || $footer -if ($share) { - const $articlePagination = document.getElementById('article-pagination') - const $footer = $('footer') - const elementToObserve = $articlePagination || $footer + const onIntersect = function (entries) { + const [entry] = entries + const hide = entry.boundingClientRect.top <= 0 || entry.isIntersecting + $share.classList.toggle('u-none', hide) + } - const onIntersect = function (entries) { - const [entry] = entries - const hide = entry.boundingClientRect.top <= 0 || entry.isIntersecting - $share.classList.toggle('u-none', hide) - } + const observer = new window.IntersectionObserver( + onIntersect, + intersectionObserverOptions + ) - const observer = new window.IntersectionObserver( - onIntersect, - intersectionObserverOptions - ) + observer.observe(elementToObserve) + } - observer.observe(elementToObserve) -} + const ALGOLIA_APPLICATION_ID = 'QK9VV9YO5F' + const ALGOLIA_SEARCH_ONLY_API_KEY = '247bb355c786b6e9f528bc382cab3039' + let algoliaIndex + + const $form = $('.ais-SearchBox-form') + const $input = $('.ais-SearchBox-input') + const $reset = $('.ais-SearchBox-reset') + const $hits = $('#hits') + + function getAlgoliaIndex() { + if (algoliaIndex) return algoliaIndex + console.log( + '🚀 ~ file: scripts.js ~ line 222 ~ getAlgoliaIndex ~ algoliaIndex', + algoliaIndex + ) + + const algoliaClient = window.algoliasearch( + ALGOLIA_APPLICATION_ID, + ALGOLIA_SEARCH_ONLY_API_KEY, + { + _useRequestCache: true + } + ) + algoliaIndex = algoliaClient.initIndex('prod_blog_content') + return algoliaIndex + } -const ALGOLIA_APPLICATION_ID = 'QK9VV9YO5F' -const ALGOLIA_SEARCH_ONLY_API_KEY = '247bb355c786b6e9f528bc382cab3039' -let algoliaIndex + $form.addEventListener('submit', function (e) { + e.preventDefault() + }) -const $form = $('.ais-SearchBox-form') -const $input = $('.ais-SearchBox-input') -const $reset = $('.ais-SearchBox-reset') -const $hits = $('#hits') + $reset.addEventListener('click', function (e) { + $input.value = '' + $hits.innerHTML = '' + }) -function getAlgoliaIndex() { - if (algoliaIndex) return algoliaIndex - console.log( - '🚀 ~ file: scripts.js ~ line 222 ~ getAlgoliaIndex ~ algoliaIndex', - algoliaIndex - ) + $input.addEventListener('input', async function (e) { + await loadScript( + 'https://cdn.jsdelivr.net/npm/algoliasearch@4.5.1/dist/algoliasearch-lite.umd.js' + ) - const algoliaClient = window.algoliasearch( - ALGOLIA_APPLICATION_ID, - ALGOLIA_SEARCH_ONLY_API_KEY, - { - _useRequestCache: true + const { value } = e.target + if (value === '') { + $hits.innerHTML = '' + $reset.setAttribute('hidden') } - ) - algoliaIndex = algoliaClient.initIndex('prod_blog_content') - return algoliaIndex -} -$form.addEventListener('submit', function (e) { - e.preventDefault() -}) + $reset.removeAttribute('hidden') + $hits.removeAttribute('hidden') -$reset.addEventListener('click', function (e) { - $input.value = '' - $hits.innerHTML = '' -}) - -$input.addEventListener('input', async function (e) { - await loadScript( - 'https://cdn.jsdelivr.net/npm/algoliasearch@4.5.1/dist/algoliasearch-lite.umd.js' - ) - - const { value } = e.target - if (value === '') { - $hits.innerHTML = '' - $reset.setAttribute('hidden') - } - - $reset.removeAttribute('hidden') - $hits.removeAttribute('hidden') + const algoliaIndex = getAlgoliaIndex() + algoliaIndex + .search(value, { + hitsPerPage: 3 + }) + .then(({ hits }) => { + let hitsHtml = '' + hits.forEach((hit) => { + const { + link, + _highlightResult: { + title: { value: title }, + description: { value: description } + } + } = hit + + hitsHtml += ` +
  • + + ${title} +
    + ${description} +
    +
    +
  • ` + }) - const algoliaIndex = getAlgoliaIndex() - algoliaIndex - .search(value, { - hitsPerPage: 3 - }) - .then(({ hits }) => { - let hitsHtml = '' - hits.forEach((hit) => { - const { - link, - _highlightResult: { - title: { value: title }, - description: { value: description } - } - } = hit - - hitsHtml += ` -
  • - - ${title} -
    - ${description} -
    -
    -
  • ` + $hits.innerHTML = hitsHtml }) + }) - $hits.innerHTML = hitsHtml - }) -}) - -// Table Of Contents script -function initTableOfContents() { - const firstTableOfContentsElement = $('#TableOfContents-container li') - if (!firstTableOfContentsElement) return null - - // activate first element of table of contents - firstTableOfContentsElement.classList.add('active') - // get all links from table of contents - const links = $$('#TableOfContents-container li a') - - const changeBgLinks = (entries) => { - entries.forEach((entry) => { - const { target, isIntersecting, intersectionRatio } = entry - if (isIntersecting && intersectionRatio >= 0.5) { - const id = target.getAttribute('id') - $('#TableOfContents-container li.active').classList.remove('active') - $(`nav li a[href="#${id}"]`).parentElement.classList.add('active') - - links.forEach((link) => { - link.addEventListener('click', (e) => { - $('#TableOfContents-container li.active').classList.remove('active') - link.parentElement.classList.add('active') + // Table Of Contents script + function initTableOfContents() { + const firstTableOfContentsElement = $('#TableOfContents-container li') + if (!firstTableOfContentsElement) return null + + // activate first element of table of contents + firstTableOfContentsElement.classList.add('active') + // get all links from table of contents + const links = $$('#TableOfContents-container li a') + + const changeBgLinks = (entries) => { + entries.forEach((entry) => { + const { target, isIntersecting, intersectionRatio } = entry + if (isIntersecting && intersectionRatio >= 0.5) { + const id = target.getAttribute('id') + $('#TableOfContents-container li.active').classList.remove('active') + $(`nav li a[href="#${id}"]`).parentElement.classList.add('active') + + links.forEach((link) => { + link.addEventListener('click', (e) => { + $('#TableOfContents-container li.active').classList.remove('active') + link.parentElement.classList.add('active') + }) }) - }) - } - }) - } + } + }) + } - const options = { - threshold: 0.5, - rootMargin: '50px 0px -55% 0px' - } + const options = { + threshold: 0.5, + rootMargin: '50px 0px -55% 0px' + } - const observer = new window.IntersectionObserver(changeBgLinks, options) + const observer = new window.IntersectionObserver(changeBgLinks, options) - const articleTitles = $$('#article-content h2') - articleTitles.forEach((section) => observer.observe(section)) -} + const articleTitles = $$('#article-content h2') + articleTitles.forEach((section) => observer.observe(section)) + } -initTableOfContents() + initTableOfContents() -fetch('https://midudev-apis.midudev.workers.dev/uptime/midudev') - .then((res) => res.json()) - .then(({ online }) => { - if (online) { - $('#live').style.display = 'block' - $( - '#live-frame' - ).innerHTML = `` - } - }) - .catch((err) => { - console.error(err) + fetch('https://midudev-apis.midudev.workers.dev/uptime/midudev') + .then((res) => res.json()) + .then(({ online }) => { + if (online) { + $('#live').style.display = 'block' + $( + '#live-frame' + ).innerHTML = `` + } + }) + .catch((err) => { + console.error(err) + }) + + $('#live-close').addEventListener('click', () => { + $('#live').style.display = 'none' + $('#live').innerHTML = '' }) -$('#live-close').addEventListener('click', () => { - $('#live').style.display = 'none' - $('#live').innerHTML = '' -}) - -async function getPageContent(url) { - // This is a really scrappy way to do this. - // Don't do this in production! - const response = await fetch(url) - const text = await response.text() - // Particularly as it uses regexp - return /]*>([\w\W]*)<\/body>/.exec(text)[1] -} + async function getPageContent(url) { + // This is a really scrappy way to do this. + // Don't do this in production! + const response = await fetch(url) + const text = await response.text() + // Particularly as it uses regexp + return /]*>([\w\W]*)<\/body>/.exec(text)[1] + } -function isBackNavigation(navigateEvent) { - if (navigateEvent.navigationType === 'push' || navigateEvent.navigationType === 'replace') { + function isBackNavigation(navigateEvent) { + if (navigateEvent.navigationType === 'push' || navigateEvent.navigationType === 'replace') { + return false + } + if ( + navigateEvent.destination.index !== -1 && + navigateEvent.destination.index < window.navigation.currentEntry.index + ) { + return true + } return false } - if ( - navigateEvent.destination.index !== -1 && - navigateEvent.destination.index < window.navigation.currentEntry.index - ) { - return true - } - return false -} -async function onLinkNavigate(callback) { - window.navigation.addEventListener('navigate', (event) => { - const toUrl = new URL(event.destination.url) + async function onLinkNavigate(callback) { + window.navigation.addEventListener('navigate', (event) => { + const toUrl = new URL(event.destination.url) - if (window.location.origin !== toUrl.origin) return + if (window.location.origin !== toUrl.origin) return - const fromPath = window.location.pathname - const isBack = isBackNavigation(event) + const fromPath = window.location.pathname + const isBack = isBackNavigation(event) - event.intercept({ - async handler() { - if (event.info === 'ignore') return + event.intercept({ + async handler() { + if (event.info === 'ignore') return - await callback({ - toPath: toUrl.pathname, - fromPath, - isBack - }) - } + await callback({ + toPath: toUrl.pathname, + fromPath, + isBack + }) + } + }) }) - }) -} + } -onLinkNavigate(async ({ toPath }) => { - const content = await getPageContent(toPath) + onLinkNavigate(async ({ toPath }) => { + const content = await getPageContent(toPath) - startViewTransition(() => { - // This is a pretty heavy-handed way to update page content. - // In production, you'd likely be modifying DOM elements directly, - // or using a framework. - // innerHTML is used here just to keep the DOM update super simple. - document.body.innerHTML = content + startViewTransition(() => { + // This is a pretty heavy-handed way to update page content. + // In production, you'd likely be modifying DOM elements directly, + // or using a framework. + // innerHTML is used here just to keep the DOM update super simple. + document.body.innerHTML = content + fnInit() + }) }) -}) -// A little helper function like this is really handy -// to handle progressive enhancement. -function startViewTransition(callback) { - if (!document.startViewTransition) { - callback() - return + // A little helper function like this is really handy + // to handle progressive enhancement. + function startViewTransition(callback) { + if (!document.startViewTransition) { + callback() + return + } + + document.startViewTransition(callback) } - document.startViewTransition(callback) + document.body.addEventListener('click', (evt) => { + if (evt.target.className === 'next-posts' || evt.target.className === 'prev-posts') { + document.documentElement.classList.add('section-posts-animation') + } else { + document.documentElement.classList.remove('section-posts-animation') + } + }) } -document.body.addEventListener('click', (evt) => { - if (evt.target.className === 'next-posts' || evt.target.className === 'prev-posts') { - document.documentElement.classList.add('section-posts-animation') - } else { - document.documentElement.classList.remove('section-posts-animation') - } -}) +fnInit() \ No newline at end of file