Browse Source

feat: add "debug mode" in workflow settings

Ahmad Kholid 3 years ago
parent
commit
4dfc35b096

+ 10 - 5
src/background/index.js

@@ -353,13 +353,18 @@ message.on('set:active-tab', (tabId) => {
   return browser.tabs.update(tabId, { active: true });
   return browser.tabs.update(tabId, { active: true });
 });
 });
 
 
-message.on('get:sender', (_, sender) => {
-  return sender;
-});
-message.on('get:tab-screenshot', (options) => {
-  return browser.tabs.captureVisibleTab(options);
+message.on('debugger:send-command', ({ tabId, method, params }) => {
+  return new Promise((resolve) => {
+    console.log(tabId, method, params);
+    chrome.debugger.sendCommand({ tabId }, method, params, resolve);
+  });
 });
 });
+
+message.on('get:sender', (_, sender) => sender);
 message.on('get:file', (path) => getFile(path));
 message.on('get:file', (path) => getFile(path));
+message.on('get:tab-screenshot', (options) =>
+  browser.tabs.captureVisibleTab(options)
+);
 
 
 message.on('collection:execute', (collection) => {
 message.on('collection:execute', (collection) => {
   const engine = new CollectionEngine(collection, {
   const engine = new CollectionEngine(collection, {

+ 5 - 18
src/background/workflow-engine/blocks-handler.js

@@ -1,25 +1,12 @@
 import { toCamelCase } from '@/utils/helper';
 import { toCamelCase } from '@/utils/helper';
 
 
 const blocksHandler = require.context('./blocks-handler', false, /\.js$/);
 const blocksHandler = require.context('./blocks-handler', false, /\.js$/);
-const handlers = blocksHandler.keys().reduce(
-  (acc, fileName) => {
-    const isDebugHandler = fileName.includes('.debug');
-    const name = toCamelCase(
-      fileName.replace(/^\.\/handler-|\.debug|\.js/g, '')
-    );
+const handlers = blocksHandler.keys().reduce((acc, key) => {
+  const name = key.replace(/^\.\/handler-|\.js/g, '');
 
 
-    const blockKey = toCamelCase(name);
-    const handler = blocksHandler(fileName).default;
+  acc[toCamelCase(name)] = blocksHandler(key).default;
 
 
-    if (isDebugHandler) {
-      acc.debug[blockKey] = handler;
-    } else {
-      acc[blockKey] = handler;
-    }
-
-    return acc;
-  },
-  { debug: {} }
-);
+  return acc;
+}, {});
 
 
 export default handlers;
 export default handlers;

+ 4 - 7
src/background/workflow-engine/blocks-handler/handler-interaction-block.js

@@ -35,6 +35,7 @@ async function interactionHandler(block, { refData }) {
     refData,
     refData,
     debugMode,
     debugMode,
     executedBlockOnWeb,
     executedBlockOnWeb,
+    activeTabId: this.activeTab.id,
     frameSelector: this.frameSelector,
     frameSelector: this.frameSelector,
   };
   };
 
 
@@ -80,13 +81,7 @@ async function interactionHandler(block, { refData }) {
       }
       }
     }
     }
 
 
-    const isJavascriptBlock = block.name === 'javascript-code';
-
-    if (block.data.assignVariable && !isJavascriptBlock) {
-      this.referenceData.variables[block.data.variableName] = data;
-    }
-
-    if (isJavascriptBlock) {
+    if (block.name === 'javascript-code') {
       if (data?.variables) {
       if (data?.variables) {
         Object.keys(data.variables).forEach((varName) => {
         Object.keys(data.variables).forEach((varName) => {
           this.referenceData.variables[varName] = data.variables[varName];
           this.referenceData.variables[varName] = data.variables[varName];
@@ -99,6 +94,8 @@ async function interactionHandler(block, { refData }) {
           : [data.columns.data];
           : [data.columns.data];
         this.addDataToColumn(arrData);
         this.addDataToColumn(arrData);
       }
       }
+    } else if (block.data.assignVariable) {
+      this.referenceData.variables[block.data.variableName] = data;
     }
     }
 
 
     return {
     return {

+ 1 - 1
src/background/workflow-engine/blocks-handler/handler-new-tab.js

@@ -40,7 +40,7 @@ async function newTab(block) {
 
 
     this.activeTab.url = url;
     this.activeTab.url = url;
     if (tab) {
     if (tab) {
-      if (this.isDebugMode) {
+      if (this.workflow.settings.debugMode) {
         await attachDebugger(tab.id, this.activeTab.id);
         await attachDebugger(tab.id, this.activeTab.id);
       }
       }
 
 

+ 5 - 1
src/background/workflow-engine/blocks-handler/handler-switch-tab.js

@@ -1,5 +1,5 @@
 import browser from 'webextension-polyfill';
 import browser from 'webextension-polyfill';
-import { getBlockConnection } from '../helper';
+import { getBlockConnection, attachDebugger } from '../helper';
 
 
 export default async function ({ data, outputs }) {
 export default async function ({ data, outputs }) {
   const nextBlockId = getBlockConnection({ outputs });
   const nextBlockId = getBlockConnection({ outputs });
@@ -32,6 +32,10 @@ export default async function ({ data, outputs }) {
     await browser.tabs.update(tab.id, { active: true });
     await browser.tabs.update(tab.id, { active: true });
   }
   }
 
 
+  if (this.workflow.settings.debugMode) {
+    await attachDebugger(tab.id, this.activeTab.id);
+  }
+
   this.activeTab.id = tab.id;
   this.activeTab.id = tab.id;
   this.activeTab.frameId = 0;
   this.activeTab.frameId = 0;
   this.activeTab.url = tab.url;
   this.activeTab.url = tab.url;

+ 18 - 14
src/background/workflow-engine/engine.js

@@ -2,7 +2,13 @@ import browser from 'webextension-polyfill';
 import { nanoid } from 'nanoid';
 import { nanoid } from 'nanoid';
 import { tasks } from '@/utils/shared';
 import { tasks } from '@/utils/shared';
 import { convertData, waitTabLoaded } from './helper';
 import { convertData, waitTabLoaded } from './helper';
-import { toCamelCase, parseJSON, isObject, objectHasKey } from '@/utils/helper';
+import {
+  toCamelCase,
+  sleep,
+  parseJSON,
+  isObject,
+  objectHasKey,
+} from '@/utils/helper';
 import referenceData from '@/utils/reference-data';
 import referenceData from '@/utils/reference-data';
 import executeContentScript from './execute-content-script';
 import executeContentScript from './execute-content-script';
 
 
@@ -18,7 +24,6 @@ class WorkflowEngine {
     this.blocksHandler = blocksHandler;
     this.blocksHandler = blocksHandler;
     this.parentWorkflow = parentWorkflow;
     this.parentWorkflow = parentWorkflow;
     this.saveLog = workflow.settings?.saveLog ?? true;
     this.saveLog = workflow.settings?.saveLog ?? true;
-    this.isDebugMode = workflow.settings?.debugMode ?? false;
 
 
     this.loopList = {};
     this.loopList = {};
     this.repeatedTasks = {};
     this.repeatedTasks = {};
@@ -234,6 +239,11 @@ class WorkflowEngine {
     try {
     try {
       if (this.isDestroyed) return;
       if (this.isDestroyed) return;
       if (this.isUsingProxy) chrome.proxy.settings.clear({});
       if (this.isUsingProxy) chrome.proxy.settings.clear({});
+      if (this.workflow.settings.debugMode && this.activeTab.id) {
+        await sleep(1000);
+
+        chrome.debugger.detach({ tabId: this.activeTab.id });
+      }
 
 
       const endedTimestamp = Date.now();
       const endedTimestamp = Date.now();
       this.executeQueue();
       this.executeQueue();
@@ -293,18 +303,12 @@ class WorkflowEngine {
     this.dispatchEvent('update', { state: this.state });
     this.dispatchEvent('update', { state: this.state });
 
 
     const startExecutedTime = Date.now();
     const startExecutedTime = Date.now();
-    const blockName = toCamelCase(block.name);
-    const isInteractionBlock = tasks[block.name].category === 'interaction';
 
 
-    const blockHandler = this.blocksHandler[blockName];
-    const debugBlockHandler = this.blocksHandler.debug[blockName];
-    let handler = blockHandler;
-
-    if (this.isDebugMode && debugBlockHandler) {
-      handler = debugBlockHandler;
-    } else if (!blockHandler && isInteractionBlock) {
-      handler = this.blocksHandler.interactionBlock;
-    }
+    const blockHandler = this.blocksHandler[toCamelCase(block.name)];
+    const handler =
+      !blockHandler && tasks[block.name].category === 'interaction'
+        ? this.blocksHandler.interactionBlock
+        : blockHandler;
 
 
     if (!handler) {
     if (!handler) {
       console.error(`"${block.name}" block doesn't have a handler`);
       console.error(`"${block.name}" block doesn't have a handler`);
@@ -360,7 +364,7 @@ class WorkflowEngine {
         }, blockDelay);
         }, blockDelay);
       } else if (onError === 'restart-workflow' && !this.parentWorkflow) {
       } else if (onError === 'restart-workflow' && !this.parentWorkflow) {
         const restartKey = `restart-count:${this.id}`;
         const restartKey = `restart-count:${this.id}`;
-        const restartCount = parseJSON(localStorage.getItem(restartKey), 0);
+        const restartCount = +localStorage.getItem(restartKey) || 0;
         const maxRestart = this.workflow.settings.restartTimes ?? 3;
         const maxRestart = this.workflow.settings.restartTimes ?? 3;
 
 
         if (restartCount >= maxRestart) {
         if (restartCount >= maxRestart) {

+ 2 - 1
src/components/newtab/workflow/edit/EditTriggerEvent.vue

@@ -11,7 +11,7 @@
       </option>
       </option>
     </ui-select>
     </ui-select>
     <button
     <button
-      class="mb-2 block flex items-center w-full text-left mt-2 focus:ring-0"
+      class="mb-2 block flex items-center w-full text-left mt-1 focus:ring-0"
       @click="showOptions = !showOptions"
       @click="showOptions = !showOptions"
     >
     >
       <v-remixicon
       <v-remixicon
@@ -49,6 +49,7 @@
         <component
         <component
           :is="eventComponents[data.eventType]"
           :is="eventComponents[data.eventType]"
           v-if="eventComponents[data.eventType]"
           v-if="eventComponents[data.eventType]"
+          :key="data.eventName"
           :params="params"
           :params="params"
           @update="updateParams({ ...params, ...$event })"
           @update="updateParams({ ...params, ...$event })"
         />
         />

+ 23 - 2
src/components/newtab/workflow/edit/TriggerEventKeyboard.vue

@@ -8,9 +8,20 @@
       {{ item }}
       {{ item }}
     </ui-checkbox>
     </ui-checkbox>
   </div>
   </div>
-  <ui-input v-model="defaultParams.code" class="w-full mt-2" label="code" />
+  <ui-input
+    v-model="defaultParams.key"
+    class="w-full mt-2"
+    label="key"
+    placeholder="a"
+    @change="findKeyDefintion"
+  />
   <div class="flex items-center mt-1 space-x-2">
   <div class="flex items-center mt-1 space-x-2">
-    <ui-input v-model="defaultParams.key" class="flex-1" label="key" />
+    <ui-input
+      v-model="defaultParams.code"
+      class="flex-1"
+      label="code"
+      placeholder="KeyA"
+    />
     <ui-input
     <ui-input
       v-model.number="defaultParams.keyCode"
       v-model.number="defaultParams.keyCode"
       type="number"
       type="number"
@@ -25,6 +36,7 @@
 <script setup>
 <script setup>
 import { shallowReactive, watch, onMounted } from 'vue';
 import { shallowReactive, watch, onMounted } from 'vue';
 import { objectHasKey } from '@/utils/helper';
 import { objectHasKey } from '@/utils/helper';
+import { keyDefinitions } from '@/utils/us-keyboard-layout';
 
 
 const props = defineProps({
 const props = defineProps({
   params: {
   params: {
@@ -45,6 +57,15 @@ const defaultParams = shallowReactive({
   repeat: false,
   repeat: false,
 });
 });
 
 
+function findKeyDefintion(value) {
+  const keyDefinition = keyDefinitions[value];
+
+  if (!keyDefinition) return;
+
+  defaultParams.code = keyDefinitions[value].code;
+  defaultParams.keyCode = keyDefinitions[value].keyCode;
+}
+
 watch(
 watch(
   defaultParams,
   defaultParams,
   (value) => {
   (value) => {

+ 29 - 3
src/content/blocks-handler/handler-event-click.js

@@ -1,9 +1,37 @@
 import handleSelector from '../handle-selector';
 import handleSelector from '../handle-selector';
+import { sendMessage } from '@/utils/message';
 
 
 function eventClick(block) {
 function eventClick(block) {
   return new Promise((resolve, reject) => {
   return new Promise((resolve, reject) => {
     handleSelector(block, {
     handleSelector(block, {
-      onSelected(element) {
+      async onSelected(element) {
+        if (block.debugMode) {
+          const { width, height, x, y } = element.getBoundingClientRect();
+          const payload = {
+            tabId: block.activeTabId,
+            method: 'Input.dispatchMouseEvent',
+            params: {
+              x: x + width / 2,
+              y: y + height / 2,
+              button: 'left',
+            },
+          };
+          const executeCommand = (type) => {
+            payload.params.type = type;
+
+            if (type === 'mousePressed') {
+              payload.params.clickCount = 1;
+            }
+
+            return sendMessage('debugger:send-command', payload, 'background');
+          };
+
+          await executeCommand('mousePressed');
+          await executeCommand('mouseReleased');
+
+          return;
+        }
+
         if (element.click) {
         if (element.click) {
           element.click();
           element.click();
         } else {
         } else {
@@ -17,8 +45,6 @@ function eventClick(block) {
         resolve('');
         resolve('');
       },
       },
     });
     });
-
-    resolve('');
   });
   });
 }
 }
 
 

+ 105 - 14
src/content/blocks-handler/handler-trigger-event.js

@@ -1,22 +1,113 @@
 import handleSelector from '../handle-selector';
 import handleSelector from '../handle-selector';
+import { sendMessage } from '@/utils/message';
 import simulateEvent from '@/utils/simulate-event';
 import simulateEvent from '@/utils/simulate-event';
+import simulateMouseEvent from '@/utils/simulate-event/mouse-event';
 
 
-function triggerEvent(block) {
-  return new Promise((resolve, reject) => {
-    const { data } = block;
-
-    handleSelector(block, {
-      onSelected(element) {
-        simulateEvent(element, data.eventName, data.eventParams);
-      },
-      onSuccess() {
-        resolve(data.eventName);
-      },
-      onError(error) {
-        reject(error);
-      },
+const modifiers = {
+  altKey: 1,
+  ctrlKey: 2,
+  metKey: 3,
+  shiftKey: 4,
+};
+const eventHandlers = {
+  'mouse-event': async ({ params, sendCommand, name }) => {
+    const mouseButtons = {
+      0: { id: 1, name: 'left' },
+      1: { id: 4, name: 'middle' },
+      2: { id: 2, name: 'right' },
+    };
+    const commandParams = {
+      button: mouseButtons[params.button]?.name || 'left',
+    };
+
+    if (params.clientX) commandParams.x = params.clientX;
+    if (params.clientY) commandParams.y = params.clientY;
+
+    Object.keys(modifiers).forEach((key) => {
+      if (commandParams.modifiers) return;
+      if (params[key]) commandParams.modifiers = modifiers[key];
+    });
+
+    const mouseEvents = simulateMouseEvent({ sendCommand, commandParams });
+    const eventHandler = {
+      mouseover: 'mouseenter',
+      mouseout: 'mouseleave',
+    };
+    const eventName = eventHandler[name] || name;
+
+    await mouseEvents[eventName]();
+  },
+  'keyboard-event': async ({ name, params, sendCommand }) => {
+    const commandParams = {
+      key: params.key ?? '',
+      code: params.code ?? '',
+      autoRepeat: params.repeat,
+      windowsVirtualKeyCode: params.keyCode ?? 0,
+      type: name === 'keyup' ? 'keyUp' : 'keyDown',
+    };
+
+    Object.keys(modifiers).forEach((key) => {
+      if (commandParams.modifiers) return;
+      if (params[key]) commandParams.modifiers = modifiers[key];
     });
     });
 
 
+    await sendCommand('Input.dispatchKeyEvent', commandParams);
+  },
+};
+
+function triggerEvent({ data, id, frameSelector, debugMode, activeTabId }) {
+  return new Promise((resolve, reject) => {
+    handleSelector(
+      { data, id, frameSelector },
+      {
+        async onSelected(element) {
+          const eventHandler = eventHandlers[data.eventType];
+
+          if (debugMode && eventHandler) {
+            const { x, y, width, height } = element.getBoundingClientRect();
+            const elCoordinate = {
+              x: x + width / 2,
+              y: y + height / 2,
+            };
+            const sendCommand = (method, params = {}) => {
+              const payload = {
+                method,
+                params: {
+                  x: elCoordinate.x,
+                  y: elCoordinate.y,
+                  ...params,
+                },
+                tabId: activeTabId,
+              };
+
+              return sendMessage(
+                'debugger:send-command',
+                payload,
+                'background'
+              );
+            };
+
+            await eventHandler({
+              element,
+              sendCommand,
+              name: data.eventName,
+              params: data.eventParams,
+            });
+
+            return;
+          }
+
+          simulateEvent(element, data.eventName, data.eventParams);
+        },
+        onSuccess() {
+          resolve(data.eventName);
+        },
+        onError(error) {
+          reject(error);
+        },
+      }
+    );
+
     resolve(data.eventName);
     resolve(data.eventName);
   });
   });
 }
 }

+ 7 - 5
src/content/handle-selector.js

@@ -91,13 +91,15 @@ export default async function (
     }
     }
 
 
     if (data.multiple && selectorType === 'cssSelector') {
     if (data.multiple && selectorType === 'cssSelector') {
-      element.forEach((el) => {
-        markElement(el, { id, data });
-        onSelected(el);
-      });
+      await Promise.allSettled(
+        element.map((el) => {
+          markElement(el, { id, data });
+          return onSelected(el);
+        })
+      );
     } else if (element) {
     } else if (element) {
       markElement(element, { id, data });
       markElement(element, { id, data });
-      onSelected(element);
+      await onSelected(element);
     }
     }
 
 
     if (onSuccess) onSuccess();
     if (onSuccess) onSuccess();

+ 11 - 5
src/utils/handle-form-element.js

@@ -1,22 +1,28 @@
+import { keyDefinitions } from '@/utils/us-keyboard-layout';
 import simulateEvent from './simulate-event';
 import simulateEvent from './simulate-event';
 
 
 function formEvent(element, data) {
 function formEvent(element, data) {
   if (data.type === 'text-field') {
   if (data.type === 'text-field') {
-    const code = /\s/.test(data.value)
-      ? 'Space'
-      : `key${data.value.toUpperCase()}`;
+    const currentKey = /\s/.test(data.value) ? 'Space' : data.value;
+    const { key, keyCode, code } = keyDefinitions[currentKey] || {
+      key: currentKey,
+      keyCode: 0,
+      code: `Key${currentKey}`,
+    };
 
 
     simulateEvent(element, 'keydown', {
     simulateEvent(element, 'keydown', {
+      key,
       code,
       code,
+      keyCode,
       bubbles: true,
       bubbles: true,
       cancelable: true,
       cancelable: true,
-      key: data.value,
     });
     });
     simulateEvent(element, 'keyup', {
     simulateEvent(element, 'keyup', {
+      key,
       code,
       code,
+      keyCode,
       bubbles: true,
       bubbles: true,
       cancelable: true,
       cancelable: true,
-      key: data.value,
     });
     });
   }
   }
 
 

+ 4 - 0
src/utils/helper.js

@@ -1,3 +1,7 @@
+export function sleep(timeout = 500) {
+  return new Promise((resolve) => setTimeout(resolve, timeout));
+}
+
 export function findTriggerBlock(drawflow = {}) {
 export function findTriggerBlock(drawflow = {}) {
   if (!drawflow) return null;
   if (!drawflow) return null;
 
 

+ 1 - 1
src/utils/shared.js

@@ -495,7 +495,7 @@ export const tasks = {
       findBy: 'cssSelector',
       findBy: 'cssSelector',
       waitForSelector: false,
       waitForSelector: false,
       waitSelectorTimeout: 5000,
       waitSelectorTimeout: 5000,
-      selector: '',
+      selector: 'html',
       markEl: false,
       markEl: false,
       multiple: false,
       multiple: false,
       eventName: '',
       eventName: '',

+ 1 - 1
src/utils/simulate-event.js → src/utils/simulate-event/index.js

@@ -1,4 +1,4 @@
-import { eventList } from './shared';
+import { eventList } from '../shared';
 
 
 export function getEventObj(name, params) {
 export function getEventObj(name, params) {
   const eventType = eventList.find(({ id }) => id === name)?.type ?? '';
   const eventType = eventList.find(({ id }) => id === name)?.type ?? '';

+ 44 - 0
src/utils/simulate-event/mouse-event.js

@@ -0,0 +1,44 @@
+export default function ({ sendCommand, commandParams }) {
+  async function mousedown() {
+    commandParams.type = 'mousePressed';
+    await sendCommand('Input.dispatchMouseEvent', commandParams);
+  }
+  async function mouseup() {
+    commandParams.type = 'mouseReleased';
+    await sendCommand('Input.dispatchMouseEvent', commandParams);
+  }
+  async function click() {
+    if (!commandParams.clickCount) commandParams.clickCount = 1;
+
+    await mousedown();
+    await mouseup();
+  }
+  async function dblclick() {
+    commandParams.clickCount = 2;
+    await click();
+  }
+  async function mousemove() {
+    commandParams.type = 'mouseMoved';
+    await sendCommand('Input.dispatchMouseEvent', commandParams);
+  }
+  async function mouseenter() {
+    await mousemove();
+  }
+  async function mouseleave() {
+    await mousemove();
+
+    commandParams.x = -100;
+    commandParams.y = -100;
+    await mousemove();
+  }
+
+  return {
+    mousedown,
+    mouseup,
+    click,
+    dblclick,
+    mousemove,
+    mouseenter,
+    mouseleave,
+  };
+}

+ 409 - 0
src/utils/us-keyboard-layout.js

@@ -0,0 +1,409 @@
+/*
+  Forked from:
+  https://github.com/puppeteer/puppeteer/blob/e11fe713407c2430526ae616d0adfb5dd278b5de/src/common/USKeyboardLayout.ts
+*/
+
+/**
+ * Copyright 2017 Google Inc. All rights reserved.
+ *
+ * Licensed under the Apache License, Version 2.0 (the 'License');
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an 'AS IS' BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/**
+ * @internal
+ */
+export const keyDefinitions = {
+  0: { keyCode: 48, key: '0', code: 'Digit0' },
+  1: { keyCode: 49, key: '1', code: 'Digit1' },
+  2: { keyCode: 50, key: '2', code: 'Digit2' },
+  3: { keyCode: 51, key: '3', code: 'Digit3' },
+  4: { keyCode: 52, key: '4', code: 'Digit4' },
+  5: { keyCode: 53, key: '5', code: 'Digit5' },
+  6: { keyCode: 54, key: '6', code: 'Digit6' },
+  7: { keyCode: 55, key: '7', code: 'Digit7' },
+  8: { keyCode: 56, key: '8', code: 'Digit8' },
+  9: { keyCode: 57, key: '9', code: 'Digit9' },
+  Power: { key: 'Power', code: 'Power' },
+  Eject: { key: 'Eject', code: 'Eject' },
+  Abort: { keyCode: 3, code: 'Abort', key: 'Cancel' },
+  Help: { keyCode: 6, code: 'Help', key: 'Help' },
+  Backspace: { keyCode: 8, code: 'Backspace', key: 'Backspace' },
+  Tab: { keyCode: 9, code: 'Tab', key: 'Tab' },
+  Numpad5: {
+    keyCode: 12,
+    shiftKeyCode: 101,
+    key: 'Clear',
+    code: 'Numpad5',
+    shiftKey: '5',
+    location: 3,
+  },
+  NumpadEnter: {
+    keyCode: 13,
+    code: 'NumpadEnter',
+    key: 'Enter',
+    text: '\r',
+    location: 3,
+  },
+  Enter: { keyCode: 13, code: 'Enter', key: 'Enter', text: '\r' },
+  '\r': { keyCode: 13, code: 'Enter', key: 'Enter', text: '\r' },
+  '\n': { keyCode: 13, code: 'Enter', key: 'Enter', text: '\r' },
+  ShiftLeft: { keyCode: 16, code: 'ShiftLeft', key: 'Shift', location: 1 },
+  ShiftRight: { keyCode: 16, code: 'ShiftRight', key: 'Shift', location: 2 },
+  ControlLeft: {
+    keyCode: 17,
+    code: 'ControlLeft',
+    key: 'Control',
+    location: 1,
+  },
+  ControlRight: {
+    keyCode: 17,
+    code: 'ControlRight',
+    key: 'Control',
+    location: 2,
+  },
+  AltLeft: { keyCode: 18, code: 'AltLeft', key: 'Alt', location: 1 },
+  AltRight: { keyCode: 18, code: 'AltRight', key: 'Alt', location: 2 },
+  Pause: { keyCode: 19, code: 'Pause', key: 'Pause' },
+  CapsLock: { keyCode: 20, code: 'CapsLock', key: 'CapsLock' },
+  Escape: { keyCode: 27, code: 'Escape', key: 'Escape' },
+  Convert: { keyCode: 28, code: 'Convert', key: 'Convert' },
+  NonConvert: { keyCode: 29, code: 'NonConvert', key: 'NonConvert' },
+  Space: { keyCode: 32, code: 'Space', key: ' ' },
+  Numpad9: {
+    keyCode: 33,
+    shiftKeyCode: 105,
+    key: 'PageUp',
+    code: 'Numpad9',
+    shiftKey: '9',
+    location: 3,
+  },
+  PageUp: { keyCode: 33, code: 'PageUp', key: 'PageUp' },
+  Numpad3: {
+    keyCode: 34,
+    shiftKeyCode: 99,
+    key: 'PageDown',
+    code: 'Numpad3',
+    shiftKey: '3',
+    location: 3,
+  },
+  PageDown: { keyCode: 34, code: 'PageDown', key: 'PageDown' },
+  End: { keyCode: 35, code: 'End', key: 'End' },
+  Numpad1: {
+    keyCode: 35,
+    shiftKeyCode: 97,
+    key: 'End',
+    code: 'Numpad1',
+    shiftKey: '1',
+    location: 3,
+  },
+  Home: { keyCode: 36, code: 'Home', key: 'Home' },
+  Numpad7: {
+    keyCode: 36,
+    shiftKeyCode: 103,
+    key: 'Home',
+    code: 'Numpad7',
+    shiftKey: '7',
+    location: 3,
+  },
+  ArrowLeft: { keyCode: 37, code: 'ArrowLeft', key: 'ArrowLeft' },
+  Numpad4: {
+    keyCode: 37,
+    shiftKeyCode: 100,
+    key: 'ArrowLeft',
+    code: 'Numpad4',
+    shiftKey: '4',
+    location: 3,
+  },
+  Numpad8: {
+    keyCode: 38,
+    shiftKeyCode: 104,
+    key: 'ArrowUp',
+    code: 'Numpad8',
+    shiftKey: '8',
+    location: 3,
+  },
+  ArrowUp: { keyCode: 38, code: 'ArrowUp', key: 'ArrowUp' },
+  ArrowRight: { keyCode: 39, code: 'ArrowRight', key: 'ArrowRight' },
+  Numpad6: {
+    keyCode: 39,
+    shiftKeyCode: 102,
+    key: 'ArrowRight',
+    code: 'Numpad6',
+    shiftKey: '6',
+    location: 3,
+  },
+  Numpad2: {
+    keyCode: 40,
+    shiftKeyCode: 98,
+    key: 'ArrowDown',
+    code: 'Numpad2',
+    shiftKey: '2',
+    location: 3,
+  },
+  ArrowDown: { keyCode: 40, code: 'ArrowDown', key: 'ArrowDown' },
+  Select: { keyCode: 41, code: 'Select', key: 'Select' },
+  Open: { keyCode: 43, code: 'Open', key: 'Execute' },
+  PrintScreen: { keyCode: 44, code: 'PrintScreen', key: 'PrintScreen' },
+  Insert: { keyCode: 45, code: 'Insert', key: 'Insert' },
+  Numpad0: {
+    keyCode: 45,
+    shiftKeyCode: 96,
+    key: 'Insert',
+    code: 'Numpad0',
+    shiftKey: '0',
+    location: 3,
+  },
+  Delete: { keyCode: 46, code: 'Delete', key: 'Delete' },
+  NumpadDecimal: {
+    keyCode: 46,
+    shiftKeyCode: 110,
+    code: 'NumpadDecimal',
+    key: '\u0000',
+    shiftKey: '.',
+    location: 3,
+  },
+  Digit0: { keyCode: 48, code: 'Digit0', shiftKey: ')', key: '0' },
+  Digit1: { keyCode: 49, code: 'Digit1', shiftKey: '!', key: '1' },
+  Digit2: { keyCode: 50, code: 'Digit2', shiftKey: '@', key: '2' },
+  Digit3: { keyCode: 51, code: 'Digit3', shiftKey: '#', key: '3' },
+  Digit4: { keyCode: 52, code: 'Digit4', shiftKey: '$', key: '4' },
+  Digit5: { keyCode: 53, code: 'Digit5', shiftKey: '%', key: '5' },
+  Digit6: { keyCode: 54, code: 'Digit6', shiftKey: '^', key: '6' },
+  Digit7: { keyCode: 55, code: 'Digit7', shiftKey: '&', key: '7' },
+  Digit8: { keyCode: 56, code: 'Digit8', shiftKey: '*', key: '8' },
+  Digit9: { keyCode: 57, code: 'Digit9', shiftKey: '(', key: '9' },
+  KeyA: { keyCode: 65, code: 'KeyA', shiftKey: 'A', key: 'a' },
+  KeyB: { keyCode: 66, code: 'KeyB', shiftKey: 'B', key: 'b' },
+  KeyC: { keyCode: 67, code: 'KeyC', shiftKey: 'C', key: 'c' },
+  KeyD: { keyCode: 68, code: 'KeyD', shiftKey: 'D', key: 'd' },
+  KeyE: { keyCode: 69, code: 'KeyE', shiftKey: 'E', key: 'e' },
+  KeyF: { keyCode: 70, code: 'KeyF', shiftKey: 'F', key: 'f' },
+  KeyG: { keyCode: 71, code: 'KeyG', shiftKey: 'G', key: 'g' },
+  KeyH: { keyCode: 72, code: 'KeyH', shiftKey: 'H', key: 'h' },
+  KeyI: { keyCode: 73, code: 'KeyI', shiftKey: 'I', key: 'i' },
+  KeyJ: { keyCode: 74, code: 'KeyJ', shiftKey: 'J', key: 'j' },
+  KeyK: { keyCode: 75, code: 'KeyK', shiftKey: 'K', key: 'k' },
+  KeyL: { keyCode: 76, code: 'KeyL', shiftKey: 'L', key: 'l' },
+  KeyM: { keyCode: 77, code: 'KeyM', shiftKey: 'M', key: 'm' },
+  KeyN: { keyCode: 78, code: 'KeyN', shiftKey: 'N', key: 'n' },
+  KeyO: { keyCode: 79, code: 'KeyO', shiftKey: 'O', key: 'o' },
+  KeyP: { keyCode: 80, code: 'KeyP', shiftKey: 'P', key: 'p' },
+  KeyQ: { keyCode: 81, code: 'KeyQ', shiftKey: 'Q', key: 'q' },
+  KeyR: { keyCode: 82, code: 'KeyR', shiftKey: 'R', key: 'r' },
+  KeyS: { keyCode: 83, code: 'KeyS', shiftKey: 'S', key: 's' },
+  KeyT: { keyCode: 84, code: 'KeyT', shiftKey: 'T', key: 't' },
+  KeyU: { keyCode: 85, code: 'KeyU', shiftKey: 'U', key: 'u' },
+  KeyV: { keyCode: 86, code: 'KeyV', shiftKey: 'V', key: 'v' },
+  KeyW: { keyCode: 87, code: 'KeyW', shiftKey: 'W', key: 'w' },
+  KeyX: { keyCode: 88, code: 'KeyX', shiftKey: 'X', key: 'x' },
+  KeyY: { keyCode: 89, code: 'KeyY', shiftKey: 'Y', key: 'y' },
+  KeyZ: { keyCode: 90, code: 'KeyZ', shiftKey: 'Z', key: 'z' },
+  MetaLeft: { keyCode: 91, code: 'MetaLeft', key: 'Meta', location: 1 },
+  MetaRight: { keyCode: 92, code: 'MetaRight', key: 'Meta', location: 2 },
+  ContextMenu: { keyCode: 93, code: 'ContextMenu', key: 'ContextMenu' },
+  NumpadMultiply: {
+    keyCode: 106,
+    code: 'NumpadMultiply',
+    key: '*',
+    location: 3,
+  },
+  NumpadAdd: { keyCode: 107, code: 'NumpadAdd', key: '+', location: 3 },
+  NumpadSubtract: {
+    keyCode: 109,
+    code: 'NumpadSubtract',
+    key: '-',
+    location: 3,
+  },
+  NumpadDivide: { keyCode: 111, code: 'NumpadDivide', key: '/', location: 3 },
+  F1: { keyCode: 112, code: 'F1', key: 'F1' },
+  F2: { keyCode: 113, code: 'F2', key: 'F2' },
+  F3: { keyCode: 114, code: 'F3', key: 'F3' },
+  F4: { keyCode: 115, code: 'F4', key: 'F4' },
+  F5: { keyCode: 116, code: 'F5', key: 'F5' },
+  F6: { keyCode: 117, code: 'F6', key: 'F6' },
+  F7: { keyCode: 118, code: 'F7', key: 'F7' },
+  F8: { keyCode: 119, code: 'F8', key: 'F8' },
+  F9: { keyCode: 120, code: 'F9', key: 'F9' },
+  F10: { keyCode: 121, code: 'F10', key: 'F10' },
+  F11: { keyCode: 122, code: 'F11', key: 'F11' },
+  F12: { keyCode: 123, code: 'F12', key: 'F12' },
+  F13: { keyCode: 124, code: 'F13', key: 'F13' },
+  F14: { keyCode: 125, code: 'F14', key: 'F14' },
+  F15: { keyCode: 126, code: 'F15', key: 'F15' },
+  F16: { keyCode: 127, code: 'F16', key: 'F16' },
+  F17: { keyCode: 128, code: 'F17', key: 'F17' },
+  F18: { keyCode: 129, code: 'F18', key: 'F18' },
+  F19: { keyCode: 130, code: 'F19', key: 'F19' },
+  F20: { keyCode: 131, code: 'F20', key: 'F20' },
+  F21: { keyCode: 132, code: 'F21', key: 'F21' },
+  F22: { keyCode: 133, code: 'F22', key: 'F22' },
+  F23: { keyCode: 134, code: 'F23', key: 'F23' },
+  F24: { keyCode: 135, code: 'F24', key: 'F24' },
+  NumLock: { keyCode: 144, code: 'NumLock', key: 'NumLock' },
+  ScrollLock: { keyCode: 145, code: 'ScrollLock', key: 'ScrollLock' },
+  AudioVolumeMute: {
+    keyCode: 173,
+    code: 'AudioVolumeMute',
+    key: 'AudioVolumeMute',
+  },
+  AudioVolumeDown: {
+    keyCode: 174,
+    code: 'AudioVolumeDown',
+    key: 'AudioVolumeDown',
+  },
+  AudioVolumeUp: { keyCode: 175, code: 'AudioVolumeUp', key: 'AudioVolumeUp' },
+  MediaTrackNext: {
+    keyCode: 176,
+    code: 'MediaTrackNext',
+    key: 'MediaTrackNext',
+  },
+  MediaTrackPrevious: {
+    keyCode: 177,
+    code: 'MediaTrackPrevious',
+    key: 'MediaTrackPrevious',
+  },
+  MediaStop: { keyCode: 178, code: 'MediaStop', key: 'MediaStop' },
+  MediaPlayPause: {
+    keyCode: 179,
+    code: 'MediaPlayPause',
+    key: 'MediaPlayPause',
+  },
+  Semicolon: { keyCode: 186, code: 'Semicolon', shiftKey: ':', key: ';' },
+  Equal: { keyCode: 187, code: 'Equal', shiftKey: '+', key: '=' },
+  NumpadEqual: { keyCode: 187, code: 'NumpadEqual', key: '=', location: 3 },
+  Comma: { keyCode: 188, code: 'Comma', shiftKey: '<', key: ',' },
+  Minus: { keyCode: 189, code: 'Minus', shiftKey: '_', key: '-' },
+  Period: { keyCode: 190, code: 'Period', shiftKey: '>', key: '.' },
+  Slash: { keyCode: 191, code: 'Slash', shiftKey: '?', key: '/' },
+  Backquote: { keyCode: 192, code: 'Backquote', shiftKey: '~', key: '`' },
+  BracketLeft: { keyCode: 219, code: 'BracketLeft', shiftKey: '{', key: '[' },
+  Backslash: { keyCode: 220, code: 'Backslash', shiftKey: '|', key: '\\' },
+  BracketRight: { keyCode: 221, code: 'BracketRight', shiftKey: '}', key: ']' },
+  Quote: { keyCode: 222, code: 'Quote', shiftKey: '"', key: "'" },
+  AltGraph: { keyCode: 225, code: 'AltGraph', key: 'AltGraph' },
+  Props: { keyCode: 247, code: 'Props', key: 'CrSel' },
+  Cancel: { keyCode: 3, key: 'Cancel', code: 'Abort' },
+  Clear: { keyCode: 12, key: 'Clear', code: 'Numpad5', location: 3 },
+  Shift: { keyCode: 16, key: 'Shift', code: 'ShiftLeft', location: 1 },
+  Control: { keyCode: 17, key: 'Control', code: 'ControlLeft', location: 1 },
+  Alt: { keyCode: 18, key: 'Alt', code: 'AltLeft', location: 1 },
+  Accept: { keyCode: 30, key: 'Accept' },
+  ModeChange: { keyCode: 31, key: 'ModeChange' },
+  ' ': { keyCode: 32, key: ' ', code: 'Space' },
+  Print: { keyCode: 42, key: 'Print' },
+  Execute: { keyCode: 43, key: 'Execute', code: 'Open' },
+  '\u0000': { keyCode: 46, key: '\u0000', code: 'NumpadDecimal', location: 3 },
+  a: { keyCode: 65, key: 'a', code: 'KeyA' },
+  b: { keyCode: 66, key: 'b', code: 'KeyB' },
+  c: { keyCode: 67, key: 'c', code: 'KeyC' },
+  d: { keyCode: 68, key: 'd', code: 'KeyD' },
+  e: { keyCode: 69, key: 'e', code: 'KeyE' },
+  f: { keyCode: 70, key: 'f', code: 'KeyF' },
+  g: { keyCode: 71, key: 'g', code: 'KeyG' },
+  h: { keyCode: 72, key: 'h', code: 'KeyH' },
+  i: { keyCode: 73, key: 'i', code: 'KeyI' },
+  j: { keyCode: 74, key: 'j', code: 'KeyJ' },
+  k: { keyCode: 75, key: 'k', code: 'KeyK' },
+  l: { keyCode: 76, key: 'l', code: 'KeyL' },
+  m: { keyCode: 77, key: 'm', code: 'KeyM' },
+  n: { keyCode: 78, key: 'n', code: 'KeyN' },
+  o: { keyCode: 79, key: 'o', code: 'KeyO' },
+  p: { keyCode: 80, key: 'p', code: 'KeyP' },
+  q: { keyCode: 81, key: 'q', code: 'KeyQ' },
+  r: { keyCode: 82, key: 'r', code: 'KeyR' },
+  s: { keyCode: 83, key: 's', code: 'KeyS' },
+  t: { keyCode: 84, key: 't', code: 'KeyT' },
+  u: { keyCode: 85, key: 'u', code: 'KeyU' },
+  v: { keyCode: 86, key: 'v', code: 'KeyV' },
+  w: { keyCode: 87, key: 'w', code: 'KeyW' },
+  x: { keyCode: 88, key: 'x', code: 'KeyX' },
+  y: { keyCode: 89, key: 'y', code: 'KeyY' },
+  z: { keyCode: 90, key: 'z', code: 'KeyZ' },
+  Meta: { keyCode: 91, key: 'Meta', code: 'MetaLeft', location: 1 },
+  '*': { keyCode: 106, key: '*', code: 'NumpadMultiply', location: 3 },
+  '+': { keyCode: 107, key: '+', code: 'NumpadAdd', location: 3 },
+  '-': { keyCode: 109, key: '-', code: 'NumpadSubtract', location: 3 },
+  '/': { keyCode: 111, key: '/', code: 'NumpadDivide', location: 3 },
+  ';': { keyCode: 186, key: ';', code: 'Semicolon' },
+  '=': { keyCode: 187, key: '=', code: 'Equal' },
+  ',': { keyCode: 188, key: ',', code: 'Comma' },
+  '.': { keyCode: 190, key: '.', code: 'Period' },
+  '`': { keyCode: 192, key: '`', code: 'Backquote' },
+  '[': { keyCode: 219, key: '[', code: 'BracketLeft' },
+  '\\': { keyCode: 220, key: '\\', code: 'Backslash' },
+  ']': { keyCode: 221, key: ']', code: 'BracketRight' },
+  "'": { keyCode: 222, key: "'", code: 'Quote' },
+  Attn: { keyCode: 246, key: 'Attn' },
+  CrSel: { keyCode: 247, key: 'CrSel', code: 'Props' },
+  ExSel: { keyCode: 248, key: 'ExSel' },
+  EraseEof: { keyCode: 249, key: 'EraseEof' },
+  Play: { keyCode: 250, key: 'Play' },
+  ZoomOut: { keyCode: 251, key: 'ZoomOut' },
+  ')': { keyCode: 48, key: ')', code: 'Digit0' },
+  '!': { keyCode: 49, key: '!', code: 'Digit1' },
+  '@': { keyCode: 50, key: '@', code: 'Digit2' },
+  '#': { keyCode: 51, key: '#', code: 'Digit3' },
+  $: { keyCode: 52, key: '$', code: 'Digit4' },
+  '%': { keyCode: 53, key: '%', code: 'Digit5' },
+  '^': { keyCode: 54, key: '^', code: 'Digit6' },
+  '&': { keyCode: 55, key: '&', code: 'Digit7' },
+  '(': { keyCode: 57, key: '(', code: 'Digit9' },
+  A: { keyCode: 65, key: 'A', code: 'KeyA' },
+  B: { keyCode: 66, key: 'B', code: 'KeyB' },
+  C: { keyCode: 67, key: 'C', code: 'KeyC' },
+  D: { keyCode: 68, key: 'D', code: 'KeyD' },
+  E: { keyCode: 69, key: 'E', code: 'KeyE' },
+  F: { keyCode: 70, key: 'F', code: 'KeyF' },
+  G: { keyCode: 71, key: 'G', code: 'KeyG' },
+  H: { keyCode: 72, key: 'H', code: 'KeyH' },
+  I: { keyCode: 73, key: 'I', code: 'KeyI' },
+  J: { keyCode: 74, key: 'J', code: 'KeyJ' },
+  K: { keyCode: 75, key: 'K', code: 'KeyK' },
+  L: { keyCode: 76, key: 'L', code: 'KeyL' },
+  M: { keyCode: 77, key: 'M', code: 'KeyM' },
+  N: { keyCode: 78, key: 'N', code: 'KeyN' },
+  O: { keyCode: 79, key: 'O', code: 'KeyO' },
+  P: { keyCode: 80, key: 'P', code: 'KeyP' },
+  Q: { keyCode: 81, key: 'Q', code: 'KeyQ' },
+  R: { keyCode: 82, key: 'R', code: 'KeyR' },
+  S: { keyCode: 83, key: 'S', code: 'KeyS' },
+  T: { keyCode: 84, key: 'T', code: 'KeyT' },
+  U: { keyCode: 85, key: 'U', code: 'KeyU' },
+  V: { keyCode: 86, key: 'V', code: 'KeyV' },
+  W: { keyCode: 87, key: 'W', code: 'KeyW' },
+  X: { keyCode: 88, key: 'X', code: 'KeyX' },
+  Y: { keyCode: 89, key: 'Y', code: 'KeyY' },
+  Z: { keyCode: 90, key: 'Z', code: 'KeyZ' },
+  ':': { keyCode: 186, key: ':', code: 'Semicolon' },
+  '<': { keyCode: 188, key: '<', code: 'Comma' },
+  _: { keyCode: 189, key: '_', code: 'Minus' },
+  '>': { keyCode: 190, key: '>', code: 'Period' },
+  '?': { keyCode: 191, key: '?', code: 'Slash' },
+  '~': { keyCode: 192, key: '~', code: 'Backquote' },
+  '{': { keyCode: 219, key: '{', code: 'BracketLeft' },
+  '|': { keyCode: 220, key: '|', code: 'Backslash' },
+  '}': { keyCode: 221, key: '}', code: 'BracketRight' },
+  '"': { keyCode: 222, key: '"', code: 'Quote' },
+  SoftLeft: { key: 'SoftLeft', code: 'SoftLeft', location: 4 },
+  SoftRight: { key: 'SoftRight', code: 'SoftRight', location: 4 },
+  Camera: { keyCode: 44, key: 'Camera', code: 'Camera', location: 4 },
+  Call: { key: 'Call', code: 'Call', location: 4 },
+  EndCall: { keyCode: 95, key: 'EndCall', code: 'EndCall', location: 4 },
+  VolumeDown: {
+    keyCode: 182,
+    key: 'VolumeDown',
+    code: 'VolumeDown',
+    location: 4,
+  },
+  VolumeUp: { keyCode: 183, key: 'VolumeUp', code: 'VolumeUp', location: 4 },
+};