blob: 673b87aad4263bba4c8586567ce0dbd3a89a50b4 [file] [log] [blame]
/*
* Copyright (C) 2009 Apple Inc. All rights reserved.
* Copyright (C) 2009 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @unrestricted
*/
export default class KeyboardShortcut {
/**
* Creates a number encoding keyCode in the lower 8 bits and modifiers mask in the higher 8 bits.
* It is useful for matching pressed keys.
*
* @param {number|string} keyCode The code of the key, or a character "a-z" which is converted to a keyCode value.
* @param {number=} modifiers Optional list of modifiers passed as additional parameters.
* @return {number}
*/
static makeKey(keyCode, modifiers) {
if (typeof keyCode === 'string') {
keyCode = keyCode.charCodeAt(0) - (/^[a-z]/.test(keyCode) ? 32 : 0);
}
modifiers = modifiers || Modifiers.None;
return KeyboardShortcut._makeKeyFromCodeAndModifiers(keyCode, modifiers);
}
/**
* @param {?KeyboardEvent} keyboardEvent
* @return {number}
*/
static makeKeyFromEvent(keyboardEvent) {
let modifiers = Modifiers.None;
if (keyboardEvent.shiftKey) {
modifiers |= Modifiers.Shift;
}
if (keyboardEvent.ctrlKey) {
modifiers |= Modifiers.Ctrl;
}
if (keyboardEvent.altKey) {
modifiers |= Modifiers.Alt;
}
if (keyboardEvent.metaKey) {
modifiers |= Modifiers.Meta;
}
// Use either a real or a synthetic keyCode (for events originating from extensions).
const keyCode = keyboardEvent.keyCode || keyboardEvent['__keyCode'];
return KeyboardShortcut._makeKeyFromCodeAndModifiers(keyCode, modifiers);
}
/**
* @param {?KeyboardEvent} keyboardEvent
* @return {number}
*/
static makeKeyFromEventIgnoringModifiers(keyboardEvent) {
const keyCode = keyboardEvent.keyCode || keyboardEvent['__keyCode'];
return KeyboardShortcut._makeKeyFromCodeAndModifiers(keyCode, Modifiers.None);
}
/**
* @param {(?KeyboardEvent|?MouseEvent)} event
* @return {boolean}
*/
static eventHasCtrlOrMeta(event) {
return Host.isMac() ? event.metaKey && !event.ctrlKey : event.ctrlKey && !event.metaKey;
}
/**
* @param {!Event} event
* @return {boolean}
*/
static hasNoModifiers(event) {
return !event.ctrlKey && !event.shiftKey && !event.altKey && !event.metaKey;
}
/**
* @param {string|!UI.KeyboardShortcut.Key} key
* @param {number=} modifiers
* @return {!KeyboardShortcut.Descriptor}
*/
static makeDescriptor(key, modifiers) {
return {
key: KeyboardShortcut.makeKey(typeof key === 'string' ? key : key.code, modifiers),
name: KeyboardShortcut.shortcutToString(key, modifiers)
};
}
/**
* @param {string} shortcut
* @return {?KeyboardShortcut.Descriptor}
*/
static makeDescriptorFromBindingShortcut(shortcut) {
const parts = shortcut.split(/\+(?!$)/);
let modifiers = 0;
let keyString;
for (let i = 0; i < parts.length; ++i) {
if (typeof Modifiers[parts[i]] !== 'undefined') {
modifiers |= Modifiers[parts[i]];
continue;
}
console.assert(
i === parts.length - 1, 'Only one key other than modifier is allowed in shortcut <' + shortcut + '>');
keyString = parts[i];
break;
}
console.assert(keyString, 'Modifiers-only shortcuts are not allowed (encountered <' + shortcut + '>)');
if (!keyString) {
return null;
}
const key = Keys[keyString] || KeyBindings[keyString];
if (key && key.shiftKey) {
modifiers |= Modifiers.Shift;
}
return KeyboardShortcut.makeDescriptor(key ? key : keyString, modifiers);
}
/**
* @param {string|!UI.KeyboardShortcut.Key} key
* @param {number=} modifiers
* @return {string}
*/
static shortcutToString(key, modifiers) {
return KeyboardShortcut._modifiersToString(modifiers) + KeyboardShortcut._keyName(key);
}
/**
* @param {string|!UI.KeyboardShortcut.Key} key
* @return {string}
*/
static _keyName(key) {
if (typeof key === 'string') {
return key.toUpperCase();
}
if (typeof key.name === 'string') {
return key.name;
}
return key.name[Host.platform()] || key.name.other || '';
}
/**
* @param {number} keyCode
* @param {?number} modifiers
* @return {number}
*/
static _makeKeyFromCodeAndModifiers(keyCode, modifiers) {
return (keyCode & 255) | (modifiers << 8);
}
/**
* @param {number} key
* @return {!{keyCode: number, modifiers: number}}
*/
static keyCodeAndModifiersFromKey(key) {
return {keyCode: key & 255, modifiers: key >> 8};
}
/**
* @param {number|undefined} modifiers
* @return {string}
*/
static _modifiersToString(modifiers) {
const isMac = Host.isMac();
const m = Modifiers;
const modifierNames = new Map([
[m.Ctrl, isMac ? 'Ctrl\u2004' : 'Ctrl\u200A+\u200A'], [m.Alt, isMac ? '\u2325\u2004' : 'Alt\u200A+\u200A'],
[m.Shift, isMac ? '\u21e7\u2004' : 'Shift\u200A+\u200A'], [m.Meta, isMac ? '\u2318\u2004' : 'Win\u200A+\u200A']
]);
return [m.Meta, m.Ctrl, m.Alt, m.Shift].map(mapModifiers).join('');
/**
* @param {number} m
* @return {string}
*/
function mapModifiers(m) {
return modifiers & m ? /** @type {string} */ (modifierNames.get(m)) : '';
}
}
}
/**
* Constants for encoding modifier key set as a bit mask.
* @see #_makeKeyFromCodeAndModifiers
*/
export const Modifiers = {
None: 0, // Constant for empty modifiers set.
Shift: 1,
Ctrl: 2,
Alt: 4,
Meta: 8, // Command key on Mac, Win key on other platforms.
get CtrlOrMeta() {
// "default" command/ctrl key for platform, Command on Mac, Ctrl on other platforms
return Host.isMac() ? this.Meta : this.Ctrl;
},
get ShiftOrOption() {
// Option on Mac, Shift on other platforms
return Host.isMac() ? this.Alt : this.Shift;
}
};
/** @type {!Object.<string, !UI.KeyboardShortcut.Key>} */
export const Keys = {
Backspace: {code: 8, name: '\u21a4'},
Tab: {code: 9, name: {mac: '\u21e5', other: 'Tab'}},
Enter: {code: 13, name: {mac: '\u21a9', other: 'Enter'}},
Shift: {code: 16, name: {mac: '\u21e7', other: 'Shift'}},
Ctrl: {code: 17, name: 'Ctrl'},
Esc: {code: 27, name: 'Esc'},
Space: {code: 32, name: 'Space'},
PageUp: {code: 33, name: {mac: '\u21de', other: 'PageUp'}}, // also NUM_NORTH_EAST
PageDown: {code: 34, name: {mac: '\u21df', other: 'PageDown'}}, // also NUM_SOUTH_EAST
End: {code: 35, name: {mac: '\u2197', other: 'End'}}, // also NUM_SOUTH_WEST
Home: {code: 36, name: {mac: '\u2196', other: 'Home'}}, // also NUM_NORTH_WEST
Left: {code: 37, name: '\u2190'}, // also NUM_WEST
Up: {code: 38, name: '\u2191'}, // also NUM_NORTH
Right: {code: 39, name: '\u2192'}, // also NUM_EAST
Down: {code: 40, name: '\u2193'}, // also NUM_SOUTH
Delete: {code: 46, name: 'Del'},
Zero: {code: 48, name: '0'},
H: {code: 72, name: 'H'},
N: {code: 78, name: 'N'},
P: {code: 80, name: 'P'},
Meta: {code: 91, name: 'Meta'},
F1: {code: 112, name: 'F1'},
F2: {code: 113, name: 'F2'},
F3: {code: 114, name: 'F3'},
F4: {code: 115, name: 'F4'},
F5: {code: 116, name: 'F5'},
F6: {code: 117, name: 'F6'},
F7: {code: 118, name: 'F7'},
F8: {code: 119, name: 'F8'},
F9: {code: 120, name: 'F9'},
F10: {code: 121, name: 'F10'},
F11: {code: 122, name: 'F11'},
F12: {code: 123, name: 'F12'},
Semicolon: {code: 186, name: ';'},
NumpadPlus: {code: 107, name: 'Numpad +'},
NumpadMinus: {code: 109, name: 'Numpad -'},
Numpad0: {code: 96, name: 'Numpad 0'},
Plus: {code: 187, name: '+'},
Comma: {code: 188, name: ','},
Minus: {code: 189, name: '-'},
Period: {code: 190, name: '.'},
Slash: {code: 191, name: '/'},
QuestionMark: {code: 191, name: '?'},
Apostrophe: {code: 192, name: '`'},
Tilde: {code: 192, name: 'Tilde'},
LeftSquareBracket: {code: 219, name: '['},
RightSquareBracket: {code: 221, name: ']'},
Backslash: {code: 220, name: '\\'},
SingleQuote: {code: 222, name: '\''},
get CtrlOrMeta() {
// "default" command/ctrl key for platform, Command on Mac, Ctrl on other platforms
return Host.isMac() ? this.Meta : this.Ctrl;
},
};
export const KeyBindings = {};
(function() {
for (const key in Keys) {
const descriptor = Keys[key];
if (typeof descriptor === 'object' && descriptor['code']) {
const name = typeof descriptor['name'] === 'string' ? descriptor['name'] : key;
KeyBindings[name] = descriptor;
}
}
})();
/* Legacy exported object*/
self.UI = self.UI || {};
/* Legacy exported object*/
UI = UI || {};
/** @constructor */
UI.KeyboardShortcut = KeyboardShortcut;
/**
* Constants for encoding modifier key set as a bit mask.
* @see #_makeKeyFromCodeAndModifiers
*/
UI.KeyboardShortcut.Modifiers = Modifiers;
/** @type {!Object.<string, !UI.KeyboardShortcut.Key>} */
UI.KeyboardShortcut.Keys = Keys;
/** @typedef {!{code: number, name: (string|!Object.<string, string>)}} */
UI.KeyboardShortcut.Key;
/** @typedef {!{key: number, name: string}} */
UI.KeyboardShortcut.Descriptor;