X-Git-Url: http://source.bookstackapp.com/bookstack/blobdiff_plain/a6633642232efd164d4708967ab59e498fbff896..refs/pull/4794/head:/resources/js/services/util.js diff --git a/resources/js/services/util.js b/resources/js/services/util.js index de2ca20c1..59affff88 100644 --- a/resources/js/services/util.js +++ b/resources/js/services/util.js @@ -1,30 +1,28 @@ - - /** * Returns a function, that, as long as it continues to be invoked, will not * be triggered. The function will be called after it stops being called for * N milliseconds. If `immediate` is passed, trigger the function on the * leading edge, instead of the trailing. * @attribution https://davidwalsh.name/javascript-debounce-function - * @param func - * @param wait - * @param immediate + * @param {Function} func + * @param {Number} waitMs + * @param {Boolean} immediate * @returns {Function} */ -export function debounce(func, wait, immediate) { +export function debounce(func, waitMs, immediate) { let timeout; - return function() { - const context = this, args = arguments; - const later = function() { + return function debouncedWrapper(...args) { + const context = this; + const later = function debouncedTimeout() { timeout = null; if (!immediate) func.apply(context, args); }; const callNow = immediate && !timeout; clearTimeout(timeout); - timeout = setTimeout(later, wait); + timeout = setTimeout(later, waitMs); if (callNow) func.apply(context, args); }; -}; +} /** * Scroll and highlight an element. @@ -32,19 +30,29 @@ export function debounce(func, wait, immediate) { */ export function scrollAndHighlightElement(element) { if (!element) return; + + const parentDetails = element.closest('details'); + if (parentDetails && !parentDetails.open) { + parentDetails.open = true; + } + element.scrollIntoView({behavior: 'smooth'}); - const color = document.getElementById('custom-styles').getAttribute('data-color-light'); - const initColor = window.getComputedStyle(element).getPropertyValue('background-color'); - element.style.backgroundColor = color; - setTimeout(() => { - element.classList.add('selectFade'); - element.style.backgroundColor = initColor; - }, 10); + const highlight = getComputedStyle(document.body).getPropertyValue('--color-link'); + element.style.outline = `2px dashed ${highlight}`; + element.style.outlineOffset = '5px'; + element.style.transition = null; setTimeout(() => { - element.classList.remove('selectFade'); - element.style.backgroundColor = ''; - }, 3000); + element.style.transition = 'outline linear 3s'; + element.style.outline = '2px dashed rgba(0, 0, 0, 0)'; + const listener = () => { + element.removeEventListener('transitionend', listener); + element.style.transition = null; + element.style.outline = null; + element.style.outlineOffset = null; + }; + element.addEventListener('transitionend', listener); + }, 1000); } /** @@ -55,11 +63,11 @@ export function scrollAndHighlightElement(element) { */ export function escapeHtml(unsafe) { return unsafe - .replace(/&/g, "&") - .replace(//g, ">") - .replace(/"/g, """) - .replace(/'/g, "'"); + .replace(/&/g, '&') + .replace(//g, '>') + .replace(/"/g, '"') + .replace(/'/g, '''); } /** @@ -68,6 +76,18 @@ export function escapeHtml(unsafe) { * @returns {string} */ export function uniqueId() { - const S4 = () => (((1+Math.random())*0x10000)|0).toString(16).substring(1); - return (S4()+S4()+"-"+S4()+"-"+S4()+"-"+S4()+"-"+S4()+S4()+S4()); -} \ No newline at end of file + // eslint-disable-next-line no-bitwise + const S4 = () => (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1); + return (`${S4() + S4()}-${S4()}-${S4()}-${S4()}-${S4()}${S4()}${S4()}`); +} + +/** + * Create a promise that resolves after the given time. + * @param {int} timeMs + * @returns {Promise} + */ +export function wait(timeMs) { + return new Promise(res => { + setTimeout(res, timeMs); + }); +}