|
import { addEventListener } from '../../helpers/event.js'; |
|
import { isFunction, isHTMLElement, isReadable } from '../../helpers/is.js'; |
|
import { get, readable } from 'svelte/store'; |
|
import { effect, executeCallbacks, kbd, noop } from '../../helpers/index.js'; |
|
|
|
|
|
|
|
|
|
|
|
const documentEscapeKeyStore = readable(undefined, (set) => { |
|
|
|
|
|
|
|
|
|
function keydown(event) { |
|
if (event && event.key === kbd.ESCAPE) { |
|
set(event); |
|
} |
|
|
|
set(undefined); |
|
} |
|
|
|
const unsubscribe = addEventListener(document, 'keydown', keydown, { |
|
passive: false, |
|
}); |
|
|
|
return unsubscribe; |
|
}); |
|
export const useEscapeKeydown = (node, config = {}) => { |
|
let unsub = noop; |
|
function update(config = {}) { |
|
unsub(); |
|
const options = { enabled: true, ...config }; |
|
const enabled = (isReadable(options.enabled) ? options.enabled : readable(options.enabled)); |
|
unsub = executeCallbacks( |
|
|
|
documentEscapeKeyStore.subscribe((e) => { |
|
if (!e || !get(enabled)) |
|
return; |
|
const target = e.target; |
|
if (!isHTMLElement(target) || target.closest('[data-escapee]') !== node) { |
|
return; |
|
} |
|
e.preventDefault(); |
|
|
|
if (options.ignore) { |
|
if (isFunction(options.ignore)) { |
|
if (options.ignore(e)) |
|
return; |
|
} |
|
|
|
else if (Array.isArray(options.ignore)) { |
|
if (options.ignore.length > 0 && |
|
options.ignore.some((ignoreEl) => { |
|
return ignoreEl && target === ignoreEl; |
|
})) |
|
return; |
|
} |
|
} |
|
|
|
options.handler?.(e); |
|
}), effect(enabled, ($enabled) => { |
|
if ($enabled) { |
|
node.dataset.escapee = ''; |
|
} |
|
else { |
|
delete node.dataset.escapee; |
|
} |
|
})); |
|
} |
|
update(config); |
|
return { |
|
update, |
|
destroy() { |
|
node.removeAttribute('data-escapee'); |
|
unsub(); |
|
}, |
|
}; |
|
}; |
|
|