| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243 | 'use strict';const readline = require('readline');const combos = require('./combos');/* eslint-disable no-control-regex */const metaKeyCodeRe = /^(?:\x1b)([a-zA-Z0-9])$/;const fnKeyRe = /^(?:\x1b+)(O|N|\[|\[\[)(?:(\d+)(?:;(\d+))?([~^$])|(?:1;)?(\d+)?([a-zA-Z]))/;const keyName = {    /* xterm/gnome ESC O letter */    'OP': 'f1',    'OQ': 'f2',    'OR': 'f3',    'OS': 'f4',    /* xterm/rxvt ESC [ number ~ */    '[11~': 'f1',    '[12~': 'f2',    '[13~': 'f3',    '[14~': 'f4',    /* from Cygwin and used in libuv */    '[[A': 'f1',    '[[B': 'f2',    '[[C': 'f3',    '[[D': 'f4',    '[[E': 'f5',    /* common */    '[15~': 'f5',    '[17~': 'f6',    '[18~': 'f7',    '[19~': 'f8',    '[20~': 'f9',    '[21~': 'f10',    '[23~': 'f11',    '[24~': 'f12',    /* xterm ESC [ letter */    '[A': 'up',    '[B': 'down',    '[C': 'right',    '[D': 'left',    '[E': 'clear',    '[F': 'end',    '[H': 'home',    /* xterm/gnome ESC O letter */    'OA': 'up',    'OB': 'down',    'OC': 'right',    'OD': 'left',    'OE': 'clear',    'OF': 'end',    'OH': 'home',    /* xterm/rxvt ESC [ number ~ */    '[1~': 'home',    '[2~': 'insert',    '[3~': 'delete',    '[4~': 'end',    '[5~': 'pageup',    '[6~': 'pagedown',    /* putty */    '[[5~': 'pageup',    '[[6~': 'pagedown',    /* rxvt */    '[7~': 'home',    '[8~': 'end',    /* rxvt keys with modifiers */    '[a': 'up',    '[b': 'down',    '[c': 'right',    '[d': 'left',    '[e': 'clear',    '[2$': 'insert',    '[3$': 'delete',    '[5$': 'pageup',    '[6$': 'pagedown',    '[7$': 'home',    '[8$': 'end',    'Oa': 'up',    'Ob': 'down',    'Oc': 'right',    'Od': 'left',    'Oe': 'clear',    '[2^': 'insert',    '[3^': 'delete',    '[5^': 'pageup',    '[6^': 'pagedown',    '[7^': 'home',    '[8^': 'end',    /* misc. */    '[Z': 'tab',}function isShiftKey(code) {    return ['[a', '[b', '[c', '[d', '[e', '[2$', '[3$', '[5$', '[6$', '[7$', '[8$', '[Z'].includes(code)}function isCtrlKey(code) {    return [ 'Oa', 'Ob', 'Oc', 'Od', 'Oe', '[2^', '[3^', '[5^', '[6^', '[7^', '[8^'].includes(code)}const keypress = (s = '', event = {}) => {  let parts;  let key = {    name: event.name,    ctrl: false,    meta: false,    shift: false,    option: false,    sequence: s,    raw: s,    ...event  };  if (Buffer.isBuffer(s)) {    if (s[0] > 127 && s[1] === void 0) {      s[0] -= 128;      s = '\x1b' + String(s);    } else {      s = String(s);    }  } else if (s !== void 0 && typeof s !== 'string') {    s = String(s);  } else if (!s) {    s = key.sequence || '';  }  key.sequence = key.sequence || s || key.name;  if (s === '\r') {    // carriage return    key.raw = void 0;    key.name = 'return';  } else if (s === '\n') {    // enter, should have been called linefeed    key.name = 'enter';  } else if (s === '\t') {    // tab    key.name = 'tab';  } else if (s === '\b' || s === '\x7f' || s === '\x1b\x7f' || s === '\x1b\b') {    // backspace or ctrl+h    key.name = 'backspace';    key.meta = s.charAt(0) === '\x1b';  } else if (s === '\x1b' || s === '\x1b\x1b') {    // escape key    key.name = 'escape';    key.meta = s.length === 2;  } else if (s === ' ' || s === '\x1b ') {    key.name = 'space';    key.meta = s.length === 2;  } else if (s <= '\x1a') {    // ctrl+letter    key.name = String.fromCharCode(s.charCodeAt(0) + 'a'.charCodeAt(0) - 1);    key.ctrl = true;  } else if (s.length === 1 && s >= '0' && s <= '9') {    // number    key.name = 'number';  } else if (s.length === 1 && s >= 'a' && s <= 'z') {    // lowercase letter    key.name = s;  } else if (s.length === 1 && s >= 'A' && s <= 'Z') {    // shift+letter    key.name = s.toLowerCase();    key.shift = true;  } else if ((parts = metaKeyCodeRe.exec(s))) {    // meta+character key    key.meta = true;    key.shift = /^[A-Z]$/.test(parts[1]);  } else if ((parts = fnKeyRe.exec(s))) {    let segs = [...s];    if (segs[0] === '\u001b' && segs[1] === '\u001b') {      key.option = true;    }    // ansi escape sequence    // reassemble the key code leaving out leading \x1b's,    // the modifier key bitflag and any meaningless "1;" sequence    let code = [parts[1], parts[2], parts[4], parts[6]].filter(Boolean).join('');    let modifier = (parts[3] || parts[5] || 1) - 1;    // Parse the key modifier    key.ctrl = !!(modifier & 4);    key.meta = !!(modifier & 10);    key.shift = !!(modifier & 1);    key.code = code;    key.name = keyName[code];    key.shift = isShiftKey(code) || key.shift;    key.ctrl = isCtrlKey(code) || key.ctrl;  }  return key;};keypress.listen = (options = {}, onKeypress) => {  let { stdin } = options;  if (!stdin || (stdin !== process.stdin && !stdin.isTTY)) {    throw new Error('Invalid stream passed');  }  let rl = readline.createInterface({ terminal: true, input: stdin });  readline.emitKeypressEvents(stdin, rl);  let on = (buf, key) => onKeypress(buf, keypress(buf, key), rl);  let isRaw = stdin.isRaw;  if (stdin.isTTY) stdin.setRawMode(true);  stdin.on('keypress', on);  rl.resume();  let off = () => {    if (stdin.isTTY) stdin.setRawMode(isRaw);    stdin.removeListener('keypress', on);    rl.pause();    rl.close();  };  return off;};keypress.action = (buf, key, customActions) => {  let obj = { ...combos, ...customActions };  if (key.ctrl) {    key.action = obj.ctrl[key.name];    return key;  }  if (key.option && obj.option) {    key.action = obj.option[key.name];    return key;  }  if (key.shift) {    key.action = obj.shift[key.name];    return key;  }  key.action = obj.keys[key.name];  return key;};module.exports = keypress;
 |