1
0
mirror of https://github.com/osmarks/mycorrhiza.git synced 2025-01-08 10:51:09 +00:00
mycorrhiza/static/shortcuts.js
handlerug b43f2836c1
Make shortcuts work outside of English layout (#227)
The keyboard event is tested two times: first time with the original key
property, second time with the key property derived from the key code.
This is done to support non-English shortcuts (which may be added by
wiki administrators).
2024-05-25 23:39:23 +03:00

336 lines
13 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

rrh.l10n('List of shortcuts', { ru: 'Горячие клавиши' })
rrh.l10n('Close this dialog', { ru: 'Закрыть диалог' })
rrh.l10n('Common', { ru: 'Общее' })
rrh.l10n('Home', { ru: 'Главная' })
rrh.l10n('Hypha', { ru: 'Гифа' })
rrh.l10n('Editor', { ru: 'Редактор' })
rrh.l10n('Format', { ru: 'Форматирование' })
rrh.shortcuts = {
// map is the whole shortcut graph. active points to the node of the
// shortcut graph that's currently "selected". When the user presses a
// key, the code finds the key node in the active subgraph and sets
// active to the found node. On each key press we get a narrower view
// until we reach a leaf and execute the action. View this property in
// the JavaScript console of your browser for easier understanding.
map: {},
groups: [],
addGroup(group) {
this.groups.push(group)
},
addBindingToGroup(name, binding) {
let group = this.groups.find(group => group.name === name)
if (!group) {
console.warn('Shortcut group', name, 'not found')
return
}
group.bind(binding)
},
_register(shortcuts, action, other = {}) {
// shortcuts looks like this: ['g r', 'Ctrl+r']
// Every item of shortcuts is a sequential key chord.
for (let chord of strToArr(shortcuts)) {
let leaf = this.map
let keys = chord.trim().split(' ')
for (let key of keys) {
// If there's no existing edge, create one
if (!leaf[key]) leaf[key] = {}
leaf = leaf[key]
if (leaf.action) throw new Error(`Shortcut ${chord} already exists`)
}
// Now we've traversed to the leaf. Bind the shortcut
leaf.action = action
Object.assign(leaf, other)
}
},
_handleKeyDown(event) {
if (event.defaultPrevented) return
if (['Control', 'Alt', 'Shift', 'Meta'].includes(event.key)) return
if ((!event.ctrlKey && !event.metaKey && !event.altKey) &&
event.target instanceof Node && isTextField(event.target)) return
let possibleShortcuts = [keyEventToShortcut(event)]
if (event.code.startsWith('Key')) {
possibleShortcuts.push(keyEventToShortcut({
...event,
key: event.code.replace(/^Key/, '').toLowerCase(),
}))
}
let shortcut = null
for (let possibleShortcut of possibleShortcuts) {
if (possibleShortcut in this.active) {
shortcut = possibleShortcut
break
}
}
if (shortcut === null) {
this._resetActive()
return
}
this.active = this.active[shortcut]
if (this.active.action && (!this.active.element || event.target === this.active.element)) {
event.stopPropagation()
if (this.active.force) event.preventDefault()
this.active.action(event)
this._resetActive()
return
}
if (this.timeout) clearTimeout(this.timeout)
this.timeout = window.setTimeout(() => this._resetActive(), 1500)
},
_resetActive() {
this.active = this.map
if (this.timeout) {
clearTimeout(this.timeout)
this.timeout = null
}
},
}
window.addEventListener('keydown', event => rrh.shortcuts._handleKeyDown(event))
rrh.shortcuts._resetActive()
// Convert a KeyboardEvent into a shortcut string for matching by
// ShortcutHandler.
function keyEventToShortcut(event) {
let elideShift = event.key.toUpperCase() === event.key && event.shiftKey
return (event.ctrlKey ? 'Ctrl+' : '') +
(event.altKey ? 'Alt+' : '') +
(event.metaKey ? 'Meta+' : '') +
(!elideShift && event.shiftKey ? 'Shift+' : '') +
(event.key === ',' ? 'Comma' : event.key === ' ' ? 'Space' : event.key)
}
// Prettify the shortcut string by replacing modifiers and arrow codes with
// Unicode symbol for presentation to the user.
function prettifyShortcut(shortcut) {
let keys = shortcut.split('+')
if (isMac) {
let cmdIdx = keys.indexOf('Meta')
if (cmdIdx !== -1 && keys.length - cmdIdx > 2) {
let tmp = keys[cmdIdx + 1]
keys[cmdIdx + 1] = 'Meta'
keys[cmdIdx] = tmp
}
}
// Add Shift into shortcut strings like Ctrl+L
let lastKey = keys[keys.length - 1]
if (!keys.includes('Shift') && lastKey.toUpperCase() === lastKey && lastKey.toLowerCase() !== lastKey) {
keys.splice(keys.length - 1, 0, 'Shift')
}
for (let i = 0; i < keys.length; i++) {
if (isMac) {
if (keys[i] === 'Ctrl') keys[i] = '⌃'
if (keys[i] === 'Alt') keys[i] = '⌥'
if (keys[i] === 'Shift') keys[i] = '⇧'
if (keys[i] === 'Meta') keys[i] = '⌘'
} else {
if (keys[i] === 'Meta') keys[i] = 'Win'
}
if (i === keys.length - 1 && i > 0 && keys[i].length === 1) {
// Make every key uppercase. This does not introduce any ambiguous
// cases because we insert a Shift modifier for upper-case keys
// earlier.
keys[i] = keys[i].toUpperCase()
}
if (keys[i] === 'ArrowLeft') keys[i] = '←'
if (keys[i] === 'ArrowRight') keys[i] = '→'
if (keys[i] === 'ArrowUp') keys[i] = '↑'
if (keys[i] === 'ArrowDown') keys[i] = '↓'
if (keys[i] === 'Comma') keys[i] = ','
if (keys[i] === 'Enter') keys[i] = '↩'
if (keys[i] === ' ') keys[i] = 'Space'
keys[i] = `<kbd>${keys[i]}</kbd>`
}
return keys.join(isMac ? '' : ' + ')
}
function isTextField(element) {
let name = element.nodeName.toLowerCase()
return name === 'textarea' ||
name === 'select' ||
(name === 'input' && !['submit', 'reset', 'checkbox', 'radio'].includes(element.type)) ||
element.isContentEditable
}
class Shortcut {
constructor(shortcuts, target, description, other) {
this.shortcuts = shortcuts
this.target = target
this.description = rrh.l10n(description)
Object.assign(this, other)
}
}
class ShortcutGroup {
constructor(name, element = null, bindings = []) {
this.name = rrh.l10n(name)
this.shortcuts = []
this.element = element
bindings.forEach(binding => this.bind(binding))
}
bind({ shortcuts, target, description, ...other }) {
shortcuts = strToArr(shortcuts).map(s => s.trim())
if (!other.element) other.element = this.element
if (target instanceof Function) {
this.shortcuts.push({ shortcuts, description })
rrh.shortcuts._register(shortcuts, target, other)
} else if (target instanceof Node) {
this.bind({
shortcuts,
target: () => {
if (isTextField(target)) target.focus()
target.click()
},
description, ...other,
})
} else if (Array.isArray(target) && (target.length === 0 || target[0] instanceof Node)) {
this.shortcuts.push({
shortcuts: shortcuts.map(s => `${s} 1 — 9`),
description,
})
for (let i = 0; i < target.length && i < 9; i++) {
let element = target[i]
rrh.shortcuts._register(shortcuts.map(s => `${s} ${i + 1}`), () => {
if (isTextField(element)) element.focus()
else element.click()
}, other)
}
} else if (typeof target === 'string') {
this.bind({
shortcuts,
target: () => window.location.href = target,
description,
...other,
})
} else if (target !== undefined && target !== null) {
throw new Error('Invalid target type')
}
}
}
function openHelp() {
if ($('.shortcuts-help')) return
document.body.overflow = 'hidden'
let prevActiveElement = document.activeElement
let backdrop = rrh.html`<div class="dialog-backdrop"></div>`
backdrop.onclick = close
document.body.appendChild(backdrop)
let dialog = rrh.html`
<div class="dialog shortcuts-help" tabindex="0">
<div class="dialog__header">
<h1 class="dialog__title">${rrh.l10n('List of shortcuts')}</h1>
<button class="dialog__close-button" aria-label="${rrh.l10n('Close this dialog')}"></button>
</div>
<div class="dialog__content"></div>
</div>
`
dialog.querySelector('.dialog__close-button').onclick = close
dialog.onkeydown = event => {
if (event.key === 'Escape') close()
}
document.body.appendChild(dialog)
dialog.focus()
function close() {
document.body.overflow = ''
document.body.removeChild(backdrop)
document.body.removeChild(dialog)
if (prevActiveElement) prevActiveElement.focus()
}
function formatShortcuts(shortcuts) {
return shortcuts.map(s => s.split(' ')
.map(prettifyShortcut).join(' '))
.join(' <span class="kbd-or">or</span> ')
}
for (let group of rrh.shortcuts.groups) {
if (group.shortcuts.length === 0) continue
dialog.querySelector('.dialog__content').appendChild(rrh.html`
<div class="shortcuts-group">
<h2 class="shortcuts-group-heading">${group.name}</h2>
<ul class="shortcuts-list">
${group.shortcuts.map(({ description, shortcuts }) => `
<li class="shortcut-row">
<div class="shortcut-row__description">${description}</div>
<div class="shortcut-row__keys">
${formatShortcuts(shortcuts)}
</div>
</li>
`)}
</ul>
</div>
`)
}
}
rrh.shortcuts.addGroup(new ShortcutGroup('Common', null, [
new Shortcut('g', $$('.top-bar__highlight-link'), 'First 9 header links'),
new Shortcut('g h', '/', 'Home'),
new Shortcut('g l', '/list/', 'List of hyphae'),
new Shortcut('g r', '/recent-changes/', 'Recent changes'),
new Shortcut('g u', $('.auth-links__user-link'), 'Your profiles hypha'),
new Shortcut(['?', isMac ? 'Meta+/' : 'Ctrl+/'], openHelp, 'Shortcut help', { force: true }),
]))
if (document.body.dataset.rrhAddr.startsWith('/hypha')) {
rrh.shortcuts.addGroup(new ShortcutGroup('Hypha', null, [
new Shortcut('', $$('article .wikilink'), 'First 9 hyphas links'),
new Shortcut(['p', 'Alt+ArrowLeft', 'Ctrl+Alt+ArrowLeft'], $('.prevnext__prev'), 'Previous hypha'),
new Shortcut(['n', 'Alt+ArrowRight', 'Ctrl+Alt+ArrowRight'], $('.prevnext__next'), 'Next hypha'),
new Shortcut(['s', 'Alt+ArrowUp', 'Ctrl+Alt+ArrowUp'], $$('.navi-title a').slice(1, -1).slice(-1)[0], 'Parent hypha'),
new Shortcut(['c', 'Alt+ArrowDown', 'Ctrl+Alt+ArrowDown'], $('.subhyphae__link'), 'First child hypha'),
new Shortcut(['e', isMac ? 'Meta+Enter' : 'Ctrl+Enter'], $('.btn__link_navititle[href^="/edit/"]'), 'Edit this hypha'),
new Shortcut('v', $('.hypha-info__link[href^="/hypha/"]'), 'Go to hyphas page'),
new Shortcut('a', $('.hypha-info__link[href^="/media/"]'), 'Go to media management'),
new Shortcut('h', $('.hypha-info__link[href^="/history/"]'), 'Go to history'),
new Shortcut('r', $('.hypha-info__link[href^="/rename/"]'), 'Rename this hypha'),
new Shortcut('b', $('.hypha-info__link[href^="/backlinks/"]'), 'Backlinks'),
]))
}
if (document.body.dataset.rrhAddr.startsWith('/edit')) {
rrh.shortcuts.addGroup(new ShortcutGroup('Editor', null, [
new Shortcut(isMac ? 'Meta+Enter' : 'Ctrl+Enter', $('.edit-form__save'), 'Save changes'),
new Shortcut(isMac ? 'Meta+Shift+Enter' : 'Ctrl+Shift+Enter', $('.edit-form__preview'), 'Preview changes'),
]))
if (editTextarea) {
rrh.shortcuts.addGroup(new ShortcutGroup('Format', null, [
new Shortcut(isMac ? 'Meta+b' : 'Ctrl+b', wrapBold, 'Bold', { force: true }),
new Shortcut(isMac ? 'Meta+i' : 'Ctrl+i', wrapItalic, 'Italic', { force: true }),
new Shortcut(isMac ? 'Meta+Shift+m' : 'Ctrl+M', wrapMonospace, 'Monospaced', { force: true }),
new Shortcut(isMac ? 'Meta+Shift+i' : 'Ctrl+I', wrapHighlighted, 'Highlight', { force: true }),
new Shortcut(isMac ? 'Meta+.' : 'Ctrl+.', wrapLifted, 'Superscript', { force: true }),
new Shortcut(isMac ? 'Meta+Comma' : 'Ctrl+Comma', wrapLowered, 'Subscript', { force: true }),
new Shortcut(isMac ? 'Meta+Shift+x' : 'Ctrl+X', wrapStrikethrough, 'Strikethrough', { force: true }),
new Shortcut(isMac ? 'Meta+k' : 'Ctrl+k', wrapLink, 'Inline link', { force: true }),
// Apparently, ⌘; conflicts with a Safari's hotkey. Whatever.
new Shortcut(isMac ? 'Meta+;' : 'Ctrl+;', insertDate, 'Insert date UTC', { force: true }),
]))
}
}