function reverseMap(map) { const reversed = {}; for (const [key, value] of Object.entries(map)) { reversed[value] = key; } return reversed; } /** * @extends {Component} */ class Shortcuts { setup() { this.container = this.$el; this.mapById = JSON.parse(this.$opts.keyMap); this.mapByShortcut = reverseMap(this.mapById); this.hintsShowing = false; this.hideHints = this.hideHints.bind(this); this.setupListeners(); } setupListeners() { window.addEventListener('keydown', event => { if (event.target.closest('input, select, textarea')) { return; } this.handleShortcutPress(event); }); window.addEventListener('keydown', event => { if (event.key === '?') { this.hintsShowing ? this.hideHints() : this.showHints(); } }); } /** * @param {KeyboardEvent} event */ handleShortcutPress(event) { const keys = [ event.ctrlKey ? 'Ctrl' : '', event.metaKey ? 'Cmd' : '', event.key, ]; const combo = keys.filter(s => Boolean(s)).join(' + '); const shortcutId = this.mapByShortcut[combo]; if (shortcutId) { const wasHandled = this.runShortcut(shortcutId); if (wasHandled) { event.preventDefault(); } } } /** * Run the given shortcut, and return a boolean to indicate if the event * was successfully handled by a shortcut action. * @param {String} id * @return {boolean} */ runShortcut(id) { const el = this.container.querySelector(`[data-shortcut="${id}"]`); console.info('Shortcut run', el); if (!el) { return false; } if (el.matches('input, textarea, select')) { el.focus(); return true; } if (el.matches('a, button')) { el.click(); return true; } if (el.matches('div[tabindex]')) { el.click(); el.focus(); return true; } console.error(`Shortcut attempted to be ran for element type that does not have handling setup`, el); return false; } showHints() { const wrapper = document.createElement('div'); wrapper.classList.add('shortcut-container'); this.container.append(wrapper); const shortcutEls = this.container.querySelectorAll('[data-shortcut]'); const displayedIds = new Set(); for (const shortcutEl of shortcutEls) { const id = shortcutEl.getAttribute('data-shortcut'); if (displayedIds.has(id)) { continue; } const key = this.mapById[id]; this.showHintLabel(shortcutEl, key, wrapper); displayedIds.add(id); } window.addEventListener('scroll', this.hideHints); window.addEventListener('focus', this.hideHints); window.addEventListener('blur', this.hideHints); window.addEventListener('click', this.hideHints); this.hintsShowing = true; } /** * @param {Element} targetEl * @param {String} key * @param {Element} wrapper */ showHintLabel(targetEl, key, wrapper) { const targetBounds = targetEl.getBoundingClientRect(); const label = document.createElement('div'); label.classList.add('shortcut-hint'); label.textContent = key; const linkage = document.createElement('div'); linkage.classList.add('shortcut-linkage'); linkage.style.left = targetBounds.x + 'px'; linkage.style.top = targetBounds.y + 'px'; linkage.style.width = targetBounds.width + 'px'; linkage.style.height = targetBounds.height + 'px'; wrapper.append(label, linkage); const labelBounds = label.getBoundingClientRect(); label.style.insetInlineStart = `${((targetBounds.x + targetBounds.width) - (labelBounds.width + 6))}px`; label.style.insetBlockStart = `${(targetBounds.y + (targetBounds.height - labelBounds.height) / 2)}px`; } hideHints() { const wrapper = this.container.querySelector('.shortcut-container'); wrapper.remove(); window.removeEventListener('scroll', this.hideHints); window.removeEventListener('focus', this.hideHints); window.removeEventListener('blur', this.hideHints); window.removeEventListener('click', this.hideHints); this.hintsShowing = false; } } export default Shortcuts;