2020-09-26 06:02:29 +08:00
|
|
|
|
|
|
|
(win => {
|
|
|
|
|
|
|
|
const
|
|
|
|
doc = document,
|
|
|
|
meta = /Mac OS X/.test(navigator.userAgent) ? 'meta' : 'ctrl',
|
2021-02-20 02:39:04 +08:00
|
|
|
_scopes = {
|
|
|
|
all: {}
|
|
|
|
},
|
2020-09-26 16:20:24 +08:00
|
|
|
toArray = v => Array.isArray(v) ? v : v.split(/\s*,\s*/),
|
2020-09-26 06:02:29 +08:00
|
|
|
|
2020-10-09 17:58:15 +08:00
|
|
|
keydown = event => {
|
2020-10-09 21:00:26 +08:00
|
|
|
let key = (event.key || event.code || '').toLowerCase().replace(' ','space'),
|
2020-10-09 17:58:15 +08:00
|
|
|
scopes = [];
|
|
|
|
scope[key] && scopes.push(scope[key]);
|
|
|
|
_scope !== 'all' && _scopes.all[key] && scopes.push(_scopes.all[key]);
|
|
|
|
if (scopes.length && win.shortcuts.filter(event.target)) {
|
|
|
|
let modifiers = [];
|
|
|
|
event.altKey && modifiers.push('alt');
|
|
|
|
event.ctrlKey && modifiers.push('ctrl');
|
|
|
|
event.metaKey && modifiers.push('meta');
|
|
|
|
event.shiftKey && modifiers.push('shift');
|
|
|
|
modifiers = modifiers.join('+');
|
|
|
|
scopes.forEach(actions => {
|
|
|
|
if (actions[modifiers]) {
|
|
|
|
// for each potential shortcut
|
|
|
|
actions[modifiers].forEach(cmd => {
|
|
|
|
try {
|
|
|
|
// call the handler and stop the event if neccessary
|
|
|
|
if (!event.defaultPrevented && cmd(event) === false) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
});
|
2020-09-26 06:02:29 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2020-09-26 15:34:23 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
let
|
|
|
|
scope = {},
|
|
|
|
_scope = 'all';
|
|
|
|
|
2020-09-26 06:02:29 +08:00
|
|
|
win.shortcuts = {
|
2020-09-26 15:34:23 +08:00
|
|
|
on: () => doc.addEventListener('keydown', keydown),
|
|
|
|
off: () => doc.removeEventListener('keydown', keydown),
|
|
|
|
add: (keys, modifiers, scopes, method) => {
|
2020-09-26 06:02:29 +08:00
|
|
|
if (method === undefined) {
|
2020-09-26 15:34:23 +08:00
|
|
|
method = scopes;
|
|
|
|
scopes = 'all';
|
2020-09-26 06:02:29 +08:00
|
|
|
}
|
2020-09-26 15:34:23 +08:00
|
|
|
toArray(scopes).forEach(scope => {
|
|
|
|
if (!_scopes[scope]) {
|
|
|
|
_scopes[scope] = {};
|
2020-09-26 06:02:29 +08:00
|
|
|
}
|
2020-09-26 15:34:23 +08:00
|
|
|
toArray(keys).forEach(key => {
|
|
|
|
key = key.toLowerCase();
|
|
|
|
if (!_scopes[scope][key]) {
|
|
|
|
_scopes[scope][key] = {};
|
|
|
|
}
|
|
|
|
modifiers = toArray(modifiers)
|
|
|
|
.map(key => 'meta' == key ? meta : key)
|
|
|
|
.unique().sort().join('+');
|
|
|
|
if (!_scopes[scope][key][modifiers]) {
|
|
|
|
_scopes[scope][key][modifiers] = [];
|
|
|
|
}
|
|
|
|
_scopes[scope][key][modifiers].push(method);
|
|
|
|
});
|
2020-09-26 06:02:29 +08:00
|
|
|
});
|
|
|
|
},
|
2020-09-26 15:34:23 +08:00
|
|
|
setScope: value => {
|
|
|
|
_scope = value || 'all';
|
|
|
|
scope = _scopes[_scope] || {};
|
|
|
|
},
|
2020-09-26 06:02:29 +08:00
|
|
|
getScope: () => _scope,
|
|
|
|
getMetaKey: () => meta,
|
|
|
|
// ignore keydown in any element that supports keyboard input
|
2020-10-09 16:31:44 +08:00
|
|
|
filter: node => !(!node.closest || node.closest('input,select,textarea,[contenteditable]'))
|
2020-09-26 06:02:29 +08:00
|
|
|
};
|
|
|
|
|
2020-09-26 15:34:23 +08:00
|
|
|
win.shortcuts.on();
|
|
|
|
|
2020-09-26 06:02:29 +08:00
|
|
|
})(this);
|