fix
This commit is contained in:
		
							
								
								
									
										23
									
								
								term/pkg/hotkeys-js/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								term/pkg/hotkeys-js/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | MIT License | ||||||
|  |  | ||||||
|  | Copyright (c) 2015-present, Kenny Wong. | ||||||
|  |  | ||||||
|  | Copyright (c) 2011-2013 Thomas Fuchs (https://github.com/madrobby/keymaster) | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | in the Software without restriction, including without limitation the rights | ||||||
|  | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  | copies of the Software, and to permit persons to whom the Software is | ||||||
|  | furnished to do so, subject to the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in all | ||||||
|  | copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||||
|  | SOFTWARE. | ||||||
							
								
								
									
										441
									
								
								term/pkg/hotkeys-js/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										441
									
								
								term/pkg/hotkeys-js/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,441 @@ | |||||||
|  | # Hotkeys | ||||||
|  |  | ||||||
|  | <!--dividing--> | ||||||
|  |  | ||||||
|  | [](https://jaywcjlove.github.io/#/sponsor) | ||||||
|  | [](https://www.npmjs.com/package/hotkeys-js) | ||||||
|  | [](https://github.com/jaywcjlove/hotkeys/stargazers) | ||||||
|  |  | ||||||
|  | [](https://github.com/jaywcjlove/hotkeys-js/actions/workflows/ci.yml) | ||||||
|  | [](https://coveralls.io/github/jaywcjlove/hotkeys?branch=master) | ||||||
|  | [](https://github.com/jaywcjlove/hotkeys-js/blob/master/README-zh.md) | ||||||
|  | [](https://gitee.com/jaywcjlove/hotkeys) | ||||||
|  |  | ||||||
|  | HotKeys.js is an input capture library with some very special features, it is easy to pick up and use, has a reasonable footprint ([~6kB](https://bundlephobia.com/result?p=hotkeys-js)) (gzipped: **`2.8kB`**), and has no dependencies. It should not interfere with any JavaScript libraries or frameworks. Official document [demo preview](https://jaywcjlove.github.io/hotkeys-js). [More examples](https://github.com/jaywcjlove/hotkeys-js/issues?q=label%3ADemo+). | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | ╭┈┈╮          ╭┈┈╮  ╭┈┈╮ | ||||||
|  | ┆  ├┈┈..┈┈┈┈┈.┆  └┈╮┆  ├┈┈..┈┈┈┈┈..┈┈.┈┈..┈┈┈┈┈. | ||||||
|  | ┆     ┆┆  □  ┆┆   ┈┤┆    < ┆  -__┘┆  ┆  ┆┆__ ┈┈┤ | ||||||
|  | ╰┈┈┴┈┈╯╰┈┈┈┈┈╯╰┈┈┈┈╯╰┈┈┴┈┈╯╰┈┈┈┈┈╯╰┈┈┈  ┆╰┈┈┈┈┈╯ | ||||||
|  |                                   ╰┈┈┈┈┈╯ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Usage | ||||||
|  |  | ||||||
|  | You will need `Node.js` installed on your system. | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm install hotkeys-js --save | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | import hotkeys from 'hotkeys-js'; | ||||||
|  |  | ||||||
|  | hotkeys('f5', function(event, handler){ | ||||||
|  |   // Prevent the default refresh event under WINDOWS system | ||||||
|  |   event.preventDefault() | ||||||
|  |   alert('you pressed F5!') | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Or manually download and link **hotkeys.js** in your HTML, It can also be downloaded via [UNPKG](https://unpkg.com/hotkeys-js/dist/): | ||||||
|  |  | ||||||
|  | CDN: [UNPKG](https://unpkg.com/hotkeys-js/dist/) | [jsDelivr](https://cdn.jsdelivr.net/npm/hotkeys-js@3.7.3/) | [Githack](https://raw.githack.com/jaywcjlove/hotkeys/master/dist/hotkeys.min.js) | [Statically](https://cdn.statically.io/gh/jaywcjlove/hotkeys/master/dist/hotkeys.min.js) | [bundle.run](https://bundle.run/hotkeys-js@3.7.3) | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <script src="https://unpkg.com/hotkeys-js/dist/hotkeys.min.js"></script> | ||||||
|  | <script type="text/javascript"> | ||||||
|  | hotkeys('ctrl+a,ctrl+b,r,f', function (event, handler){ | ||||||
|  |   switch (handler.key) { | ||||||
|  |     case 'ctrl+a': alert('you pressed ctrl+a!'); | ||||||
|  |       break; | ||||||
|  |     case 'ctrl+b': alert('you pressed ctrl+b!'); | ||||||
|  |       break; | ||||||
|  |     case 'r': alert('you pressed r!'); | ||||||
|  |       break; | ||||||
|  |     case 'f': alert('you pressed f!'); | ||||||
|  |       break; | ||||||
|  |     default: alert(event); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Used in React | ||||||
|  |  | ||||||
|  | [react-hotkeys](https://github.com/jaywcjlove/react-hotkeys) is the React component that listen to keydown and keyup keyboard events, defining and dispatching keyboard shortcuts. Detailed use method please see its documentation [react-hotkeys](https://github.com/jaywcjlove/react-hotkeys). | ||||||
|  |  | ||||||
|  | [react-hotkeys-hook](https://github.com/JohannesKlauss/react-hotkeys-hook) - React hook for using keyboard shortcuts in components. Make sure that you have at least version 16.8 of react and react-dom installed, or otherwise hooks won't work for you. | ||||||
|  |  | ||||||
|  | ## Browser Support | ||||||
|  |  | ||||||
|  | Hotkeys.js has been tested and should work in. | ||||||
|  |  | ||||||
|  | ```shell | ||||||
|  | Internet Explorer 6+ | ||||||
|  | Safari | ||||||
|  | Firefox | ||||||
|  | Chrome | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Supported Keys | ||||||
|  |  | ||||||
|  | HotKeys understands the following modifiers: `⇧`, `shift`, `option`, `⌥`, `alt`, `ctrl`, `control`, `command`, and `⌘`. | ||||||
|  |  | ||||||
|  | The following special keys can be used for shortcuts: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown, del, delete, f1 through f19, num_0 through num_9, num_multiply, num_add, num_enter, num_subtract, num_decimal, num_divide. | ||||||
|  |  | ||||||
|  | `⌘` Command() | ||||||
|  | `⌃` Control | ||||||
|  | `⌥` Option(alt) | ||||||
|  | `⇧` Shift | ||||||
|  | `⇪` Caps Lock(Capital) | ||||||
|  | ~~`fn` Does not support fn~~ | ||||||
|  | `↩︎` return/Enter space | ||||||
|  |  | ||||||
|  | ## Defining Shortcuts | ||||||
|  |  | ||||||
|  | One global method is exposed, key which defines shortcuts when called directly. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys([keys:<String>], [option:[string|object|function]], [callback:<function>]) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('f5', function(event, handler) { | ||||||
|  |   // Prevent the default refresh event under WINDOWS system | ||||||
|  |   event.preventDefault(); | ||||||
|  |   alert('you pressed F5!'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Returning false stops the event and prevents default browser events | ||||||
|  | // Mac OS system defines `command + r` as a refresh shortcut | ||||||
|  | hotkeys('ctrl+r, command+r', function() { | ||||||
|  |   alert('stopped reload!'); | ||||||
|  |   return false; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Single key | ||||||
|  | hotkeys('a', function(event,handler){ | ||||||
|  |   //event.srcElement: input | ||||||
|  |   //event.target: input | ||||||
|  |   if(event.target === "input"){ | ||||||
|  |       alert('you pressed a!') | ||||||
|  |   } | ||||||
|  |   alert('you pressed a!') | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Key Combination | ||||||
|  | hotkeys('ctrl+a,ctrl+b,r,f', function (event, handler){ | ||||||
|  |   switch (handler.key) { | ||||||
|  |     case 'ctrl+a': alert('you pressed ctrl+a!'); | ||||||
|  |       break; | ||||||
|  |     case 'ctrl+b': alert('you pressed ctrl+b!'); | ||||||
|  |       break; | ||||||
|  |     case 'r': alert('you pressed r!'); | ||||||
|  |       break; | ||||||
|  |     case 'f': alert('you pressed f!'); | ||||||
|  |       break; | ||||||
|  |     default: alert(event); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | hotkeys('ctrl+a+s', function() { | ||||||
|  |     alert('you pressed ctrl+a+s!'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Using a scope | ||||||
|  | hotkeys('*','wcj', function(event){ | ||||||
|  |   console.log('do something', event); | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | #### option | ||||||
|  |  | ||||||
|  | - `scope<String>` | ||||||
|  | - `element<HTMLElement>` | ||||||
|  | - `keyup<Boolean>` | ||||||
|  | - `keydown<Boolean>` | ||||||
|  | - `splitKey<string>` (default is `+`) | ||||||
|  | - `capture<Boolean>` | ||||||
|  | - `single<Boolean>` | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('o, enter', { | ||||||
|  |   scope: 'wcj', | ||||||
|  |   element: document.getElementById('wrapper'), | ||||||
|  | }, function() { | ||||||
|  |   console.log('do something else'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | hotkeys('ctrl-+', { splitKey: '-' }, function(e) { | ||||||
|  |   console.log('you pressed ctrl and +'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | hotkeys('+', { splitKey: '-' }, function(e){ | ||||||
|  |   console.log('you pressed +'); | ||||||
|  | }) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | **keyup** | ||||||
|  |  | ||||||
|  | **key down** and **key up** both perform callback events. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('ctrl+a,alt+a+s', {keyup: true}, function(event, handler) { | ||||||
|  |   if (event.type === 'keydown') { | ||||||
|  |     console.log('keydown:', event.type, handler, handler.key); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (event.type === 'keyup') { | ||||||
|  |     console.log('keyup:', event.type, handler, handler.key); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## API REFERENCE | ||||||
|  |  | ||||||
|  | Asterisk "*" | ||||||
|  |  | ||||||
|  | Modifier key judgments | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('*', function() { | ||||||
|  |   if (hotkeys.shift) { | ||||||
|  |     console.log('shift is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.ctrl) { | ||||||
|  |     console.log('ctrl is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.alt) { | ||||||
|  |     console.log('alt is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.option) { | ||||||
|  |     console.log('option is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.control) { | ||||||
|  |     console.log('control is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.cmd) { | ||||||
|  |     console.log('cmd is pressed!'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (hotkeys.command) { | ||||||
|  |     console.log('command is pressed!'); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### setScope | ||||||
|  |  | ||||||
|  | Use the `hotkeys.setScope` method to set scope. There can only be one active scope besides 'all'.  By default 'all' is always active. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | // Define shortcuts with a scope | ||||||
|  | hotkeys('ctrl+o, ctrl+alt+enter', 'issues', function() { | ||||||
|  |   console.log('do something'); | ||||||
|  | }); | ||||||
|  | hotkeys('o, enter', 'files', function() { | ||||||
|  |   console.log('do something else'); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Set the scope (only 'all' and 'issues' shortcuts will be honored) | ||||||
|  | hotkeys.setScope('issues'); // default scope is 'all' | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### getScope | ||||||
|  |  | ||||||
|  | Use the `hotkeys.getScope` method to get scope. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.getScope(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### deleteScope | ||||||
|  |  | ||||||
|  | Use the `hotkeys.deleteScope` method to delete a scope. This will also remove all associated hotkeys with it. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.deleteScope('issues'); | ||||||
|  | ``` | ||||||
|  | You can use second argument, if need set new scope after deleting. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.deleteScope('issues', 'newScopeName'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### unbind | ||||||
|  |  | ||||||
|  | Similar to defining shortcuts, they can be unbound using `hotkeys.unbind`. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | // unbind 'a' handler | ||||||
|  | hotkeys.unbind('a'); | ||||||
|  |  | ||||||
|  | // Unbind a hotkeys only for a single scope | ||||||
|  | // If no scope is specified it defaults to the current scope (hotkeys.getScope()) | ||||||
|  | hotkeys.unbind('o, enter', 'issues'); | ||||||
|  | hotkeys.unbind('o, enter', 'files'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Unbind events through functions. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | function example() { | ||||||
|  |   hotkeys('a', example); | ||||||
|  |   hotkeys.unbind('a', example); | ||||||
|  |  | ||||||
|  |   hotkeys('a', 'issues', example); | ||||||
|  |   hotkeys.unbind('a', 'issues', example); | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | To unbind everything. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.unbind(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### isPressed | ||||||
|  |  | ||||||
|  | For example, `hotkeys.isPressed(77)` is true if the `M` key is currently pressed. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('a', function() { | ||||||
|  |   console.log(hotkeys.isPressed('a')); //=> true | ||||||
|  |   console.log(hotkeys.isPressed('A')); //=> true | ||||||
|  |   console.log(hotkeys.isPressed(65)); //=> true | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### trigger | ||||||
|  |  | ||||||
|  | trigger shortcut key event | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.trigger('ctrl+o'); | ||||||
|  | hotkeys.trigger('ctrl+o', 'scope2'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### getPressedKeyCodes | ||||||
|  |  | ||||||
|  | Returns an array of key codes currently pressed. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |   console.log(hotkeys.getPressedKeyCodes()); //=> [17, 65] or [70] | ||||||
|  | }) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### getPressedKeyString | ||||||
|  |  | ||||||
|  | Returns an array of key codes currently pressed. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |   console.log(hotkeys.getPressedKeyString()); //=> ['⌘', '⌃', '⇧', 'A', 'F'] | ||||||
|  | }) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### getAllKeyCodes | ||||||
|  |  | ||||||
|  | Get a list of all registration codes. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |   console.log(hotkeys.getAllKeyCodes()); | ||||||
|  |   // [ | ||||||
|  |   //   { scope: 'all', shortcut: 'command+ctrl+shift+a', mods: [91, 17, 16], keys: [91, 17, 16, 65] }, | ||||||
|  |   //   { scope: 'all', shortcut: 'f', mods: [], keys: [42] } | ||||||
|  |   // ] | ||||||
|  | }) | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### filter | ||||||
|  |  | ||||||
|  | By default hotkeys are not enabled for `INPUT` `SELECT` `TEXTAREA` elements. `Hotkeys.filter` to return to the `true` shortcut keys set to play a role, `false` shortcut keys set up failure. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | hotkeys.filter = function(event){ | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  | //How to add the filter to edit labels. <div contentEditable="true"></div> | ||||||
|  | //"contentEditable" Older browsers that do not support drops | ||||||
|  | hotkeys.filter = function(event) { | ||||||
|  |   var target = event.target || event.srcElement; | ||||||
|  |   var tagName = target.tagName; | ||||||
|  |   return !(target.isContentEditable || tagName == 'INPUT' || tagName == 'SELECT' || tagName == 'TEXTAREA'); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | hotkeys.filter = function(event){ | ||||||
|  |   var tagName = (event.target || event.srcElement).tagName; | ||||||
|  |   hotkeys.setScope(/^(INPUT|TEXTAREA|SELECT)$/.test(tagName) ? 'input' : 'other'); | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### noConflict | ||||||
|  |  | ||||||
|  | Relinquish HotKeys’s control of the `hotkeys` variable. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | var k = hotkeys.noConflict(); | ||||||
|  | k('a', function() { | ||||||
|  |   console.log("do something") | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | hotkeys() | ||||||
|  | // -->Uncaught TypeError: hotkeys is not a function(anonymous function) | ||||||
|  | // @ VM2170:2InjectedScript._evaluateOn | ||||||
|  | // @ VM2165:883InjectedScript._evaluateAndWrap | ||||||
|  | // @ VM2165:816InjectedScript.evaluate @ VM2165:682 | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Development | ||||||
|  |  | ||||||
|  | To develop, Install dependencies, Get the code: | ||||||
|  |  | ||||||
|  | ```shell | ||||||
|  | $ git https://github.com/jaywcjlove/hotkeys.git | ||||||
|  | $ cd hotkeys     # Into the directory | ||||||
|  | $ npm install    # or  yarn install | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | To develop, run the self-reloading build: | ||||||
|  |  | ||||||
|  | ```shell | ||||||
|  | $ npm run watch | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Run Document Website Environment. | ||||||
|  |  | ||||||
|  | ```shell | ||||||
|  | $ npm run doc | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | To contribute, please fork Hotkeys.js, add your patch and tests for it (in the `test/` folder) and submit a pull request. | ||||||
|  |  | ||||||
|  | ```shell | ||||||
|  | $ npm run test | ||||||
|  | $ npm run test:watch # Development model | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Contributors | ||||||
|  |  | ||||||
|  | As always, thanks to our amazing contributors! | ||||||
|  |  | ||||||
|  | <a href="https://github.com/jaywcjlove/hotkeys-js/graphs/contributors"> | ||||||
|  |   <img src="https://jaywcjlove.github.io/hotkeys-js/CONTRIBUTORS.svg" /> | ||||||
|  | </a> | ||||||
|  |  | ||||||
|  | Made with [github-action-contributors](https://github.com/jaywcjlove/github-action-contributors). | ||||||
|  |  | ||||||
|  | ## License | ||||||
|  |  | ||||||
|  | [MIT © Kenny Wong](./LICENSE) | ||||||
							
								
								
									
										13
									
								
								term/pkg/hotkeys-js/dist/README.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								term/pkg/hotkeys-js/dist/README.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
|  | Explanation of Build Files | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | | UMD | Size | CommonJS | Size | ES Module | Size | | ||||||
|  | | ---- | ---- | ---- | ---- | ---- | ---- | | ||||||
|  | | hotkeys.js | 8.37kb | hotkeys.common.js | 8.13kb | hotkeys.esm.js | 8.12kb | | ||||||
|  | | hotkeys.min.js | 3.62kb (gzipped: 1.73kb) | hotkeys.common.min.js | (gzipped: 1.84kb) | - | - | | ||||||
|  |  | ||||||
|  | - [UMD](https://github.com/umdjs/umd): UMD builds can be used directly in the browser via a `<script>` tag.  | ||||||
|  | - [CommonJS](http://wiki.commonjs.org/wiki/Modules/1.1): CommonJS builds are intended for use with older bundlers like [browserify](http://browserify.org/) or [webpack 1](https://webpack.github.io/).  | ||||||
|  | - [ES Module](http://exploringjs.com/es6/ch_modules.html): ES module builds are intended for use with modern bundlers like [webpack 2](https://webpack.js.org/) or [rollup](http://rollupjs.org/).  | ||||||
							
								
								
									
										677
									
								
								term/pkg/hotkeys-js/dist/hotkeys.common.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										677
									
								
								term/pkg/hotkeys-js/dist/hotkeys.common.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,677 @@ | |||||||
|  | /**!  | ||||||
|  |  * hotkeys-js v3.13.7  | ||||||
|  |  * A simple micro-library for defining and dispatching keyboard shortcuts. It has no dependencies.  | ||||||
|  |  *  | ||||||
|  |  * Copyright (c) 2024 kenny wong <wowohoo@qq.com>  | ||||||
|  |  * https://github.com/jaywcjlove/hotkeys-js.git  | ||||||
|  |  *  | ||||||
|  |  * @website: https://jaywcjlove.github.io/hotkeys-js | ||||||
|  |   | ||||||
|  |  * Licensed under the MIT license  | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | 'use strict'; | ||||||
|  |  | ||||||
|  | const isff = typeof navigator !== 'undefined' ? navigator.userAgent.toLowerCase().indexOf('firefox') > 0 : false; | ||||||
|  |  | ||||||
|  | // 绑定事件 | ||||||
|  | function addEvent(object, event, method, useCapture) { | ||||||
|  |   if (object.addEventListener) { | ||||||
|  |     object.addEventListener(event, method, useCapture); | ||||||
|  |   } else if (object.attachEvent) { | ||||||
|  |     object.attachEvent("on".concat(event), method); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function removeEvent(object, event, method, useCapture) { | ||||||
|  |   if (object.removeEventListener) { | ||||||
|  |     object.removeEventListener(event, method, useCapture); | ||||||
|  |   } else if (object.detachEvent) { | ||||||
|  |     object.detachEvent("on".concat(event), method); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 修饰键转换成对应的键码 | ||||||
|  | function getMods(modifier, key) { | ||||||
|  |   const mods = key.slice(0, key.length - 1); | ||||||
|  |   for (let i = 0; i < mods.length; i++) mods[i] = modifier[mods[i].toLowerCase()]; | ||||||
|  |   return mods; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 处理传的key字符串转换成数组 | ||||||
|  | function getKeys(key) { | ||||||
|  |   if (typeof key !== 'string') key = ''; | ||||||
|  |   key = key.replace(/\s/g, ''); // 匹配任何空白字符,包括空格、制表符、换页符等等 | ||||||
|  |   const keys = key.split(','); // 同时设置多个快捷键,以','分割 | ||||||
|  |   let index = keys.lastIndexOf(''); | ||||||
|  |  | ||||||
|  |   // 快捷键可能包含',',需特殊处理 | ||||||
|  |   for (; index >= 0;) { | ||||||
|  |     keys[index - 1] += ','; | ||||||
|  |     keys.splice(index, 1); | ||||||
|  |     index = keys.lastIndexOf(''); | ||||||
|  |   } | ||||||
|  |   return keys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 比较修饰键的数组 | ||||||
|  | function compareArray(a1, a2) { | ||||||
|  |   const arr1 = a1.length >= a2.length ? a1 : a2; | ||||||
|  |   const arr2 = a1.length >= a2.length ? a2 : a1; | ||||||
|  |   let isIndex = true; | ||||||
|  |   for (let i = 0; i < arr1.length; i++) { | ||||||
|  |     if (arr2.indexOf(arr1[i]) === -1) isIndex = false; | ||||||
|  |   } | ||||||
|  |   return isIndex; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Special Keys | ||||||
|  | const _keyMap = { | ||||||
|  |   backspace: 8, | ||||||
|  |   '⌫': 8, | ||||||
|  |   tab: 9, | ||||||
|  |   clear: 12, | ||||||
|  |   enter: 13, | ||||||
|  |   '↩': 13, | ||||||
|  |   return: 13, | ||||||
|  |   esc: 27, | ||||||
|  |   escape: 27, | ||||||
|  |   space: 32, | ||||||
|  |   left: 37, | ||||||
|  |   up: 38, | ||||||
|  |   right: 39, | ||||||
|  |   down: 40, | ||||||
|  |   del: 46, | ||||||
|  |   delete: 46, | ||||||
|  |   ins: 45, | ||||||
|  |   insert: 45, | ||||||
|  |   home: 36, | ||||||
|  |   end: 35, | ||||||
|  |   pageup: 33, | ||||||
|  |   pagedown: 34, | ||||||
|  |   capslock: 20, | ||||||
|  |   num_0: 96, | ||||||
|  |   num_1: 97, | ||||||
|  |   num_2: 98, | ||||||
|  |   num_3: 99, | ||||||
|  |   num_4: 100, | ||||||
|  |   num_5: 101, | ||||||
|  |   num_6: 102, | ||||||
|  |   num_7: 103, | ||||||
|  |   num_8: 104, | ||||||
|  |   num_9: 105, | ||||||
|  |   num_multiply: 106, | ||||||
|  |   num_add: 107, | ||||||
|  |   num_enter: 108, | ||||||
|  |   num_subtract: 109, | ||||||
|  |   num_decimal: 110, | ||||||
|  |   num_divide: 111, | ||||||
|  |   '⇪': 20, | ||||||
|  |   ',': 188, | ||||||
|  |   '.': 190, | ||||||
|  |   '/': 191, | ||||||
|  |   '`': 192, | ||||||
|  |   '-': isff ? 173 : 189, | ||||||
|  |   '=': isff ? 61 : 187, | ||||||
|  |   ';': isff ? 59 : 186, | ||||||
|  |   '\'': 222, | ||||||
|  |   '[': 219, | ||||||
|  |   ']': 221, | ||||||
|  |   '\\': 220 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Modifier Keys | ||||||
|  | const _modifier = { | ||||||
|  |   // shiftKey | ||||||
|  |   '⇧': 16, | ||||||
|  |   shift: 16, | ||||||
|  |   // altKey | ||||||
|  |   '⌥': 18, | ||||||
|  |   alt: 18, | ||||||
|  |   option: 18, | ||||||
|  |   // ctrlKey | ||||||
|  |   '⌃': 17, | ||||||
|  |   ctrl: 17, | ||||||
|  |   control: 17, | ||||||
|  |   // metaKey | ||||||
|  |   '⌘': 91, | ||||||
|  |   cmd: 91, | ||||||
|  |   command: 91 | ||||||
|  | }; | ||||||
|  | const modifierMap = { | ||||||
|  |   16: 'shiftKey', | ||||||
|  |   18: 'altKey', | ||||||
|  |   17: 'ctrlKey', | ||||||
|  |   91: 'metaKey', | ||||||
|  |   shiftKey: 16, | ||||||
|  |   ctrlKey: 17, | ||||||
|  |   altKey: 18, | ||||||
|  |   metaKey: 91 | ||||||
|  | }; | ||||||
|  | const _mods = { | ||||||
|  |   16: false, | ||||||
|  |   18: false, | ||||||
|  |   17: false, | ||||||
|  |   91: false | ||||||
|  | }; | ||||||
|  | const _handlers = {}; | ||||||
|  |  | ||||||
|  | // F1~F12 special key | ||||||
|  | for (let k = 1; k < 20; k++) { | ||||||
|  |   _keyMap["f".concat(k)] = 111 + k; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | let _downKeys = []; // 记录摁下的绑定键 | ||||||
|  | let winListendFocus = null; // window是否已经监听了focus事件 | ||||||
|  | let _scope = 'all'; // 默认热键范围 | ||||||
|  | const elementEventMap = new Map(); // 已绑定事件的节点记录 | ||||||
|  |  | ||||||
|  | // 返回键码 | ||||||
|  | const code = x => _keyMap[x.toLowerCase()] || _modifier[x.toLowerCase()] || x.toUpperCase().charCodeAt(0); | ||||||
|  | const getKey = x => Object.keys(_keyMap).find(k => _keyMap[k] === x); | ||||||
|  | const getModifier = x => Object.keys(_modifier).find(k => _modifier[k] === x); | ||||||
|  |  | ||||||
|  | // 设置获取当前范围(默认为'所有') | ||||||
|  | function setScope(scope) { | ||||||
|  |   _scope = scope || 'all'; | ||||||
|  | } | ||||||
|  | // 获取当前范围 | ||||||
|  | function getScope() { | ||||||
|  |   return _scope || 'all'; | ||||||
|  | } | ||||||
|  | // 获取摁下绑定键的键值 | ||||||
|  | function getPressedKeyCodes() { | ||||||
|  |   return _downKeys.slice(0); | ||||||
|  | } | ||||||
|  | function getPressedKeyString() { | ||||||
|  |   return _downKeys.map(c => getKey(c) || getModifier(c) || String.fromCharCode(c)); | ||||||
|  | } | ||||||
|  | function getAllKeyCodes() { | ||||||
|  |   const result = []; | ||||||
|  |   Object.keys(_handlers).forEach(k => { | ||||||
|  |     _handlers[k].forEach(_ref => { | ||||||
|  |       let { | ||||||
|  |         key, | ||||||
|  |         scope, | ||||||
|  |         mods, | ||||||
|  |         shortcut | ||||||
|  |       } = _ref; | ||||||
|  |       result.push({ | ||||||
|  |         scope, | ||||||
|  |         shortcut, | ||||||
|  |         mods, | ||||||
|  |         keys: key.split('+').map(v => code(v)) | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |   return result; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 表单控件控件判断 返回 Boolean | ||||||
|  | // hotkey is effective only when filter return true | ||||||
|  | function filter(event) { | ||||||
|  |   const target = event.target || event.srcElement; | ||||||
|  |   const { | ||||||
|  |     tagName | ||||||
|  |   } = target; | ||||||
|  |   let flag = true; | ||||||
|  |   const isInput = tagName === 'INPUT' && !['checkbox', 'radio', 'range', 'button', 'file', 'reset', 'submit', 'color'].includes(target.type); | ||||||
|  |   // ignore: isContentEditable === 'true', <input> and <textarea> when readOnly state is false, <select> | ||||||
|  |   if (target.isContentEditable || (isInput || tagName === 'TEXTAREA' || tagName === 'SELECT') && !target.readOnly) { | ||||||
|  |     flag = false; | ||||||
|  |   } | ||||||
|  |   return flag; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 判断摁下的键是否为某个键,返回true或者false | ||||||
|  | function isPressed(keyCode) { | ||||||
|  |   if (typeof keyCode === 'string') { | ||||||
|  |     keyCode = code(keyCode); // 转换成键码 | ||||||
|  |   } | ||||||
|  |   return _downKeys.indexOf(keyCode) !== -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 循环删除handlers中的所有 scope(范围) | ||||||
|  | function deleteScope(scope, newScope) { | ||||||
|  |   let handlers; | ||||||
|  |   let i; | ||||||
|  |  | ||||||
|  |   // 没有指定scope,获取scope | ||||||
|  |   if (!scope) scope = getScope(); | ||||||
|  |   for (const key in _handlers) { | ||||||
|  |     if (Object.prototype.hasOwnProperty.call(_handlers, key)) { | ||||||
|  |       handlers = _handlers[key]; | ||||||
|  |       for (i = 0; i < handlers.length;) { | ||||||
|  |         if (handlers[i].scope === scope) { | ||||||
|  |           const deleteItems = handlers.splice(i, 1); | ||||||
|  |           deleteItems.forEach(_ref2 => { | ||||||
|  |             let { | ||||||
|  |               element | ||||||
|  |             } = _ref2; | ||||||
|  |             return removeKeyEvent(element); | ||||||
|  |           }); | ||||||
|  |         } else { | ||||||
|  |           i++; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 如果scope被删除,将scope重置为all | ||||||
|  |   if (getScope() === scope) setScope(newScope || 'all'); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 清除修饰键 | ||||||
|  | function clearModifier(event) { | ||||||
|  |   let key = event.keyCode || event.which || event.charCode; | ||||||
|  |   const i = _downKeys.indexOf(key); | ||||||
|  |  | ||||||
|  |   // 从列表中清除按压过的键 | ||||||
|  |   if (i >= 0) { | ||||||
|  |     _downKeys.splice(i, 1); | ||||||
|  |   } | ||||||
|  |   // 特殊处理 cmmand 键,在 cmmand 组合快捷键 keyup 只执行一次的问题 | ||||||
|  |   if (event.key && event.key.toLowerCase() === 'meta') { | ||||||
|  |     _downKeys.splice(0, _downKeys.length); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 修饰键 shiftKey altKey ctrlKey (command||metaKey) 清除 | ||||||
|  |   if (key === 93 || key === 224) key = 91; | ||||||
|  |   if (key in _mods) { | ||||||
|  |     _mods[key] = false; | ||||||
|  |  | ||||||
|  |     // 将修饰键重置为false | ||||||
|  |     for (const k in _modifier) if (_modifier[k] === key) hotkeys[k] = false; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function unbind(keysInfo) { | ||||||
|  |   // unbind(), unbind all keys | ||||||
|  |   if (typeof keysInfo === 'undefined') { | ||||||
|  |     Object.keys(_handlers).forEach(key => { | ||||||
|  |       Array.isArray(_handlers[key]) && _handlers[key].forEach(info => eachUnbind(info)); | ||||||
|  |       delete _handlers[key]; | ||||||
|  |     }); | ||||||
|  |     removeKeyEvent(null); | ||||||
|  |   } else if (Array.isArray(keysInfo)) { | ||||||
|  |     // support like : unbind([{key: 'ctrl+a', scope: 's1'}, {key: 'ctrl-a', scope: 's2', splitKey: '-'}]) | ||||||
|  |     keysInfo.forEach(info => { | ||||||
|  |       if (info.key) eachUnbind(info); | ||||||
|  |     }); | ||||||
|  |   } else if (typeof keysInfo === 'object') { | ||||||
|  |     // support like unbind({key: 'ctrl+a, ctrl+b', scope:'abc'}) | ||||||
|  |     if (keysInfo.key) eachUnbind(keysInfo); | ||||||
|  |   } else if (typeof keysInfo === 'string') { | ||||||
|  |     for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { | ||||||
|  |       args[_key - 1] = arguments[_key]; | ||||||
|  |     } | ||||||
|  |     // support old method | ||||||
|  |     // eslint-disable-line | ||||||
|  |     let [scope, method] = args; | ||||||
|  |     if (typeof scope === 'function') { | ||||||
|  |       method = scope; | ||||||
|  |       scope = ''; | ||||||
|  |     } | ||||||
|  |     eachUnbind({ | ||||||
|  |       key: keysInfo, | ||||||
|  |       scope, | ||||||
|  |       method, | ||||||
|  |       splitKey: '+' | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 解除绑定某个范围的快捷键 | ||||||
|  | const eachUnbind = _ref3 => { | ||||||
|  |   let { | ||||||
|  |     key, | ||||||
|  |     scope, | ||||||
|  |     method, | ||||||
|  |     splitKey = '+' | ||||||
|  |   } = _ref3; | ||||||
|  |   const multipleKeys = getKeys(key); | ||||||
|  |   multipleKeys.forEach(originKey => { | ||||||
|  |     const unbindKeys = originKey.split(splitKey); | ||||||
|  |     const len = unbindKeys.length; | ||||||
|  |     const lastKey = unbindKeys[len - 1]; | ||||||
|  |     const keyCode = lastKey === '*' ? '*' : code(lastKey); | ||||||
|  |     if (!_handlers[keyCode]) return; | ||||||
|  |     // 判断是否传入范围,没有就获取范围 | ||||||
|  |     if (!scope) scope = getScope(); | ||||||
|  |     const mods = len > 1 ? getMods(_modifier, unbindKeys) : []; | ||||||
|  |     const unbindElements = []; | ||||||
|  |     _handlers[keyCode] = _handlers[keyCode].filter(record => { | ||||||
|  |       // 通过函数判断,是否解除绑定,函数相等直接返回 | ||||||
|  |       const isMatchingMethod = method ? record.method === method : true; | ||||||
|  |       const isUnbind = isMatchingMethod && record.scope === scope && compareArray(record.mods, mods); | ||||||
|  |       if (isUnbind) unbindElements.push(record.element); | ||||||
|  |       return !isUnbind; | ||||||
|  |     }); | ||||||
|  |     unbindElements.forEach(element => removeKeyEvent(element)); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // 对监听对应快捷键的回调函数进行处理 | ||||||
|  | function eventHandler(event, handler, scope, element) { | ||||||
|  |   if (handler.element !== element) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   let modifiersMatch; | ||||||
|  |  | ||||||
|  |   // 看它是否在当前范围 | ||||||
|  |   if (handler.scope === scope || handler.scope === 'all') { | ||||||
|  |     // 检查是否匹配修饰符(如果有返回true) | ||||||
|  |     modifiersMatch = handler.mods.length > 0; | ||||||
|  |     for (const y in _mods) { | ||||||
|  |       if (Object.prototype.hasOwnProperty.call(_mods, y)) { | ||||||
|  |         if (!_mods[y] && handler.mods.indexOf(+y) > -1 || _mods[y] && handler.mods.indexOf(+y) === -1) { | ||||||
|  |           modifiersMatch = false; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 调用处理程序,如果是修饰键不做处理 | ||||||
|  |     if (handler.mods.length === 0 && !_mods[16] && !_mods[18] && !_mods[17] && !_mods[91] || modifiersMatch || handler.shortcut === '*') { | ||||||
|  |       handler.keys = []; | ||||||
|  |       handler.keys = handler.keys.concat(_downKeys); | ||||||
|  |       if (handler.method(event, handler) === false) { | ||||||
|  |         if (event.preventDefault) event.preventDefault();else event.returnValue = false; | ||||||
|  |         if (event.stopPropagation) event.stopPropagation(); | ||||||
|  |         if (event.cancelBubble) event.cancelBubble = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 处理keydown事件 | ||||||
|  | function dispatch(event, element) { | ||||||
|  |   const asterisk = _handlers['*']; | ||||||
|  |   let key = event.keyCode || event.which || event.charCode; | ||||||
|  |  | ||||||
|  |   // 表单控件过滤 默认表单控件不触发快捷键 | ||||||
|  |   if (!hotkeys.filter.call(this, event)) return; | ||||||
|  |  | ||||||
|  |   // Gecko(Firefox)的command键值224,在Webkit(Chrome)中保持一致 | ||||||
|  |   // Webkit左右 command 键值不一样 | ||||||
|  |   if (key === 93 || key === 224) key = 91; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Collect bound keys | ||||||
|  |    * If an Input Method Editor is processing key input and the event is keydown, return 229. | ||||||
|  |    * https://stackoverflow.com/questions/25043934/is-it-ok-to-ignore-keydown-events-with-keycode-229 | ||||||
|  |    * http://lists.w3.org/Archives/Public/www-dom/2010JulSep/att-0182/keyCode-spec.html | ||||||
|  |    */ | ||||||
|  |   if (_downKeys.indexOf(key) === -1 && key !== 229) _downKeys.push(key); | ||||||
|  |   /** | ||||||
|  |    * Jest test cases are required. | ||||||
|  |    * =============================== | ||||||
|  |    */ | ||||||
|  |   ['ctrlKey', 'altKey', 'shiftKey', 'metaKey'].forEach(keyName => { | ||||||
|  |     const keyNum = modifierMap[keyName]; | ||||||
|  |     if (event[keyName] && _downKeys.indexOf(keyNum) === -1) { | ||||||
|  |       _downKeys.push(keyNum); | ||||||
|  |     } else if (!event[keyName] && _downKeys.indexOf(keyNum) > -1) { | ||||||
|  |       _downKeys.splice(_downKeys.indexOf(keyNum), 1); | ||||||
|  |     } else if (keyName === 'metaKey' && event[keyName] && _downKeys.length === 3) { | ||||||
|  |       /** | ||||||
|  |        * Fix if Command is pressed: | ||||||
|  |        * =============================== | ||||||
|  |        */ | ||||||
|  |       if (!(event.ctrlKey || event.shiftKey || event.altKey)) { | ||||||
|  |         _downKeys = _downKeys.slice(_downKeys.indexOf(keyNum)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   /** | ||||||
|  |    * ------------------------------- | ||||||
|  |    */ | ||||||
|  |  | ||||||
|  |   if (key in _mods) { | ||||||
|  |     _mods[key] = true; | ||||||
|  |  | ||||||
|  |     // 将特殊字符的key注册到 hotkeys 上 | ||||||
|  |     for (const k in _modifier) { | ||||||
|  |       if (_modifier[k] === key) hotkeys[k] = true; | ||||||
|  |     } | ||||||
|  |     if (!asterisk) return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 将 modifierMap 里面的修饰键绑定到 event 中 | ||||||
|  |   for (const e in _mods) { | ||||||
|  |     if (Object.prototype.hasOwnProperty.call(_mods, e)) { | ||||||
|  |       _mods[e] = event[modifierMap[e]]; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   /** | ||||||
|  |    * https://github.com/jaywcjlove/hotkeys/pull/129 | ||||||
|  |    * This solves the issue in Firefox on Windows where hotkeys corresponding to special characters would not trigger. | ||||||
|  |    * An example of this is ctrl+alt+m on a Swedish keyboard which is used to type μ. | ||||||
|  |    * Browser support: https://caniuse.com/#feat=keyboardevent-getmodifierstate | ||||||
|  |    */ | ||||||
|  |   if (event.getModifierState && !(event.altKey && !event.ctrlKey) && event.getModifierState('AltGraph')) { | ||||||
|  |     if (_downKeys.indexOf(17) === -1) { | ||||||
|  |       _downKeys.push(17); | ||||||
|  |     } | ||||||
|  |     if (_downKeys.indexOf(18) === -1) { | ||||||
|  |       _downKeys.push(18); | ||||||
|  |     } | ||||||
|  |     _mods[17] = true; | ||||||
|  |     _mods[18] = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 获取范围 默认为 `all` | ||||||
|  |   const scope = getScope(); | ||||||
|  |   // 对任何快捷键都需要做的处理 | ||||||
|  |   if (asterisk) { | ||||||
|  |     for (let i = 0; i < asterisk.length; i++) { | ||||||
|  |       if (asterisk[i].scope === scope && (event.type === 'keydown' && asterisk[i].keydown || event.type === 'keyup' && asterisk[i].keyup)) { | ||||||
|  |         eventHandler(event, asterisk[i], scope, element); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   // key 不在 _handlers 中返回 | ||||||
|  |   if (!(key in _handlers)) return; | ||||||
|  |   const handlerKey = _handlers[key]; | ||||||
|  |   const keyLen = handlerKey.length; | ||||||
|  |   for (let i = 0; i < keyLen; i++) { | ||||||
|  |     if (event.type === 'keydown' && handlerKey[i].keydown || event.type === 'keyup' && handlerKey[i].keyup) { | ||||||
|  |       if (handlerKey[i].key) { | ||||||
|  |         const record = handlerKey[i]; | ||||||
|  |         const { | ||||||
|  |           splitKey | ||||||
|  |         } = record; | ||||||
|  |         const keyShortcut = record.key.split(splitKey); | ||||||
|  |         const _downKeysCurrent = []; // 记录当前按键键值 | ||||||
|  |         for (let a = 0; a < keyShortcut.length; a++) { | ||||||
|  |           _downKeysCurrent.push(code(keyShortcut[a])); | ||||||
|  |         } | ||||||
|  |         if (_downKeysCurrent.sort().join('') === _downKeys.sort().join('')) { | ||||||
|  |           // 找到处理内容 | ||||||
|  |           eventHandler(event, record, scope, element); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function hotkeys(key, option, method) { | ||||||
|  |   _downKeys = []; | ||||||
|  |   const keys = getKeys(key); // 需要处理的快捷键列表 | ||||||
|  |   let mods = []; | ||||||
|  |   let scope = 'all'; // scope默认为all,所有范围都有效 | ||||||
|  |   let element = document; // 快捷键事件绑定节点 | ||||||
|  |   let i = 0; | ||||||
|  |   let keyup = false; | ||||||
|  |   let keydown = true; | ||||||
|  |   let splitKey = '+'; | ||||||
|  |   let capture = false; | ||||||
|  |   let single = false; // 单个callback | ||||||
|  |  | ||||||
|  |   // 对为设定范围的判断 | ||||||
|  |   if (method === undefined && typeof option === 'function') { | ||||||
|  |     method = option; | ||||||
|  |   } | ||||||
|  |   if (Object.prototype.toString.call(option) === '[object Object]') { | ||||||
|  |     if (option.scope) scope = option.scope; // eslint-disable-line | ||||||
|  |     if (option.element) element = option.element; // eslint-disable-line | ||||||
|  |     if (option.keyup) keyup = option.keyup; // eslint-disable-line | ||||||
|  |     if (option.keydown !== undefined) keydown = option.keydown; // eslint-disable-line | ||||||
|  |     if (option.capture !== undefined) capture = option.capture; // eslint-disable-line | ||||||
|  |     if (typeof option.splitKey === 'string') splitKey = option.splitKey; // eslint-disable-line | ||||||
|  |     if (option.single === true) single = true; // eslint-disable-line | ||||||
|  |   } | ||||||
|  |   if (typeof option === 'string') scope = option; | ||||||
|  |  | ||||||
|  |   // 如果只允许单个callback,先unbind | ||||||
|  |   if (single) unbind(key, scope); | ||||||
|  |  | ||||||
|  |   // 对于每个快捷键进行处理 | ||||||
|  |   for (; i < keys.length; i++) { | ||||||
|  |     key = keys[i].split(splitKey); // 按键列表 | ||||||
|  |     mods = []; | ||||||
|  |  | ||||||
|  |     // 如果是组合快捷键取得组合快捷键 | ||||||
|  |     if (key.length > 1) mods = getMods(_modifier, key); | ||||||
|  |  | ||||||
|  |     // 将非修饰键转化为键码 | ||||||
|  |     key = key[key.length - 1]; | ||||||
|  |     key = key === '*' ? '*' : code(key); // *表示匹配所有快捷键 | ||||||
|  |  | ||||||
|  |     // 判断key是否在_handlers中,不在就赋一个空数组 | ||||||
|  |     if (!(key in _handlers)) _handlers[key] = []; | ||||||
|  |     _handlers[key].push({ | ||||||
|  |       keyup, | ||||||
|  |       keydown, | ||||||
|  |       scope, | ||||||
|  |       mods, | ||||||
|  |       shortcut: keys[i], | ||||||
|  |       method, | ||||||
|  |       key: keys[i], | ||||||
|  |       splitKey, | ||||||
|  |       element | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |   // 在全局document上设置快捷键 | ||||||
|  |   if (typeof element !== 'undefined' && window) { | ||||||
|  |     if (!elementEventMap.has(element)) { | ||||||
|  |       const keydownListener = function () { | ||||||
|  |         let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |         return dispatch(event, element); | ||||||
|  |       }; | ||||||
|  |       const keyupListenr = function () { | ||||||
|  |         let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |         dispatch(event, element); | ||||||
|  |         clearModifier(event); | ||||||
|  |       }; | ||||||
|  |       elementEventMap.set(element, { | ||||||
|  |         keydownListener, | ||||||
|  |         keyupListenr, | ||||||
|  |         capture | ||||||
|  |       }); | ||||||
|  |       addEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |       addEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |     } | ||||||
|  |     if (!winListendFocus) { | ||||||
|  |       const listener = () => { | ||||||
|  |         _downKeys = []; | ||||||
|  |       }; | ||||||
|  |       winListendFocus = { | ||||||
|  |         listener, | ||||||
|  |         capture | ||||||
|  |       }; | ||||||
|  |       addEvent(window, 'focus', listener, capture); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function trigger(shortcut) { | ||||||
|  |   let scope = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'all'; | ||||||
|  |   Object.keys(_handlers).forEach(key => { | ||||||
|  |     const dataList = _handlers[key].filter(item => item.scope === scope && item.shortcut === shortcut); | ||||||
|  |     dataList.forEach(data => { | ||||||
|  |       if (data && data.method) { | ||||||
|  |         data.method(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 销毁事件,unbind之后判断element上是否还有键盘快捷键,如果没有移除监听 | ||||||
|  | function removeKeyEvent(element) { | ||||||
|  |   const values = Object.values(_handlers).flat(); | ||||||
|  |   const findindex = values.findIndex(_ref4 => { | ||||||
|  |     let { | ||||||
|  |       element: el | ||||||
|  |     } = _ref4; | ||||||
|  |     return el === element; | ||||||
|  |   }); | ||||||
|  |   if (findindex < 0) { | ||||||
|  |     const { | ||||||
|  |       keydownListener, | ||||||
|  |       keyupListenr, | ||||||
|  |       capture | ||||||
|  |     } = elementEventMap.get(element) || {}; | ||||||
|  |     if (keydownListener && keyupListenr) { | ||||||
|  |       removeEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |       removeEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |       elementEventMap.delete(element); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (values.length <= 0 || elementEventMap.size <= 0) { | ||||||
|  |     // 移除所有的元素上的监听 | ||||||
|  |     const eventKeys = Object.keys(elementEventMap); | ||||||
|  |     eventKeys.forEach(el => { | ||||||
|  |       const { | ||||||
|  |         keydownListener, | ||||||
|  |         keyupListenr, | ||||||
|  |         capture | ||||||
|  |       } = elementEventMap.get(el) || {}; | ||||||
|  |       if (keydownListener && keyupListenr) { | ||||||
|  |         removeEvent(el, 'keyup', keyupListenr, capture); | ||||||
|  |         removeEvent(el, 'keydown', keydownListener, capture); | ||||||
|  |         elementEventMap.delete(el); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     // 清空 elementEventMap | ||||||
|  |     elementEventMap.clear(); | ||||||
|  |     // 清空 _handlers | ||||||
|  |     Object.keys(_handlers).forEach(key => delete _handlers[key]); | ||||||
|  |     // 移除window上的focus监听 | ||||||
|  |     if (winListendFocus) { | ||||||
|  |       const { | ||||||
|  |         listener, | ||||||
|  |         capture | ||||||
|  |       } = winListendFocus; | ||||||
|  |       removeEvent(window, 'focus', listener, capture); | ||||||
|  |       winListendFocus = null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | const _api = { | ||||||
|  |   getPressedKeyString, | ||||||
|  |   setScope, | ||||||
|  |   getScope, | ||||||
|  |   deleteScope, | ||||||
|  |   getPressedKeyCodes, | ||||||
|  |   getAllKeyCodes, | ||||||
|  |   isPressed, | ||||||
|  |   filter, | ||||||
|  |   trigger, | ||||||
|  |   unbind, | ||||||
|  |   keyMap: _keyMap, | ||||||
|  |   modifier: _modifier, | ||||||
|  |   modifierMap | ||||||
|  | }; | ||||||
|  | for (const a in _api) { | ||||||
|  |   if (Object.prototype.hasOwnProperty.call(_api, a)) { | ||||||
|  |     hotkeys[a] = _api[a]; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | if (typeof window !== 'undefined') { | ||||||
|  |   const _hotkeys = window.hotkeys; | ||||||
|  |   hotkeys.noConflict = deep => { | ||||||
|  |     if (deep && window.hotkeys === hotkeys) { | ||||||
|  |       window.hotkeys = _hotkeys; | ||||||
|  |     } | ||||||
|  |     return hotkeys; | ||||||
|  |   }; | ||||||
|  |   window.hotkeys = hotkeys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = hotkeys; | ||||||
							
								
								
									
										2
									
								
								term/pkg/hotkeys-js/dist/hotkeys.common.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								term/pkg/hotkeys-js/dist/hotkeys.common.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										675
									
								
								term/pkg/hotkeys-js/dist/hotkeys.esm.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										675
									
								
								term/pkg/hotkeys-js/dist/hotkeys.esm.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,675 @@ | |||||||
|  | /**!  | ||||||
|  |  * hotkeys-js v3.13.7  | ||||||
|  |  * A simple micro-library for defining and dispatching keyboard shortcuts. It has no dependencies.  | ||||||
|  |  *  | ||||||
|  |  * Copyright (c) 2024 kenny wong <wowohoo@qq.com>  | ||||||
|  |  * https://github.com/jaywcjlove/hotkeys-js.git  | ||||||
|  |  *  | ||||||
|  |  * @website: https://jaywcjlove.github.io/hotkeys-js | ||||||
|  |   | ||||||
|  |  * Licensed under the MIT license  | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | const isff = typeof navigator !== 'undefined' ? navigator.userAgent.toLowerCase().indexOf('firefox') > 0 : false; | ||||||
|  |  | ||||||
|  | // 绑定事件 | ||||||
|  | function addEvent(object, event, method, useCapture) { | ||||||
|  |   if (object.addEventListener) { | ||||||
|  |     object.addEventListener(event, method, useCapture); | ||||||
|  |   } else if (object.attachEvent) { | ||||||
|  |     object.attachEvent("on".concat(event), method); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function removeEvent(object, event, method, useCapture) { | ||||||
|  |   if (object.removeEventListener) { | ||||||
|  |     object.removeEventListener(event, method, useCapture); | ||||||
|  |   } else if (object.detachEvent) { | ||||||
|  |     object.detachEvent("on".concat(event), method); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 修饰键转换成对应的键码 | ||||||
|  | function getMods(modifier, key) { | ||||||
|  |   const mods = key.slice(0, key.length - 1); | ||||||
|  |   for (let i = 0; i < mods.length; i++) mods[i] = modifier[mods[i].toLowerCase()]; | ||||||
|  |   return mods; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 处理传的key字符串转换成数组 | ||||||
|  | function getKeys(key) { | ||||||
|  |   if (typeof key !== 'string') key = ''; | ||||||
|  |   key = key.replace(/\s/g, ''); // 匹配任何空白字符,包括空格、制表符、换页符等等 | ||||||
|  |   const keys = key.split(','); // 同时设置多个快捷键,以','分割 | ||||||
|  |   let index = keys.lastIndexOf(''); | ||||||
|  |  | ||||||
|  |   // 快捷键可能包含',',需特殊处理 | ||||||
|  |   for (; index >= 0;) { | ||||||
|  |     keys[index - 1] += ','; | ||||||
|  |     keys.splice(index, 1); | ||||||
|  |     index = keys.lastIndexOf(''); | ||||||
|  |   } | ||||||
|  |   return keys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 比较修饰键的数组 | ||||||
|  | function compareArray(a1, a2) { | ||||||
|  |   const arr1 = a1.length >= a2.length ? a1 : a2; | ||||||
|  |   const arr2 = a1.length >= a2.length ? a2 : a1; | ||||||
|  |   let isIndex = true; | ||||||
|  |   for (let i = 0; i < arr1.length; i++) { | ||||||
|  |     if (arr2.indexOf(arr1[i]) === -1) isIndex = false; | ||||||
|  |   } | ||||||
|  |   return isIndex; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Special Keys | ||||||
|  | const _keyMap = { | ||||||
|  |   backspace: 8, | ||||||
|  |   '⌫': 8, | ||||||
|  |   tab: 9, | ||||||
|  |   clear: 12, | ||||||
|  |   enter: 13, | ||||||
|  |   '↩': 13, | ||||||
|  |   return: 13, | ||||||
|  |   esc: 27, | ||||||
|  |   escape: 27, | ||||||
|  |   space: 32, | ||||||
|  |   left: 37, | ||||||
|  |   up: 38, | ||||||
|  |   right: 39, | ||||||
|  |   down: 40, | ||||||
|  |   del: 46, | ||||||
|  |   delete: 46, | ||||||
|  |   ins: 45, | ||||||
|  |   insert: 45, | ||||||
|  |   home: 36, | ||||||
|  |   end: 35, | ||||||
|  |   pageup: 33, | ||||||
|  |   pagedown: 34, | ||||||
|  |   capslock: 20, | ||||||
|  |   num_0: 96, | ||||||
|  |   num_1: 97, | ||||||
|  |   num_2: 98, | ||||||
|  |   num_3: 99, | ||||||
|  |   num_4: 100, | ||||||
|  |   num_5: 101, | ||||||
|  |   num_6: 102, | ||||||
|  |   num_7: 103, | ||||||
|  |   num_8: 104, | ||||||
|  |   num_9: 105, | ||||||
|  |   num_multiply: 106, | ||||||
|  |   num_add: 107, | ||||||
|  |   num_enter: 108, | ||||||
|  |   num_subtract: 109, | ||||||
|  |   num_decimal: 110, | ||||||
|  |   num_divide: 111, | ||||||
|  |   '⇪': 20, | ||||||
|  |   ',': 188, | ||||||
|  |   '.': 190, | ||||||
|  |   '/': 191, | ||||||
|  |   '`': 192, | ||||||
|  |   '-': isff ? 173 : 189, | ||||||
|  |   '=': isff ? 61 : 187, | ||||||
|  |   ';': isff ? 59 : 186, | ||||||
|  |   '\'': 222, | ||||||
|  |   '[': 219, | ||||||
|  |   ']': 221, | ||||||
|  |   '\\': 220 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Modifier Keys | ||||||
|  | const _modifier = { | ||||||
|  |   // shiftKey | ||||||
|  |   '⇧': 16, | ||||||
|  |   shift: 16, | ||||||
|  |   // altKey | ||||||
|  |   '⌥': 18, | ||||||
|  |   alt: 18, | ||||||
|  |   option: 18, | ||||||
|  |   // ctrlKey | ||||||
|  |   '⌃': 17, | ||||||
|  |   ctrl: 17, | ||||||
|  |   control: 17, | ||||||
|  |   // metaKey | ||||||
|  |   '⌘': 91, | ||||||
|  |   cmd: 91, | ||||||
|  |   command: 91 | ||||||
|  | }; | ||||||
|  | const modifierMap = { | ||||||
|  |   16: 'shiftKey', | ||||||
|  |   18: 'altKey', | ||||||
|  |   17: 'ctrlKey', | ||||||
|  |   91: 'metaKey', | ||||||
|  |   shiftKey: 16, | ||||||
|  |   ctrlKey: 17, | ||||||
|  |   altKey: 18, | ||||||
|  |   metaKey: 91 | ||||||
|  | }; | ||||||
|  | const _mods = { | ||||||
|  |   16: false, | ||||||
|  |   18: false, | ||||||
|  |   17: false, | ||||||
|  |   91: false | ||||||
|  | }; | ||||||
|  | const _handlers = {}; | ||||||
|  |  | ||||||
|  | // F1~F12 special key | ||||||
|  | for (let k = 1; k < 20; k++) { | ||||||
|  |   _keyMap["f".concat(k)] = 111 + k; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | let _downKeys = []; // 记录摁下的绑定键 | ||||||
|  | let winListendFocus = null; // window是否已经监听了focus事件 | ||||||
|  | let _scope = 'all'; // 默认热键范围 | ||||||
|  | const elementEventMap = new Map(); // 已绑定事件的节点记录 | ||||||
|  |  | ||||||
|  | // 返回键码 | ||||||
|  | const code = x => _keyMap[x.toLowerCase()] || _modifier[x.toLowerCase()] || x.toUpperCase().charCodeAt(0); | ||||||
|  | const getKey = x => Object.keys(_keyMap).find(k => _keyMap[k] === x); | ||||||
|  | const getModifier = x => Object.keys(_modifier).find(k => _modifier[k] === x); | ||||||
|  |  | ||||||
|  | // 设置获取当前范围(默认为'所有') | ||||||
|  | function setScope(scope) { | ||||||
|  |   _scope = scope || 'all'; | ||||||
|  | } | ||||||
|  | // 获取当前范围 | ||||||
|  | function getScope() { | ||||||
|  |   return _scope || 'all'; | ||||||
|  | } | ||||||
|  | // 获取摁下绑定键的键值 | ||||||
|  | function getPressedKeyCodes() { | ||||||
|  |   return _downKeys.slice(0); | ||||||
|  | } | ||||||
|  | function getPressedKeyString() { | ||||||
|  |   return _downKeys.map(c => getKey(c) || getModifier(c) || String.fromCharCode(c)); | ||||||
|  | } | ||||||
|  | function getAllKeyCodes() { | ||||||
|  |   const result = []; | ||||||
|  |   Object.keys(_handlers).forEach(k => { | ||||||
|  |     _handlers[k].forEach(_ref => { | ||||||
|  |       let { | ||||||
|  |         key, | ||||||
|  |         scope, | ||||||
|  |         mods, | ||||||
|  |         shortcut | ||||||
|  |       } = _ref; | ||||||
|  |       result.push({ | ||||||
|  |         scope, | ||||||
|  |         shortcut, | ||||||
|  |         mods, | ||||||
|  |         keys: key.split('+').map(v => code(v)) | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |   return result; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 表单控件控件判断 返回 Boolean | ||||||
|  | // hotkey is effective only when filter return true | ||||||
|  | function filter(event) { | ||||||
|  |   const target = event.target || event.srcElement; | ||||||
|  |   const { | ||||||
|  |     tagName | ||||||
|  |   } = target; | ||||||
|  |   let flag = true; | ||||||
|  |   const isInput = tagName === 'INPUT' && !['checkbox', 'radio', 'range', 'button', 'file', 'reset', 'submit', 'color'].includes(target.type); | ||||||
|  |   // ignore: isContentEditable === 'true', <input> and <textarea> when readOnly state is false, <select> | ||||||
|  |   if (target.isContentEditable || (isInput || tagName === 'TEXTAREA' || tagName === 'SELECT') && !target.readOnly) { | ||||||
|  |     flag = false; | ||||||
|  |   } | ||||||
|  |   return flag; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 判断摁下的键是否为某个键,返回true或者false | ||||||
|  | function isPressed(keyCode) { | ||||||
|  |   if (typeof keyCode === 'string') { | ||||||
|  |     keyCode = code(keyCode); // 转换成键码 | ||||||
|  |   } | ||||||
|  |   return _downKeys.indexOf(keyCode) !== -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 循环删除handlers中的所有 scope(范围) | ||||||
|  | function deleteScope(scope, newScope) { | ||||||
|  |   let handlers; | ||||||
|  |   let i; | ||||||
|  |  | ||||||
|  |   // 没有指定scope,获取scope | ||||||
|  |   if (!scope) scope = getScope(); | ||||||
|  |   for (const key in _handlers) { | ||||||
|  |     if (Object.prototype.hasOwnProperty.call(_handlers, key)) { | ||||||
|  |       handlers = _handlers[key]; | ||||||
|  |       for (i = 0; i < handlers.length;) { | ||||||
|  |         if (handlers[i].scope === scope) { | ||||||
|  |           const deleteItems = handlers.splice(i, 1); | ||||||
|  |           deleteItems.forEach(_ref2 => { | ||||||
|  |             let { | ||||||
|  |               element | ||||||
|  |             } = _ref2; | ||||||
|  |             return removeKeyEvent(element); | ||||||
|  |           }); | ||||||
|  |         } else { | ||||||
|  |           i++; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 如果scope被删除,将scope重置为all | ||||||
|  |   if (getScope() === scope) setScope(newScope || 'all'); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 清除修饰键 | ||||||
|  | function clearModifier(event) { | ||||||
|  |   let key = event.keyCode || event.which || event.charCode; | ||||||
|  |   const i = _downKeys.indexOf(key); | ||||||
|  |  | ||||||
|  |   // 从列表中清除按压过的键 | ||||||
|  |   if (i >= 0) { | ||||||
|  |     _downKeys.splice(i, 1); | ||||||
|  |   } | ||||||
|  |   // 特殊处理 cmmand 键,在 cmmand 组合快捷键 keyup 只执行一次的问题 | ||||||
|  |   if (event.key && event.key.toLowerCase() === 'meta') { | ||||||
|  |     _downKeys.splice(0, _downKeys.length); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 修饰键 shiftKey altKey ctrlKey (command||metaKey) 清除 | ||||||
|  |   if (key === 93 || key === 224) key = 91; | ||||||
|  |   if (key in _mods) { | ||||||
|  |     _mods[key] = false; | ||||||
|  |  | ||||||
|  |     // 将修饰键重置为false | ||||||
|  |     for (const k in _modifier) if (_modifier[k] === key) hotkeys[k] = false; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function unbind(keysInfo) { | ||||||
|  |   // unbind(), unbind all keys | ||||||
|  |   if (typeof keysInfo === 'undefined') { | ||||||
|  |     Object.keys(_handlers).forEach(key => { | ||||||
|  |       Array.isArray(_handlers[key]) && _handlers[key].forEach(info => eachUnbind(info)); | ||||||
|  |       delete _handlers[key]; | ||||||
|  |     }); | ||||||
|  |     removeKeyEvent(null); | ||||||
|  |   } else if (Array.isArray(keysInfo)) { | ||||||
|  |     // support like : unbind([{key: 'ctrl+a', scope: 's1'}, {key: 'ctrl-a', scope: 's2', splitKey: '-'}]) | ||||||
|  |     keysInfo.forEach(info => { | ||||||
|  |       if (info.key) eachUnbind(info); | ||||||
|  |     }); | ||||||
|  |   } else if (typeof keysInfo === 'object') { | ||||||
|  |     // support like unbind({key: 'ctrl+a, ctrl+b', scope:'abc'}) | ||||||
|  |     if (keysInfo.key) eachUnbind(keysInfo); | ||||||
|  |   } else if (typeof keysInfo === 'string') { | ||||||
|  |     for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { | ||||||
|  |       args[_key - 1] = arguments[_key]; | ||||||
|  |     } | ||||||
|  |     // support old method | ||||||
|  |     // eslint-disable-line | ||||||
|  |     let [scope, method] = args; | ||||||
|  |     if (typeof scope === 'function') { | ||||||
|  |       method = scope; | ||||||
|  |       scope = ''; | ||||||
|  |     } | ||||||
|  |     eachUnbind({ | ||||||
|  |       key: keysInfo, | ||||||
|  |       scope, | ||||||
|  |       method, | ||||||
|  |       splitKey: '+' | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 解除绑定某个范围的快捷键 | ||||||
|  | const eachUnbind = _ref3 => { | ||||||
|  |   let { | ||||||
|  |     key, | ||||||
|  |     scope, | ||||||
|  |     method, | ||||||
|  |     splitKey = '+' | ||||||
|  |   } = _ref3; | ||||||
|  |   const multipleKeys = getKeys(key); | ||||||
|  |   multipleKeys.forEach(originKey => { | ||||||
|  |     const unbindKeys = originKey.split(splitKey); | ||||||
|  |     const len = unbindKeys.length; | ||||||
|  |     const lastKey = unbindKeys[len - 1]; | ||||||
|  |     const keyCode = lastKey === '*' ? '*' : code(lastKey); | ||||||
|  |     if (!_handlers[keyCode]) return; | ||||||
|  |     // 判断是否传入范围,没有就获取范围 | ||||||
|  |     if (!scope) scope = getScope(); | ||||||
|  |     const mods = len > 1 ? getMods(_modifier, unbindKeys) : []; | ||||||
|  |     const unbindElements = []; | ||||||
|  |     _handlers[keyCode] = _handlers[keyCode].filter(record => { | ||||||
|  |       // 通过函数判断,是否解除绑定,函数相等直接返回 | ||||||
|  |       const isMatchingMethod = method ? record.method === method : true; | ||||||
|  |       const isUnbind = isMatchingMethod && record.scope === scope && compareArray(record.mods, mods); | ||||||
|  |       if (isUnbind) unbindElements.push(record.element); | ||||||
|  |       return !isUnbind; | ||||||
|  |     }); | ||||||
|  |     unbindElements.forEach(element => removeKeyEvent(element)); | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // 对监听对应快捷键的回调函数进行处理 | ||||||
|  | function eventHandler(event, handler, scope, element) { | ||||||
|  |   if (handler.element !== element) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   let modifiersMatch; | ||||||
|  |  | ||||||
|  |   // 看它是否在当前范围 | ||||||
|  |   if (handler.scope === scope || handler.scope === 'all') { | ||||||
|  |     // 检查是否匹配修饰符(如果有返回true) | ||||||
|  |     modifiersMatch = handler.mods.length > 0; | ||||||
|  |     for (const y in _mods) { | ||||||
|  |       if (Object.prototype.hasOwnProperty.call(_mods, y)) { | ||||||
|  |         if (!_mods[y] && handler.mods.indexOf(+y) > -1 || _mods[y] && handler.mods.indexOf(+y) === -1) { | ||||||
|  |           modifiersMatch = false; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 调用处理程序,如果是修饰键不做处理 | ||||||
|  |     if (handler.mods.length === 0 && !_mods[16] && !_mods[18] && !_mods[17] && !_mods[91] || modifiersMatch || handler.shortcut === '*') { | ||||||
|  |       handler.keys = []; | ||||||
|  |       handler.keys = handler.keys.concat(_downKeys); | ||||||
|  |       if (handler.method(event, handler) === false) { | ||||||
|  |         if (event.preventDefault) event.preventDefault();else event.returnValue = false; | ||||||
|  |         if (event.stopPropagation) event.stopPropagation(); | ||||||
|  |         if (event.cancelBubble) event.cancelBubble = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 处理keydown事件 | ||||||
|  | function dispatch(event, element) { | ||||||
|  |   const asterisk = _handlers['*']; | ||||||
|  |   let key = event.keyCode || event.which || event.charCode; | ||||||
|  |  | ||||||
|  |   // 表单控件过滤 默认表单控件不触发快捷键 | ||||||
|  |   if (!hotkeys.filter.call(this, event)) return; | ||||||
|  |  | ||||||
|  |   // Gecko(Firefox)的command键值224,在Webkit(Chrome)中保持一致 | ||||||
|  |   // Webkit左右 command 键值不一样 | ||||||
|  |   if (key === 93 || key === 224) key = 91; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Collect bound keys | ||||||
|  |    * If an Input Method Editor is processing key input and the event is keydown, return 229. | ||||||
|  |    * https://stackoverflow.com/questions/25043934/is-it-ok-to-ignore-keydown-events-with-keycode-229 | ||||||
|  |    * http://lists.w3.org/Archives/Public/www-dom/2010JulSep/att-0182/keyCode-spec.html | ||||||
|  |    */ | ||||||
|  |   if (_downKeys.indexOf(key) === -1 && key !== 229) _downKeys.push(key); | ||||||
|  |   /** | ||||||
|  |    * Jest test cases are required. | ||||||
|  |    * =============================== | ||||||
|  |    */ | ||||||
|  |   ['ctrlKey', 'altKey', 'shiftKey', 'metaKey'].forEach(keyName => { | ||||||
|  |     const keyNum = modifierMap[keyName]; | ||||||
|  |     if (event[keyName] && _downKeys.indexOf(keyNum) === -1) { | ||||||
|  |       _downKeys.push(keyNum); | ||||||
|  |     } else if (!event[keyName] && _downKeys.indexOf(keyNum) > -1) { | ||||||
|  |       _downKeys.splice(_downKeys.indexOf(keyNum), 1); | ||||||
|  |     } else if (keyName === 'metaKey' && event[keyName] && _downKeys.length === 3) { | ||||||
|  |       /** | ||||||
|  |        * Fix if Command is pressed: | ||||||
|  |        * =============================== | ||||||
|  |        */ | ||||||
|  |       if (!(event.ctrlKey || event.shiftKey || event.altKey)) { | ||||||
|  |         _downKeys = _downKeys.slice(_downKeys.indexOf(keyNum)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   /** | ||||||
|  |    * ------------------------------- | ||||||
|  |    */ | ||||||
|  |  | ||||||
|  |   if (key in _mods) { | ||||||
|  |     _mods[key] = true; | ||||||
|  |  | ||||||
|  |     // 将特殊字符的key注册到 hotkeys 上 | ||||||
|  |     for (const k in _modifier) { | ||||||
|  |       if (_modifier[k] === key) hotkeys[k] = true; | ||||||
|  |     } | ||||||
|  |     if (!asterisk) return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 将 modifierMap 里面的修饰键绑定到 event 中 | ||||||
|  |   for (const e in _mods) { | ||||||
|  |     if (Object.prototype.hasOwnProperty.call(_mods, e)) { | ||||||
|  |       _mods[e] = event[modifierMap[e]]; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   /** | ||||||
|  |    * https://github.com/jaywcjlove/hotkeys/pull/129 | ||||||
|  |    * This solves the issue in Firefox on Windows where hotkeys corresponding to special characters would not trigger. | ||||||
|  |    * An example of this is ctrl+alt+m on a Swedish keyboard which is used to type μ. | ||||||
|  |    * Browser support: https://caniuse.com/#feat=keyboardevent-getmodifierstate | ||||||
|  |    */ | ||||||
|  |   if (event.getModifierState && !(event.altKey && !event.ctrlKey) && event.getModifierState('AltGraph')) { | ||||||
|  |     if (_downKeys.indexOf(17) === -1) { | ||||||
|  |       _downKeys.push(17); | ||||||
|  |     } | ||||||
|  |     if (_downKeys.indexOf(18) === -1) { | ||||||
|  |       _downKeys.push(18); | ||||||
|  |     } | ||||||
|  |     _mods[17] = true; | ||||||
|  |     _mods[18] = true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 获取范围 默认为 `all` | ||||||
|  |   const scope = getScope(); | ||||||
|  |   // 对任何快捷键都需要做的处理 | ||||||
|  |   if (asterisk) { | ||||||
|  |     for (let i = 0; i < asterisk.length; i++) { | ||||||
|  |       if (asterisk[i].scope === scope && (event.type === 'keydown' && asterisk[i].keydown || event.type === 'keyup' && asterisk[i].keyup)) { | ||||||
|  |         eventHandler(event, asterisk[i], scope, element); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   // key 不在 _handlers 中返回 | ||||||
|  |   if (!(key in _handlers)) return; | ||||||
|  |   const handlerKey = _handlers[key]; | ||||||
|  |   const keyLen = handlerKey.length; | ||||||
|  |   for (let i = 0; i < keyLen; i++) { | ||||||
|  |     if (event.type === 'keydown' && handlerKey[i].keydown || event.type === 'keyup' && handlerKey[i].keyup) { | ||||||
|  |       if (handlerKey[i].key) { | ||||||
|  |         const record = handlerKey[i]; | ||||||
|  |         const { | ||||||
|  |           splitKey | ||||||
|  |         } = record; | ||||||
|  |         const keyShortcut = record.key.split(splitKey); | ||||||
|  |         const _downKeysCurrent = []; // 记录当前按键键值 | ||||||
|  |         for (let a = 0; a < keyShortcut.length; a++) { | ||||||
|  |           _downKeysCurrent.push(code(keyShortcut[a])); | ||||||
|  |         } | ||||||
|  |         if (_downKeysCurrent.sort().join('') === _downKeys.sort().join('')) { | ||||||
|  |           // 找到处理内容 | ||||||
|  |           eventHandler(event, record, scope, element); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function hotkeys(key, option, method) { | ||||||
|  |   _downKeys = []; | ||||||
|  |   const keys = getKeys(key); // 需要处理的快捷键列表 | ||||||
|  |   let mods = []; | ||||||
|  |   let scope = 'all'; // scope默认为all,所有范围都有效 | ||||||
|  |   let element = document; // 快捷键事件绑定节点 | ||||||
|  |   let i = 0; | ||||||
|  |   let keyup = false; | ||||||
|  |   let keydown = true; | ||||||
|  |   let splitKey = '+'; | ||||||
|  |   let capture = false; | ||||||
|  |   let single = false; // 单个callback | ||||||
|  |  | ||||||
|  |   // 对为设定范围的判断 | ||||||
|  |   if (method === undefined && typeof option === 'function') { | ||||||
|  |     method = option; | ||||||
|  |   } | ||||||
|  |   if (Object.prototype.toString.call(option) === '[object Object]') { | ||||||
|  |     if (option.scope) scope = option.scope; // eslint-disable-line | ||||||
|  |     if (option.element) element = option.element; // eslint-disable-line | ||||||
|  |     if (option.keyup) keyup = option.keyup; // eslint-disable-line | ||||||
|  |     if (option.keydown !== undefined) keydown = option.keydown; // eslint-disable-line | ||||||
|  |     if (option.capture !== undefined) capture = option.capture; // eslint-disable-line | ||||||
|  |     if (typeof option.splitKey === 'string') splitKey = option.splitKey; // eslint-disable-line | ||||||
|  |     if (option.single === true) single = true; // eslint-disable-line | ||||||
|  |   } | ||||||
|  |   if (typeof option === 'string') scope = option; | ||||||
|  |  | ||||||
|  |   // 如果只允许单个callback,先unbind | ||||||
|  |   if (single) unbind(key, scope); | ||||||
|  |  | ||||||
|  |   // 对于每个快捷键进行处理 | ||||||
|  |   for (; i < keys.length; i++) { | ||||||
|  |     key = keys[i].split(splitKey); // 按键列表 | ||||||
|  |     mods = []; | ||||||
|  |  | ||||||
|  |     // 如果是组合快捷键取得组合快捷键 | ||||||
|  |     if (key.length > 1) mods = getMods(_modifier, key); | ||||||
|  |  | ||||||
|  |     // 将非修饰键转化为键码 | ||||||
|  |     key = key[key.length - 1]; | ||||||
|  |     key = key === '*' ? '*' : code(key); // *表示匹配所有快捷键 | ||||||
|  |  | ||||||
|  |     // 判断key是否在_handlers中,不在就赋一个空数组 | ||||||
|  |     if (!(key in _handlers)) _handlers[key] = []; | ||||||
|  |     _handlers[key].push({ | ||||||
|  |       keyup, | ||||||
|  |       keydown, | ||||||
|  |       scope, | ||||||
|  |       mods, | ||||||
|  |       shortcut: keys[i], | ||||||
|  |       method, | ||||||
|  |       key: keys[i], | ||||||
|  |       splitKey, | ||||||
|  |       element | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |   // 在全局document上设置快捷键 | ||||||
|  |   if (typeof element !== 'undefined' && window) { | ||||||
|  |     if (!elementEventMap.has(element)) { | ||||||
|  |       const keydownListener = function () { | ||||||
|  |         let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |         return dispatch(event, element); | ||||||
|  |       }; | ||||||
|  |       const keyupListenr = function () { | ||||||
|  |         let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |         dispatch(event, element); | ||||||
|  |         clearModifier(event); | ||||||
|  |       }; | ||||||
|  |       elementEventMap.set(element, { | ||||||
|  |         keydownListener, | ||||||
|  |         keyupListenr, | ||||||
|  |         capture | ||||||
|  |       }); | ||||||
|  |       addEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |       addEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |     } | ||||||
|  |     if (!winListendFocus) { | ||||||
|  |       const listener = () => { | ||||||
|  |         _downKeys = []; | ||||||
|  |       }; | ||||||
|  |       winListendFocus = { | ||||||
|  |         listener, | ||||||
|  |         capture | ||||||
|  |       }; | ||||||
|  |       addEvent(window, 'focus', listener, capture); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | function trigger(shortcut) { | ||||||
|  |   let scope = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'all'; | ||||||
|  |   Object.keys(_handlers).forEach(key => { | ||||||
|  |     const dataList = _handlers[key].filter(item => item.scope === scope && item.shortcut === shortcut); | ||||||
|  |     dataList.forEach(data => { | ||||||
|  |       if (data && data.method) { | ||||||
|  |         data.method(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // 销毁事件,unbind之后判断element上是否还有键盘快捷键,如果没有移除监听 | ||||||
|  | function removeKeyEvent(element) { | ||||||
|  |   const values = Object.values(_handlers).flat(); | ||||||
|  |   const findindex = values.findIndex(_ref4 => { | ||||||
|  |     let { | ||||||
|  |       element: el | ||||||
|  |     } = _ref4; | ||||||
|  |     return el === element; | ||||||
|  |   }); | ||||||
|  |   if (findindex < 0) { | ||||||
|  |     const { | ||||||
|  |       keydownListener, | ||||||
|  |       keyupListenr, | ||||||
|  |       capture | ||||||
|  |     } = elementEventMap.get(element) || {}; | ||||||
|  |     if (keydownListener && keyupListenr) { | ||||||
|  |       removeEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |       removeEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |       elementEventMap.delete(element); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (values.length <= 0 || elementEventMap.size <= 0) { | ||||||
|  |     // 移除所有的元素上的监听 | ||||||
|  |     const eventKeys = Object.keys(elementEventMap); | ||||||
|  |     eventKeys.forEach(el => { | ||||||
|  |       const { | ||||||
|  |         keydownListener, | ||||||
|  |         keyupListenr, | ||||||
|  |         capture | ||||||
|  |       } = elementEventMap.get(el) || {}; | ||||||
|  |       if (keydownListener && keyupListenr) { | ||||||
|  |         removeEvent(el, 'keyup', keyupListenr, capture); | ||||||
|  |         removeEvent(el, 'keydown', keydownListener, capture); | ||||||
|  |         elementEventMap.delete(el); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     // 清空 elementEventMap | ||||||
|  |     elementEventMap.clear(); | ||||||
|  |     // 清空 _handlers | ||||||
|  |     Object.keys(_handlers).forEach(key => delete _handlers[key]); | ||||||
|  |     // 移除window上的focus监听 | ||||||
|  |     if (winListendFocus) { | ||||||
|  |       const { | ||||||
|  |         listener, | ||||||
|  |         capture | ||||||
|  |       } = winListendFocus; | ||||||
|  |       removeEvent(window, 'focus', listener, capture); | ||||||
|  |       winListendFocus = null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | const _api = { | ||||||
|  |   getPressedKeyString, | ||||||
|  |   setScope, | ||||||
|  |   getScope, | ||||||
|  |   deleteScope, | ||||||
|  |   getPressedKeyCodes, | ||||||
|  |   getAllKeyCodes, | ||||||
|  |   isPressed, | ||||||
|  |   filter, | ||||||
|  |   trigger, | ||||||
|  |   unbind, | ||||||
|  |   keyMap: _keyMap, | ||||||
|  |   modifier: _modifier, | ||||||
|  |   modifierMap | ||||||
|  | }; | ||||||
|  | for (const a in _api) { | ||||||
|  |   if (Object.prototype.hasOwnProperty.call(_api, a)) { | ||||||
|  |     hotkeys[a] = _api[a]; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | if (typeof window !== 'undefined') { | ||||||
|  |   const _hotkeys = window.hotkeys; | ||||||
|  |   hotkeys.noConflict = deep => { | ||||||
|  |     if (deep && window.hotkeys === hotkeys) { | ||||||
|  |       window.hotkeys = _hotkeys; | ||||||
|  |     } | ||||||
|  |     return hotkeys; | ||||||
|  |   }; | ||||||
|  |   window.hotkeys = hotkeys; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export { hotkeys as default }; | ||||||
							
								
								
									
										683
									
								
								term/pkg/hotkeys-js/dist/hotkeys.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										683
									
								
								term/pkg/hotkeys-js/dist/hotkeys.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,683 @@ | |||||||
|  | /**!  | ||||||
|  |  * hotkeys-js v3.13.7  | ||||||
|  |  * A simple micro-library for defining and dispatching keyboard shortcuts. It has no dependencies.  | ||||||
|  |  *  | ||||||
|  |  * Copyright (c) 2024 kenny wong <wowohoo@qq.com>  | ||||||
|  |  * https://github.com/jaywcjlove/hotkeys-js.git  | ||||||
|  |  *  | ||||||
|  |  * @website: https://jaywcjlove.github.io/hotkeys-js | ||||||
|  |   | ||||||
|  |  * Licensed under the MIT license  | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | (function (global, factory) { | ||||||
|  |   typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : | ||||||
|  |   typeof define === 'function' && define.amd ? define(factory) : | ||||||
|  |   (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.hotkeys = factory()); | ||||||
|  | })(this, (function () { 'use strict'; | ||||||
|  |  | ||||||
|  |   const isff = typeof navigator !== 'undefined' ? navigator.userAgent.toLowerCase().indexOf('firefox') > 0 : false; | ||||||
|  |  | ||||||
|  |   // 绑定事件 | ||||||
|  |   function addEvent(object, event, method, useCapture) { | ||||||
|  |     if (object.addEventListener) { | ||||||
|  |       object.addEventListener(event, method, useCapture); | ||||||
|  |     } else if (object.attachEvent) { | ||||||
|  |       object.attachEvent("on".concat(event), method); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function removeEvent(object, event, method, useCapture) { | ||||||
|  |     if (object.removeEventListener) { | ||||||
|  |       object.removeEventListener(event, method, useCapture); | ||||||
|  |     } else if (object.detachEvent) { | ||||||
|  |       object.detachEvent("on".concat(event), method); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 修饰键转换成对应的键码 | ||||||
|  |   function getMods(modifier, key) { | ||||||
|  |     const mods = key.slice(0, key.length - 1); | ||||||
|  |     for (let i = 0; i < mods.length; i++) mods[i] = modifier[mods[i].toLowerCase()]; | ||||||
|  |     return mods; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 处理传的key字符串转换成数组 | ||||||
|  |   function getKeys(key) { | ||||||
|  |     if (typeof key !== 'string') key = ''; | ||||||
|  |     key = key.replace(/\s/g, ''); // 匹配任何空白字符,包括空格、制表符、换页符等等 | ||||||
|  |     const keys = key.split(','); // 同时设置多个快捷键,以','分割 | ||||||
|  |     let index = keys.lastIndexOf(''); | ||||||
|  |  | ||||||
|  |     // 快捷键可能包含',',需特殊处理 | ||||||
|  |     for (; index >= 0;) { | ||||||
|  |       keys[index - 1] += ','; | ||||||
|  |       keys.splice(index, 1); | ||||||
|  |       index = keys.lastIndexOf(''); | ||||||
|  |     } | ||||||
|  |     return keys; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 比较修饰键的数组 | ||||||
|  |   function compareArray(a1, a2) { | ||||||
|  |     const arr1 = a1.length >= a2.length ? a1 : a2; | ||||||
|  |     const arr2 = a1.length >= a2.length ? a2 : a1; | ||||||
|  |     let isIndex = true; | ||||||
|  |     for (let i = 0; i < arr1.length; i++) { | ||||||
|  |       if (arr2.indexOf(arr1[i]) === -1) isIndex = false; | ||||||
|  |     } | ||||||
|  |     return isIndex; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Special Keys | ||||||
|  |   const _keyMap = { | ||||||
|  |     backspace: 8, | ||||||
|  |     '⌫': 8, | ||||||
|  |     tab: 9, | ||||||
|  |     clear: 12, | ||||||
|  |     enter: 13, | ||||||
|  |     '↩': 13, | ||||||
|  |     return: 13, | ||||||
|  |     esc: 27, | ||||||
|  |     escape: 27, | ||||||
|  |     space: 32, | ||||||
|  |     left: 37, | ||||||
|  |     up: 38, | ||||||
|  |     right: 39, | ||||||
|  |     down: 40, | ||||||
|  |     del: 46, | ||||||
|  |     delete: 46, | ||||||
|  |     ins: 45, | ||||||
|  |     insert: 45, | ||||||
|  |     home: 36, | ||||||
|  |     end: 35, | ||||||
|  |     pageup: 33, | ||||||
|  |     pagedown: 34, | ||||||
|  |     capslock: 20, | ||||||
|  |     num_0: 96, | ||||||
|  |     num_1: 97, | ||||||
|  |     num_2: 98, | ||||||
|  |     num_3: 99, | ||||||
|  |     num_4: 100, | ||||||
|  |     num_5: 101, | ||||||
|  |     num_6: 102, | ||||||
|  |     num_7: 103, | ||||||
|  |     num_8: 104, | ||||||
|  |     num_9: 105, | ||||||
|  |     num_multiply: 106, | ||||||
|  |     num_add: 107, | ||||||
|  |     num_enter: 108, | ||||||
|  |     num_subtract: 109, | ||||||
|  |     num_decimal: 110, | ||||||
|  |     num_divide: 111, | ||||||
|  |     '⇪': 20, | ||||||
|  |     ',': 188, | ||||||
|  |     '.': 190, | ||||||
|  |     '/': 191, | ||||||
|  |     '`': 192, | ||||||
|  |     '-': isff ? 173 : 189, | ||||||
|  |     '=': isff ? 61 : 187, | ||||||
|  |     ';': isff ? 59 : 186, | ||||||
|  |     '\'': 222, | ||||||
|  |     '[': 219, | ||||||
|  |     ']': 221, | ||||||
|  |     '\\': 220 | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // Modifier Keys | ||||||
|  |   const _modifier = { | ||||||
|  |     // shiftKey | ||||||
|  |     '⇧': 16, | ||||||
|  |     shift: 16, | ||||||
|  |     // altKey | ||||||
|  |     '⌥': 18, | ||||||
|  |     alt: 18, | ||||||
|  |     option: 18, | ||||||
|  |     // ctrlKey | ||||||
|  |     '⌃': 17, | ||||||
|  |     ctrl: 17, | ||||||
|  |     control: 17, | ||||||
|  |     // metaKey | ||||||
|  |     '⌘': 91, | ||||||
|  |     cmd: 91, | ||||||
|  |     command: 91 | ||||||
|  |   }; | ||||||
|  |   const modifierMap = { | ||||||
|  |     16: 'shiftKey', | ||||||
|  |     18: 'altKey', | ||||||
|  |     17: 'ctrlKey', | ||||||
|  |     91: 'metaKey', | ||||||
|  |     shiftKey: 16, | ||||||
|  |     ctrlKey: 17, | ||||||
|  |     altKey: 18, | ||||||
|  |     metaKey: 91 | ||||||
|  |   }; | ||||||
|  |   const _mods = { | ||||||
|  |     16: false, | ||||||
|  |     18: false, | ||||||
|  |     17: false, | ||||||
|  |     91: false | ||||||
|  |   }; | ||||||
|  |   const _handlers = {}; | ||||||
|  |  | ||||||
|  |   // F1~F12 special key | ||||||
|  |   for (let k = 1; k < 20; k++) { | ||||||
|  |     _keyMap["f".concat(k)] = 111 + k; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   let _downKeys = []; // 记录摁下的绑定键 | ||||||
|  |   let winListendFocus = null; // window是否已经监听了focus事件 | ||||||
|  |   let _scope = 'all'; // 默认热键范围 | ||||||
|  |   const elementEventMap = new Map(); // 已绑定事件的节点记录 | ||||||
|  |  | ||||||
|  |   // 返回键码 | ||||||
|  |   const code = x => _keyMap[x.toLowerCase()] || _modifier[x.toLowerCase()] || x.toUpperCase().charCodeAt(0); | ||||||
|  |   const getKey = x => Object.keys(_keyMap).find(k => _keyMap[k] === x); | ||||||
|  |   const getModifier = x => Object.keys(_modifier).find(k => _modifier[k] === x); | ||||||
|  |  | ||||||
|  |   // 设置获取当前范围(默认为'所有') | ||||||
|  |   function setScope(scope) { | ||||||
|  |     _scope = scope || 'all'; | ||||||
|  |   } | ||||||
|  |   // 获取当前范围 | ||||||
|  |   function getScope() { | ||||||
|  |     return _scope || 'all'; | ||||||
|  |   } | ||||||
|  |   // 获取摁下绑定键的键值 | ||||||
|  |   function getPressedKeyCodes() { | ||||||
|  |     return _downKeys.slice(0); | ||||||
|  |   } | ||||||
|  |   function getPressedKeyString() { | ||||||
|  |     return _downKeys.map(c => getKey(c) || getModifier(c) || String.fromCharCode(c)); | ||||||
|  |   } | ||||||
|  |   function getAllKeyCodes() { | ||||||
|  |     const result = []; | ||||||
|  |     Object.keys(_handlers).forEach(k => { | ||||||
|  |       _handlers[k].forEach(_ref => { | ||||||
|  |         let { | ||||||
|  |           key, | ||||||
|  |           scope, | ||||||
|  |           mods, | ||||||
|  |           shortcut | ||||||
|  |         } = _ref; | ||||||
|  |         result.push({ | ||||||
|  |           scope, | ||||||
|  |           shortcut, | ||||||
|  |           mods, | ||||||
|  |           keys: key.split('+').map(v => code(v)) | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 表单控件控件判断 返回 Boolean | ||||||
|  |   // hotkey is effective only when filter return true | ||||||
|  |   function filter(event) { | ||||||
|  |     const target = event.target || event.srcElement; | ||||||
|  |     const { | ||||||
|  |       tagName | ||||||
|  |     } = target; | ||||||
|  |     let flag = true; | ||||||
|  |     const isInput = tagName === 'INPUT' && !['checkbox', 'radio', 'range', 'button', 'file', 'reset', 'submit', 'color'].includes(target.type); | ||||||
|  |     // ignore: isContentEditable === 'true', <input> and <textarea> when readOnly state is false, <select> | ||||||
|  |     if (target.isContentEditable || (isInput || tagName === 'TEXTAREA' || tagName === 'SELECT') && !target.readOnly) { | ||||||
|  |       flag = false; | ||||||
|  |     } | ||||||
|  |     return flag; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 判断摁下的键是否为某个键,返回true或者false | ||||||
|  |   function isPressed(keyCode) { | ||||||
|  |     if (typeof keyCode === 'string') { | ||||||
|  |       keyCode = code(keyCode); // 转换成键码 | ||||||
|  |     } | ||||||
|  |     return _downKeys.indexOf(keyCode) !== -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 循环删除handlers中的所有 scope(范围) | ||||||
|  |   function deleteScope(scope, newScope) { | ||||||
|  |     let handlers; | ||||||
|  |     let i; | ||||||
|  |  | ||||||
|  |     // 没有指定scope,获取scope | ||||||
|  |     if (!scope) scope = getScope(); | ||||||
|  |     for (const key in _handlers) { | ||||||
|  |       if (Object.prototype.hasOwnProperty.call(_handlers, key)) { | ||||||
|  |         handlers = _handlers[key]; | ||||||
|  |         for (i = 0; i < handlers.length;) { | ||||||
|  |           if (handlers[i].scope === scope) { | ||||||
|  |             const deleteItems = handlers.splice(i, 1); | ||||||
|  |             deleteItems.forEach(_ref2 => { | ||||||
|  |               let { | ||||||
|  |                 element | ||||||
|  |               } = _ref2; | ||||||
|  |               return removeKeyEvent(element); | ||||||
|  |             }); | ||||||
|  |           } else { | ||||||
|  |             i++; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 如果scope被删除,将scope重置为all | ||||||
|  |     if (getScope() === scope) setScope(newScope || 'all'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 清除修饰键 | ||||||
|  |   function clearModifier(event) { | ||||||
|  |     let key = event.keyCode || event.which || event.charCode; | ||||||
|  |     const i = _downKeys.indexOf(key); | ||||||
|  |  | ||||||
|  |     // 从列表中清除按压过的键 | ||||||
|  |     if (i >= 0) { | ||||||
|  |       _downKeys.splice(i, 1); | ||||||
|  |     } | ||||||
|  |     // 特殊处理 cmmand 键,在 cmmand 组合快捷键 keyup 只执行一次的问题 | ||||||
|  |     if (event.key && event.key.toLowerCase() === 'meta') { | ||||||
|  |       _downKeys.splice(0, _downKeys.length); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 修饰键 shiftKey altKey ctrlKey (command||metaKey) 清除 | ||||||
|  |     if (key === 93 || key === 224) key = 91; | ||||||
|  |     if (key in _mods) { | ||||||
|  |       _mods[key] = false; | ||||||
|  |  | ||||||
|  |       // 将修饰键重置为false | ||||||
|  |       for (const k in _modifier) if (_modifier[k] === key) hotkeys[k] = false; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function unbind(keysInfo) { | ||||||
|  |     // unbind(), unbind all keys | ||||||
|  |     if (typeof keysInfo === 'undefined') { | ||||||
|  |       Object.keys(_handlers).forEach(key => { | ||||||
|  |         Array.isArray(_handlers[key]) && _handlers[key].forEach(info => eachUnbind(info)); | ||||||
|  |         delete _handlers[key]; | ||||||
|  |       }); | ||||||
|  |       removeKeyEvent(null); | ||||||
|  |     } else if (Array.isArray(keysInfo)) { | ||||||
|  |       // support like : unbind([{key: 'ctrl+a', scope: 's1'}, {key: 'ctrl-a', scope: 's2', splitKey: '-'}]) | ||||||
|  |       keysInfo.forEach(info => { | ||||||
|  |         if (info.key) eachUnbind(info); | ||||||
|  |       }); | ||||||
|  |     } else if (typeof keysInfo === 'object') { | ||||||
|  |       // support like unbind({key: 'ctrl+a, ctrl+b', scope:'abc'}) | ||||||
|  |       if (keysInfo.key) eachUnbind(keysInfo); | ||||||
|  |     } else if (typeof keysInfo === 'string') { | ||||||
|  |       for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { | ||||||
|  |         args[_key - 1] = arguments[_key]; | ||||||
|  |       } | ||||||
|  |       // support old method | ||||||
|  |       // eslint-disable-line | ||||||
|  |       let [scope, method] = args; | ||||||
|  |       if (typeof scope === 'function') { | ||||||
|  |         method = scope; | ||||||
|  |         scope = ''; | ||||||
|  |       } | ||||||
|  |       eachUnbind({ | ||||||
|  |         key: keysInfo, | ||||||
|  |         scope, | ||||||
|  |         method, | ||||||
|  |         splitKey: '+' | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 解除绑定某个范围的快捷键 | ||||||
|  |   const eachUnbind = _ref3 => { | ||||||
|  |     let { | ||||||
|  |       key, | ||||||
|  |       scope, | ||||||
|  |       method, | ||||||
|  |       splitKey = '+' | ||||||
|  |     } = _ref3; | ||||||
|  |     const multipleKeys = getKeys(key); | ||||||
|  |     multipleKeys.forEach(originKey => { | ||||||
|  |       const unbindKeys = originKey.split(splitKey); | ||||||
|  |       const len = unbindKeys.length; | ||||||
|  |       const lastKey = unbindKeys[len - 1]; | ||||||
|  |       const keyCode = lastKey === '*' ? '*' : code(lastKey); | ||||||
|  |       if (!_handlers[keyCode]) return; | ||||||
|  |       // 判断是否传入范围,没有就获取范围 | ||||||
|  |       if (!scope) scope = getScope(); | ||||||
|  |       const mods = len > 1 ? getMods(_modifier, unbindKeys) : []; | ||||||
|  |       const unbindElements = []; | ||||||
|  |       _handlers[keyCode] = _handlers[keyCode].filter(record => { | ||||||
|  |         // 通过函数判断,是否解除绑定,函数相等直接返回 | ||||||
|  |         const isMatchingMethod = method ? record.method === method : true; | ||||||
|  |         const isUnbind = isMatchingMethod && record.scope === scope && compareArray(record.mods, mods); | ||||||
|  |         if (isUnbind) unbindElements.push(record.element); | ||||||
|  |         return !isUnbind; | ||||||
|  |       }); | ||||||
|  |       unbindElements.forEach(element => removeKeyEvent(element)); | ||||||
|  |     }); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // 对监听对应快捷键的回调函数进行处理 | ||||||
|  |   function eventHandler(event, handler, scope, element) { | ||||||
|  |     if (handler.element !== element) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     let modifiersMatch; | ||||||
|  |  | ||||||
|  |     // 看它是否在当前范围 | ||||||
|  |     if (handler.scope === scope || handler.scope === 'all') { | ||||||
|  |       // 检查是否匹配修饰符(如果有返回true) | ||||||
|  |       modifiersMatch = handler.mods.length > 0; | ||||||
|  |       for (const y in _mods) { | ||||||
|  |         if (Object.prototype.hasOwnProperty.call(_mods, y)) { | ||||||
|  |           if (!_mods[y] && handler.mods.indexOf(+y) > -1 || _mods[y] && handler.mods.indexOf(+y) === -1) { | ||||||
|  |             modifiersMatch = false; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // 调用处理程序,如果是修饰键不做处理 | ||||||
|  |       if (handler.mods.length === 0 && !_mods[16] && !_mods[18] && !_mods[17] && !_mods[91] || modifiersMatch || handler.shortcut === '*') { | ||||||
|  |         handler.keys = []; | ||||||
|  |         handler.keys = handler.keys.concat(_downKeys); | ||||||
|  |         if (handler.method(event, handler) === false) { | ||||||
|  |           if (event.preventDefault) event.preventDefault();else event.returnValue = false; | ||||||
|  |           if (event.stopPropagation) event.stopPropagation(); | ||||||
|  |           if (event.cancelBubble) event.cancelBubble = true; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 处理keydown事件 | ||||||
|  |   function dispatch(event, element) { | ||||||
|  |     const asterisk = _handlers['*']; | ||||||
|  |     let key = event.keyCode || event.which || event.charCode; | ||||||
|  |  | ||||||
|  |     // 表单控件过滤 默认表单控件不触发快捷键 | ||||||
|  |     if (!hotkeys.filter.call(this, event)) return; | ||||||
|  |  | ||||||
|  |     // Gecko(Firefox)的command键值224,在Webkit(Chrome)中保持一致 | ||||||
|  |     // Webkit左右 command 键值不一样 | ||||||
|  |     if (key === 93 || key === 224) key = 91; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Collect bound keys | ||||||
|  |      * If an Input Method Editor is processing key input and the event is keydown, return 229. | ||||||
|  |      * https://stackoverflow.com/questions/25043934/is-it-ok-to-ignore-keydown-events-with-keycode-229 | ||||||
|  |      * http://lists.w3.org/Archives/Public/www-dom/2010JulSep/att-0182/keyCode-spec.html | ||||||
|  |      */ | ||||||
|  |     if (_downKeys.indexOf(key) === -1 && key !== 229) _downKeys.push(key); | ||||||
|  |     /** | ||||||
|  |      * Jest test cases are required. | ||||||
|  |      * =============================== | ||||||
|  |      */ | ||||||
|  |     ['ctrlKey', 'altKey', 'shiftKey', 'metaKey'].forEach(keyName => { | ||||||
|  |       const keyNum = modifierMap[keyName]; | ||||||
|  |       if (event[keyName] && _downKeys.indexOf(keyNum) === -1) { | ||||||
|  |         _downKeys.push(keyNum); | ||||||
|  |       } else if (!event[keyName] && _downKeys.indexOf(keyNum) > -1) { | ||||||
|  |         _downKeys.splice(_downKeys.indexOf(keyNum), 1); | ||||||
|  |       } else if (keyName === 'metaKey' && event[keyName] && _downKeys.length === 3) { | ||||||
|  |         /** | ||||||
|  |          * Fix if Command is pressed: | ||||||
|  |          * =============================== | ||||||
|  |          */ | ||||||
|  |         if (!(event.ctrlKey || event.shiftKey || event.altKey)) { | ||||||
|  |           _downKeys = _downKeys.slice(_downKeys.indexOf(keyNum)); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     /** | ||||||
|  |      * ------------------------------- | ||||||
|  |      */ | ||||||
|  |  | ||||||
|  |     if (key in _mods) { | ||||||
|  |       _mods[key] = true; | ||||||
|  |  | ||||||
|  |       // 将特殊字符的key注册到 hotkeys 上 | ||||||
|  |       for (const k in _modifier) { | ||||||
|  |         if (_modifier[k] === key) hotkeys[k] = true; | ||||||
|  |       } | ||||||
|  |       if (!asterisk) return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 将 modifierMap 里面的修饰键绑定到 event 中 | ||||||
|  |     for (const e in _mods) { | ||||||
|  |       if (Object.prototype.hasOwnProperty.call(_mods, e)) { | ||||||
|  |         _mods[e] = event[modifierMap[e]]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     /** | ||||||
|  |      * https://github.com/jaywcjlove/hotkeys/pull/129 | ||||||
|  |      * This solves the issue in Firefox on Windows where hotkeys corresponding to special characters would not trigger. | ||||||
|  |      * An example of this is ctrl+alt+m on a Swedish keyboard which is used to type μ. | ||||||
|  |      * Browser support: https://caniuse.com/#feat=keyboardevent-getmodifierstate | ||||||
|  |      */ | ||||||
|  |     if (event.getModifierState && !(event.altKey && !event.ctrlKey) && event.getModifierState('AltGraph')) { | ||||||
|  |       if (_downKeys.indexOf(17) === -1) { | ||||||
|  |         _downKeys.push(17); | ||||||
|  |       } | ||||||
|  |       if (_downKeys.indexOf(18) === -1) { | ||||||
|  |         _downKeys.push(18); | ||||||
|  |       } | ||||||
|  |       _mods[17] = true; | ||||||
|  |       _mods[18] = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // 获取范围 默认为 `all` | ||||||
|  |     const scope = getScope(); | ||||||
|  |     // 对任何快捷键都需要做的处理 | ||||||
|  |     if (asterisk) { | ||||||
|  |       for (let i = 0; i < asterisk.length; i++) { | ||||||
|  |         if (asterisk[i].scope === scope && (event.type === 'keydown' && asterisk[i].keydown || event.type === 'keyup' && asterisk[i].keyup)) { | ||||||
|  |           eventHandler(event, asterisk[i], scope, element); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     // key 不在 _handlers 中返回 | ||||||
|  |     if (!(key in _handlers)) return; | ||||||
|  |     const handlerKey = _handlers[key]; | ||||||
|  |     const keyLen = handlerKey.length; | ||||||
|  |     for (let i = 0; i < keyLen; i++) { | ||||||
|  |       if (event.type === 'keydown' && handlerKey[i].keydown || event.type === 'keyup' && handlerKey[i].keyup) { | ||||||
|  |         if (handlerKey[i].key) { | ||||||
|  |           const record = handlerKey[i]; | ||||||
|  |           const { | ||||||
|  |             splitKey | ||||||
|  |           } = record; | ||||||
|  |           const keyShortcut = record.key.split(splitKey); | ||||||
|  |           const _downKeysCurrent = []; // 记录当前按键键值 | ||||||
|  |           for (let a = 0; a < keyShortcut.length; a++) { | ||||||
|  |             _downKeysCurrent.push(code(keyShortcut[a])); | ||||||
|  |           } | ||||||
|  |           if (_downKeysCurrent.sort().join('') === _downKeys.sort().join('')) { | ||||||
|  |             // 找到处理内容 | ||||||
|  |             eventHandler(event, record, scope, element); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function hotkeys(key, option, method) { | ||||||
|  |     _downKeys = []; | ||||||
|  |     const keys = getKeys(key); // 需要处理的快捷键列表 | ||||||
|  |     let mods = []; | ||||||
|  |     let scope = 'all'; // scope默认为all,所有范围都有效 | ||||||
|  |     let element = document; // 快捷键事件绑定节点 | ||||||
|  |     let i = 0; | ||||||
|  |     let keyup = false; | ||||||
|  |     let keydown = true; | ||||||
|  |     let splitKey = '+'; | ||||||
|  |     let capture = false; | ||||||
|  |     let single = false; // 单个callback | ||||||
|  |  | ||||||
|  |     // 对为设定范围的判断 | ||||||
|  |     if (method === undefined && typeof option === 'function') { | ||||||
|  |       method = option; | ||||||
|  |     } | ||||||
|  |     if (Object.prototype.toString.call(option) === '[object Object]') { | ||||||
|  |       if (option.scope) scope = option.scope; // eslint-disable-line | ||||||
|  |       if (option.element) element = option.element; // eslint-disable-line | ||||||
|  |       if (option.keyup) keyup = option.keyup; // eslint-disable-line | ||||||
|  |       if (option.keydown !== undefined) keydown = option.keydown; // eslint-disable-line | ||||||
|  |       if (option.capture !== undefined) capture = option.capture; // eslint-disable-line | ||||||
|  |       if (typeof option.splitKey === 'string') splitKey = option.splitKey; // eslint-disable-line | ||||||
|  |       if (option.single === true) single = true; // eslint-disable-line | ||||||
|  |     } | ||||||
|  |     if (typeof option === 'string') scope = option; | ||||||
|  |  | ||||||
|  |     // 如果只允许单个callback,先unbind | ||||||
|  |     if (single) unbind(key, scope); | ||||||
|  |  | ||||||
|  |     // 对于每个快捷键进行处理 | ||||||
|  |     for (; i < keys.length; i++) { | ||||||
|  |       key = keys[i].split(splitKey); // 按键列表 | ||||||
|  |       mods = []; | ||||||
|  |  | ||||||
|  |       // 如果是组合快捷键取得组合快捷键 | ||||||
|  |       if (key.length > 1) mods = getMods(_modifier, key); | ||||||
|  |  | ||||||
|  |       // 将非修饰键转化为键码 | ||||||
|  |       key = key[key.length - 1]; | ||||||
|  |       key = key === '*' ? '*' : code(key); // *表示匹配所有快捷键 | ||||||
|  |  | ||||||
|  |       // 判断key是否在_handlers中,不在就赋一个空数组 | ||||||
|  |       if (!(key in _handlers)) _handlers[key] = []; | ||||||
|  |       _handlers[key].push({ | ||||||
|  |         keyup, | ||||||
|  |         keydown, | ||||||
|  |         scope, | ||||||
|  |         mods, | ||||||
|  |         shortcut: keys[i], | ||||||
|  |         method, | ||||||
|  |         key: keys[i], | ||||||
|  |         splitKey, | ||||||
|  |         element | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     // 在全局document上设置快捷键 | ||||||
|  |     if (typeof element !== 'undefined' && window) { | ||||||
|  |       if (!elementEventMap.has(element)) { | ||||||
|  |         const keydownListener = function () { | ||||||
|  |           let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |           return dispatch(event, element); | ||||||
|  |         }; | ||||||
|  |         const keyupListenr = function () { | ||||||
|  |           let event = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : window.event; | ||||||
|  |           dispatch(event, element); | ||||||
|  |           clearModifier(event); | ||||||
|  |         }; | ||||||
|  |         elementEventMap.set(element, { | ||||||
|  |           keydownListener, | ||||||
|  |           keyupListenr, | ||||||
|  |           capture | ||||||
|  |         }); | ||||||
|  |         addEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |         addEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |       } | ||||||
|  |       if (!winListendFocus) { | ||||||
|  |         const listener = () => { | ||||||
|  |           _downKeys = []; | ||||||
|  |         }; | ||||||
|  |         winListendFocus = { | ||||||
|  |           listener, | ||||||
|  |           capture | ||||||
|  |         }; | ||||||
|  |         addEvent(window, 'focus', listener, capture); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function trigger(shortcut) { | ||||||
|  |     let scope = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'all'; | ||||||
|  |     Object.keys(_handlers).forEach(key => { | ||||||
|  |       const dataList = _handlers[key].filter(item => item.scope === scope && item.shortcut === shortcut); | ||||||
|  |       dataList.forEach(data => { | ||||||
|  |         if (data && data.method) { | ||||||
|  |           data.method(); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // 销毁事件,unbind之后判断element上是否还有键盘快捷键,如果没有移除监听 | ||||||
|  |   function removeKeyEvent(element) { | ||||||
|  |     const values = Object.values(_handlers).flat(); | ||||||
|  |     const findindex = values.findIndex(_ref4 => { | ||||||
|  |       let { | ||||||
|  |         element: el | ||||||
|  |       } = _ref4; | ||||||
|  |       return el === element; | ||||||
|  |     }); | ||||||
|  |     if (findindex < 0) { | ||||||
|  |       const { | ||||||
|  |         keydownListener, | ||||||
|  |         keyupListenr, | ||||||
|  |         capture | ||||||
|  |       } = elementEventMap.get(element) || {}; | ||||||
|  |       if (keydownListener && keyupListenr) { | ||||||
|  |         removeEvent(element, 'keyup', keyupListenr, capture); | ||||||
|  |         removeEvent(element, 'keydown', keydownListener, capture); | ||||||
|  |         elementEventMap.delete(element); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (values.length <= 0 || elementEventMap.size <= 0) { | ||||||
|  |       // 移除所有的元素上的监听 | ||||||
|  |       const eventKeys = Object.keys(elementEventMap); | ||||||
|  |       eventKeys.forEach(el => { | ||||||
|  |         const { | ||||||
|  |           keydownListener, | ||||||
|  |           keyupListenr, | ||||||
|  |           capture | ||||||
|  |         } = elementEventMap.get(el) || {}; | ||||||
|  |         if (keydownListener && keyupListenr) { | ||||||
|  |           removeEvent(el, 'keyup', keyupListenr, capture); | ||||||
|  |           removeEvent(el, 'keydown', keydownListener, capture); | ||||||
|  |           elementEventMap.delete(el); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |       // 清空 elementEventMap | ||||||
|  |       elementEventMap.clear(); | ||||||
|  |       // 清空 _handlers | ||||||
|  |       Object.keys(_handlers).forEach(key => delete _handlers[key]); | ||||||
|  |       // 移除window上的focus监听 | ||||||
|  |       if (winListendFocus) { | ||||||
|  |         const { | ||||||
|  |           listener, | ||||||
|  |           capture | ||||||
|  |         } = winListendFocus; | ||||||
|  |         removeEvent(window, 'focus', listener, capture); | ||||||
|  |         winListendFocus = null; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   const _api = { | ||||||
|  |     getPressedKeyString, | ||||||
|  |     setScope, | ||||||
|  |     getScope, | ||||||
|  |     deleteScope, | ||||||
|  |     getPressedKeyCodes, | ||||||
|  |     getAllKeyCodes, | ||||||
|  |     isPressed, | ||||||
|  |     filter, | ||||||
|  |     trigger, | ||||||
|  |     unbind, | ||||||
|  |     keyMap: _keyMap, | ||||||
|  |     modifier: _modifier, | ||||||
|  |     modifierMap | ||||||
|  |   }; | ||||||
|  |   for (const a in _api) { | ||||||
|  |     if (Object.prototype.hasOwnProperty.call(_api, a)) { | ||||||
|  |       hotkeys[a] = _api[a]; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (typeof window !== 'undefined') { | ||||||
|  |     const _hotkeys = window.hotkeys; | ||||||
|  |     hotkeys.noConflict = deep => { | ||||||
|  |       if (deep && window.hotkeys === hotkeys) { | ||||||
|  |         window.hotkeys = _hotkeys; | ||||||
|  |       } | ||||||
|  |       return hotkeys; | ||||||
|  |     }; | ||||||
|  |     window.hotkeys = hotkeys; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return hotkeys; | ||||||
|  |  | ||||||
|  | })); | ||||||
							
								
								
									
										2
									
								
								term/pkg/hotkeys-js/dist/hotkeys.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								term/pkg/hotkeys-js/dist/hotkeys.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										18
									
								
								term/pkg/hotkeys-js/dist/terminal.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								term/pkg/hotkeys-js/dist/terminal.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | hotkeys('ctrl+enter', function() { | ||||||
|  | 	let e = document.querySelector('home-layout'); | ||||||
|  | 	e.style.display = "block"; | ||||||
|  | 	var ele = document.getElementById('aiterm'); | ||||||
|  | 	var win = document.getElementById('window'); | ||||||
|  | 	var svg = document.getElementById('aisvg'); | ||||||
|  | 	var term = document.getElementById('terminal'); | ||||||
|  | 	term.click(); | ||||||
|  | 	//var par = document.getElementById('particles-js'); | ||||||
|  | 	//par.style.display = 'block';  | ||||||
|  | 	if (ele.style.display == 'none') { | ||||||
|  | 		ele.style.display = 'block';  | ||||||
|  | 		svg.style.display = 'none';  | ||||||
|  | 	} else { | ||||||
|  | 		ele.style.display = 'none'; | ||||||
|  | 		svg.style.display = 'block';  | ||||||
|  | 	} | ||||||
|  | }); | ||||||
							
								
								
									
										181
									
								
								term/pkg/hotkeys-js/index.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								term/pkg/hotkeys-js/index.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,181 @@ | |||||||
|  | export interface HotkeysEvent { | ||||||
|  |   key: string; | ||||||
|  |   keys: number[]; | ||||||
|  |   method: KeyHandler; | ||||||
|  |   mods: number[]; | ||||||
|  |   scope: string; | ||||||
|  |   shortcut: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface KeyHandler { | ||||||
|  |   (keyboardEvent: KeyboardEvent, hotkeysEvent: HotkeysEvent): void | boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type Options = { | ||||||
|  |   scope?: string; | ||||||
|  |   element?: HTMLElement | null; | ||||||
|  |   keyup?: boolean | null; | ||||||
|  |   keydown?: boolean | null; | ||||||
|  |   capture?: boolean | ||||||
|  |   splitKey?: string; | ||||||
|  |   single?: boolean; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface Hotkeys { | ||||||
|  |   (key: string, method: KeyHandler): void; | ||||||
|  |   (key: string, scope: string, method: KeyHandler): void; | ||||||
|  |   (key: string, options: Options, method: KeyHandler): void; | ||||||
|  |  | ||||||
|  |   shift: boolean; | ||||||
|  |   ctrl: boolean; | ||||||
|  |   alt: boolean; | ||||||
|  |   option: boolean; | ||||||
|  |   control: boolean; | ||||||
|  |   cmd: boolean; | ||||||
|  |   command: boolean; | ||||||
|  |  | ||||||
|  |   keyMap: Record<string, number>; | ||||||
|  |   modifier: Record<string, number>; | ||||||
|  |   modifierMap: Record<string, number | string>; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Use the `hotkeys.setScope` method to set scope. There can only be one active scope besides 'all'.  By default 'all' is always active. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * // Define shortcuts with a scope | ||||||
|  |    * hotkeys('ctrl+o, ctrl+alt+enter', 'issues', function() { | ||||||
|  |    *   console.log('do something'); | ||||||
|  |    * }); | ||||||
|  |    * hotkeys('o, enter', 'files', function() { | ||||||
|  |    *   console.log('do something else'); | ||||||
|  |    * }); | ||||||
|  |    * | ||||||
|  |    * // Set the scope (only 'all' and 'issues' shortcuts will be honored) | ||||||
|  |    * hotkeys.setScope('issues'); // default scope is 'all' | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   setScope(scopeName: string): void; | ||||||
|  |   /** | ||||||
|  |    * Use the `hotkeys.getScope` method to get scope. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys.getScope(); | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   getScope(): string; | ||||||
|  |   /** | ||||||
|  |    * Use the `hotkeys.deleteScope` method to delete a scope. This will also remove all associated hotkeys with it. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys.deleteScope('issues'); | ||||||
|  |    * ``` | ||||||
|  |    * You can use second argument, if need set new scope after deleting. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys.deleteScope('issues', 'newScopeName'); | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   deleteScope(scopeName: string, newScopeName?: string): void; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Relinquish HotKeys’s control of the `hotkeys` variable. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * var k = hotkeys.noConflict(); | ||||||
|  |    * k('a', function() { | ||||||
|  |    *   console.log("do something") | ||||||
|  |    * }); | ||||||
|  |    * | ||||||
|  |    * hotkeys() | ||||||
|  |    * // -->Uncaught TypeError: hotkeys is not a function(anonymous function) | ||||||
|  |    * // @ VM2170:2InjectedScript._evaluateOn | ||||||
|  |    * // @ VM2165:883InjectedScript._evaluateAndWrap | ||||||
|  |    * // @ VM2165:816InjectedScript.evaluate @ VM2165:682 | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   noConflict(): Hotkeys; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * trigger shortcut key event | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys.trigger('ctrl+o'); | ||||||
|  |    * hotkeys.trigger('ctrl+o', 'scope2'); | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   trigger(shortcut: string, scope?: string): void; | ||||||
|  |  | ||||||
|  |   unbind(key?: string): void; | ||||||
|  |   unbind(key: string, scopeName: string): void; | ||||||
|  |   unbind(key: string, scopeName: string, method: KeyHandler): void; | ||||||
|  |   unbind(key: string, method: KeyHandler): void; | ||||||
|  |  | ||||||
|  |   /** For example, `hotkeys.isPressed(77)` is true if the `M` key is currently pressed. */ | ||||||
|  |   isPressed(keyCode: number): boolean; | ||||||
|  |   /** For example, `hotkeys.isPressed('m')` is true if the `M` key is currently pressed. */ | ||||||
|  |   isPressed(keyCode: string): boolean; | ||||||
|  |   /** | ||||||
|  |    * Returns an array of key codes currently pressed. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |    *   console.log(hotkeys.getPressedKeyCodes()); //=> [17, 65] or [70] | ||||||
|  |    * }) | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   getPressedKeyCodes(): number[]; | ||||||
|  |   /** | ||||||
|  |    * Returns an array of key codes currently pressed. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |    *   console.log(hotkeys.getPressedKeyString()); //=> ['⌘', '⌃', '⇧', 'A', 'F'] | ||||||
|  |    * }) | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   getPressedKeyString(): string[]; | ||||||
|  |   /** | ||||||
|  |    * Get a list of all registration codes. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys('command+ctrl+shift+a,f', function() { | ||||||
|  |    *   console.log(hotkeys.getAllKeyCodes()); | ||||||
|  |    *   // [ | ||||||
|  |    *   //   { scope: 'all', shortcut: 'command+ctrl+shift+a', mods: [91, 17, 16], keys: [91, 17, 16, 65] }, | ||||||
|  |    *   //   { scope: 'all', shortcut: 'f', mods: [], keys: [42] } | ||||||
|  |    *   // ] | ||||||
|  |    * }) | ||||||
|  |    * ``` | ||||||
|  |    * | ||||||
|  |    */ | ||||||
|  |   getAllKeyCodes(): Omit<HotkeysEvent, 'method' | 'key'>[]; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * By default hotkeys are not enabled for `INPUT` `SELECT` `TEXTAREA` elements. | ||||||
|  |    * `Hotkeys.filter` to return to the `true` shortcut keys set to play a role, | ||||||
|  |    * `false` shortcut keys set up failure. | ||||||
|  |    * | ||||||
|  |    * ```js | ||||||
|  |    * hotkeys.filter = function(event){ | ||||||
|  |    *   return true; | ||||||
|  |    * } | ||||||
|  |    * //How to add the filter to edit labels. <div contentEditable="true"></div> | ||||||
|  |    * //"contentEditable" Older browsers that do not support drops | ||||||
|  |    * hotkeys.filter = function(event) { | ||||||
|  |    *   var target = event.target || event.srcElement; | ||||||
|  |    *   var tagName = target.tagName; | ||||||
|  |    *   return !(target.isContentEditable || tagName == 'INPUT' || tagName == 'SELECT' || tagName == 'TEXTAREA'); | ||||||
|  |    * } | ||||||
|  |    * | ||||||
|  |    * hotkeys.filter = function(event){ | ||||||
|  |    *   var tagName = (event.target || event.srcElement).tagName; | ||||||
|  |    *   hotkeys.setScope(/^(INPUT|TEXTAREA|SELECT)$/.test(tagName) ? 'input' : 'other'); | ||||||
|  |    *   return true; | ||||||
|  |    * } | ||||||
|  |    * ``` | ||||||
|  |    */ | ||||||
|  |   filter(event: KeyboardEvent): boolean; | ||||||
|  | } | ||||||
|  | // https://github.com/eiriklv/react-masonry-component/issues/57 | ||||||
|  | declare var hotkeys: Hotkeys; | ||||||
|  | export default hotkeys; | ||||||
							
								
								
									
										7
									
								
								term/pkg/hotkeys-js/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								term/pkg/hotkeys-js/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | if (process.env.NODE_ENV === 'production') { | ||||||
|  |   // eslint-disable-next-line global-require | ||||||
|  |   module.exports = require('./dist/hotkeys.common.min.js'); | ||||||
|  | } else { | ||||||
|  |   // eslint-disable-next-line global-require | ||||||
|  |   module.exports = require('./dist/hotkeys.common.js'); | ||||||
|  | } | ||||||
							
								
								
									
										32
									
								
								term/pkg/hotkeys-js/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								term/pkg/hotkeys-js/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | { | ||||||
|  |   "name": "hotkeys-js", | ||||||
|  |   "description": "A simple micro-library for defining and dispatching keyboard shortcuts. It has no dependencies.", | ||||||
|  |   "version": "3.13.7", | ||||||
|  |   "main": "index.js", | ||||||
|  |   "types": "index.d.ts", | ||||||
|  |   "module": "dist/hotkeys.esm.js", | ||||||
|  |   "files": [ | ||||||
|  |     "index.d.ts", | ||||||
|  |     "dist", | ||||||
|  |     "doc" | ||||||
|  |   ], | ||||||
|  |   "keywords": [ | ||||||
|  |     "hotkey", | ||||||
|  |     "hotkeys", | ||||||
|  |     "hotkeys-js", | ||||||
|  |     "hotkeysjs", | ||||||
|  |     "key", | ||||||
|  |     "keys", | ||||||
|  |     "keyboard", | ||||||
|  |     "shortcuts", | ||||||
|  |     "keypress" | ||||||
|  |   ], | ||||||
|  |   "author": "kenny wong <wowohoo@qq.com>", | ||||||
|  |   "license": "MIT", | ||||||
|  |   "homepage": "https://jaywcjlove.github.io/hotkeys-js", | ||||||
|  |   "funding": "https://jaywcjlove.github.io/#/sponsor", | ||||||
|  |   "repository": { | ||||||
|  |     "type": "git", | ||||||
|  |     "url": "https://github.com/jaywcjlove/hotkeys-js.git" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								term/pkg/particles/config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								term/pkg/particles/config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | particlesJS("particles-js", {"particles":{"number":{"value":160,"density":{"enable":false,"value_area":800}},"color":{"value":"#fff700"},"shape":{"type":"circle","stroke":{"width":0,"color":"#000000"},"polygon":{"nb_sides":5},"image":{"src":"","width":120,"height":80}},"opacity":{"value":1,"random":true,"anim":{"enable":true,"speed":0.16240621041348632,"opacity_min":0,"sync":true}},"size":{"value":3,"random":true,"anim":{"enable":true,"speed":0,"size_min":0,"sync":false}},"line_linked":{"enable":true,"distance":128.27296486924183,"color":"#fff700","opacity":0.01,"width":0},"move":{"enable":true,"speed":1,"direction":"none","random":true,"straight":false,"out_mode":"out","bounce":false,"attract":{"enable":false,"rotateX":600,"rotateY":600}}},"interactivity":{"detect_on":"canvas","events":{"onhover":{"enable":true,"mode":"repulse"},"onclick":{"enable":true,"mode":"push"},"resize":true},"modes":{"grab":{"distance":400,"line_linked":{"opacity":1}},"bubble":{"distance":250,"size":0,"duration":2,"opacity":0,"speed":3},"repulse":{"distance":400,"duration":0.4},"push":{"particles_nb":4},"remove":{"particles_nb":2}}},"retina_detect":false});var count_particles, stats, update; stats = new Stats; stats.setMode(0); stats.domElement.style.position = 'absolute'; stats.domElement.style.left = '0px'; stats.domElement.style.top = '0px'; document.body.appendChild(stats.domElement); count_particles = document.querySelector('.js-count-particles'); update = function() { stats.begin(); stats.end(); if (window.pJSDom[0].pJS.particles && window.pJSDom[0].pJS.particles.array) { count_particles.innerText = window.pJSDom[0].pJS.particles.array.length; } requestAnimationFrame(update); }; requestAnimationFrame(update);; | ||||||
							
								
								
									
										34
									
								
								term/pkg/particles/particles.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								term/pkg/particles/particles.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | canvas{ | ||||||
|  | 	display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #particles-js{ | ||||||
|  | 	display: none; | ||||||
|  | 	width:100%; | ||||||
|  | 	position:absolute; | ||||||
|  | 	background-color: #313131; | ||||||
|  | } | ||||||
|  | .count-particles{ | ||||||
|  | 	background: #000022; | ||||||
|  | 	position: absolute; | ||||||
|  | 	color: #13E8E9; | ||||||
|  | 	font-size: .8em; | ||||||
|  | 	text-align: left; | ||||||
|  | 	text-indent: 4px; | ||||||
|  | 	line-height: 14px; | ||||||
|  | 	padding-bottom: 2px; | ||||||
|  | 	font-family: Helvetica, Arial, sans-serif; | ||||||
|  | 	font-weight: bold; | ||||||
|  | } | ||||||
|  | .js-count-particles{ | ||||||
|  | 	font-size: 1.1em; | ||||||
|  | } | ||||||
|  | #stats, .count-particles{ | ||||||
|  | 	-webkit-user-select: none; | ||||||
|  | } | ||||||
|  | #stats{ | ||||||
|  | 	overflow: hidden; | ||||||
|  | } | ||||||
|  | .count-particles{ | ||||||
|  | 	border-radius: 0 0 3px 3px; | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								term/pkg/particles/particles.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								term/pkg/particles/particles.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										89
									
								
								term/pkg/particles/stats.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								term/pkg/particles/stats.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html> | ||||||
|  |   <head> | ||||||
|  |     <meta http-equiv="Content-type" content="text/html; charset=utf-8"> | ||||||
|  |     <meta http-equiv="Content-Security-Policy" content="default-src 'none'; style-src 'unsafe-inline'; img-src data:; connect-src 'self'"> | ||||||
|  |     <title>Page not found · GitHub Pages</title> | ||||||
|  |     <style type="text/css" media="screen"> | ||||||
|  |       body { | ||||||
|  |         background-color: #f1f1f1; | ||||||
|  |         margin: 0; | ||||||
|  |         font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .container { margin: 50px auto 40px auto; width: 600px; text-align: center; } | ||||||
|  |  | ||||||
|  |       a { color: #4183c4; text-decoration: none; } | ||||||
|  |       a:hover { text-decoration: underline; } | ||||||
|  |  | ||||||
|  |       h1 { width: 800px; position:relative; left: -100px; letter-spacing: -1px; line-height: 60px; font-size: 60px; font-weight: 100; margin: 0px 0 50px 0; text-shadow: 0 1px 0 #fff; } | ||||||
|  |       p { color: rgba(0, 0, 0, 0.5); margin: 20px 0; line-height: 1.6; } | ||||||
|  |  | ||||||
|  |       ul { list-style: none; margin: 25px 0; padding: 0; } | ||||||
|  |       li { display: table-cell; font-weight: bold; width: 1%; } | ||||||
|  |  | ||||||
|  |       .logo { display: inline-block; margin-top: 35px; } | ||||||
|  |       .logo-img-2x { display: none; } | ||||||
|  |       @media | ||||||
|  |       only screen and (-webkit-min-device-pixel-ratio: 2), | ||||||
|  |       only screen and (   min--moz-device-pixel-ratio: 2), | ||||||
|  |       only screen and (     -o-min-device-pixel-ratio: 2/1), | ||||||
|  |       only screen and (        min-device-pixel-ratio: 2), | ||||||
|  |       only screen and (                min-resolution: 192dpi), | ||||||
|  |       only screen and (                min-resolution: 2dppx) { | ||||||
|  |         .logo-img-1x { display: none; } | ||||||
|  |         .logo-img-2x { display: inline-block; } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       #suggestions { | ||||||
|  |         margin-top: 35px; | ||||||
|  |         color: #ccc; | ||||||
|  |       } | ||||||
|  |       #suggestions a { | ||||||
|  |         color: #666666; | ||||||
|  |         font-weight: 200; | ||||||
|  |         font-size: 14px; | ||||||
|  |         margin: 0 10px; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |     </style> | ||||||
|  |   </head> | ||||||
|  |   <body> | ||||||
|  |  | ||||||
|  |     <div class="container"> | ||||||
|  |  | ||||||
|  |       <h1>404</h1> | ||||||
|  |       <p><strong>File not found</strong></p> | ||||||
|  |  | ||||||
|  |       <p> | ||||||
|  |         The site configured at this address does not | ||||||
|  |         contain the requested file. | ||||||
|  |       </p> | ||||||
|  |  | ||||||
|  |       <p> | ||||||
|  |         If this is your site, make sure that the filename case matches the URL | ||||||
|  |         as well as any file permissions.<br> | ||||||
|  |         For root URLs (like <code>http://example.com/</code>) you must provide an | ||||||
|  |         <code>index.html</code> file. | ||||||
|  |       </p> | ||||||
|  |  | ||||||
|  |       <p> | ||||||
|  |         <a href="https://help.github.com/pages/">Read the full documentation</a> | ||||||
|  |         for more information about using <strong>GitHub Pages</strong>. | ||||||
|  |       </p> | ||||||
|  |  | ||||||
|  |       <div id="suggestions"> | ||||||
|  |         <a href="https://githubstatus.com">GitHub Status</a> — | ||||||
|  |         <a href="https://twitter.com/githubstatus">@githubstatus</a> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|  |       <a href="/" class="logo logo-img-1x"> | ||||||
|  |         <img width="32" height="32" title="" alt="" src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyRpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoTWFjaW50b3NoKSIgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpFMTZCRDY3REIzRjAxMUUyQUQzREIxQzRENUFFNUM5NiIgeG1wTU06RG9jdW1lbnRJRD0ieG1wLmRpZDpFMTZCRDY3RUIzRjAxMUUyQUQzREIxQzRENUFFNUM5NiI+IDx4bXBNTTpEZXJpdmVkRnJvbSBzdFJlZjppbnN0YW5jZUlEPSJ4bXAuaWlkOkUxNkJENjdCQjNGMDExRTJBRDNEQjFDNEQ1QUU1Qzk2IiBzdFJlZjpkb2N1bWVudElEPSJ4bXAuZGlkOkUxNkJENjdDQjNGMDExRTJBRDNEQjFDNEQ1QUU1Qzk2Ii8+IDwvcmRmOkRlc2NyaXB0aW9uPiA8L3JkZjpSREY+IDwveDp4bXBtZXRhPiA8P3hwYWNrZXQgZW5kPSJyIj8+SM9MCAAAA+5JREFUeNrEV11Ik1EY3s4+ddOp29Q5b0opCgKFsoKoi5Kg6CIhuwi6zLJLoYLopq4qsKKgi4i6CYIoU/q5iDAKs6syoS76IRWtyJ+p7cdt7sf1PGOD+e0c3dygAx/67ZzzPM95/877GYdHRg3ZjMXFxepQKNS6sLCwJxqNNuFpiMfjVs4ZjUa/pmmjeD6VlJS8NpvNT4QQ7mxwjSsJiEQim/1+/9lgMHgIr5ohuxG1WCw9Vqv1clFR0dCqBODElV6v90ogEDjGdYbVjXhpaendioqK07CIR7ZAqE49PT09BPL2PMgTByQGsYiZlQD4uMXtdr+JxWINhgINYhGT2MsKgMrm2dnZXgRXhaHAg5jEJodUAHxux4LudHJE9RdEdA+i3Juz7bGHe4mhE9FNrgwBCLirMFV9Okh5eflFh8PR5nK5nDabrR2BNJlKO0T35+Li4n4+/J+/JQCxhmu5h3uJoXNHPbmWZAHMshWB8l5/ipqammaAf0zPDDx1ONV3vurdidqwAQL+pEc8sLcAe1CCvQ3YHxIW8Pl85xSWNC1hADDIv0rIE/o4J0k3kww4xSlwIhcq3EFFOm7KN/hUGOQkt0CFa5WpNJlMvxBEz/IVQAxg/ZRZl9wiHA63yDYieM7DnLP5CiAGsC7I5sgtYKJGWe2A8seFqgFJrJjEPY1Cn3pJ8/9W1e5VWsFDTEmFrBcoDhZJEQkXuhICMyKpjhahqN21hRYATKfUOlDmkygrR4o4C0VOLGJKrOITKB4jijzdXygBKixyC5TDQdnk/Pz8qRw6oOWGlsTKGOQW6OH6FBWsyePxdOXLTgxiyebILZCjz+GLgMIKnXNzc49YMlcRdHXcSwxFVgTInQhC9G33UhNoJLuqq6t345p9y3eUy8OTk5PjAHuI9uo4b07FBaOhsu0A4Unc+T1TU1Nj3KsSSE5yJ65jqF2DDd8QqWYmAZrIM2VlZTdnZmb6AbpdV9V6ec9znf5Q7HjYumdRE0JOp3MjitO4SFa+cZz8Umqe3TCbSLvdfkR/kWDdNQl5InuTcysOcpFT35ZrbBxx4p3JAHlZVVW1D/634VRt+FvLBgK/v5LV9WS+10xMTEwtRw7XvqOL+e2Q8V3AYIOIAXQ26/heWVnZCVfcyKHg2CBgTpmPmjYM8l24GyaUHyaIh7XwfR9ErE8qHoDfn2LTNAVC0HX6MFcBIP8Bi+6F6cdW/DICkANRfx99fEYFQ7Nph5i/uQiA214gno7K+guhaiKg9gC62+M8eR7XsBsYJ4ilam60Fb7r7uAj8wFyuwM1oIOWgfmDy6RXEEQzJMPe23DXrVS7rtyD3Df8z/FPgAEAzWU5Ku59ZAUAAAAASUVORK5CYII="> | ||||||
|  |       </a> | ||||||
|  |  | ||||||
|  |       <a href="/" class="logo logo-img-2x"> | ||||||
|  |         <img width="32" height="32" title="" alt="" src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyRpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoTWFjaW50b3NoKSIgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpEQUM1QkUxRUI0MUMxMUUyQUQzREIxQzRENUFFNUM5NiIgeG1wTU06RG9jdW1lbnRJRD0ieG1wLmRpZDpEQUM1QkUxRkI0MUMxMUUyQUQzREIxQzRENUFFNUM5NiI+IDx4bXBNTTpEZXJpdmVkRnJvbSBzdFJlZjppbnN0YW5jZUlEPSJ4bXAuaWlkOkUxNkJENjdGQjNGMDExRTJBRDNEQjFDNEQ1QUU1Qzk2IiBzdFJlZjpkb2N1bWVudElEPSJ4bXAuZGlkOkUxNkJENjgwQjNGMDExRTJBRDNEQjFDNEQ1QUU1Qzk2Ii8+IDwvcmRmOkRlc2NyaXB0aW9uPiA8L3JkZjpSREY+IDwveDp4bXBtZXRhPiA8P3hwYWNrZXQgZW5kPSJyIj8+hfPRaQAAB6lJREFUeNrsW2mME2UYbodtt+2222u35QheoCCYGBQligIJgkZJNPzgigoaTEj8AdFEMfADfyABkgWiiWcieK4S+QOiHAYUj2hMNKgYlEujpNttu9vttbvdw+chU1K6M535pt3ubHCSyezR+b73eb73+t7vrfXsufOW4bz6+vom9/b23ovnNNw34b5xYGAgODg46Mbt4mesVmsWd1qSpHhdXd2fuP/Afcput5/A88xwymcdBgLqenp6FuRyuWV4zu/v759QyWBjxoz5t76+/gun09mK5xFyakoCAPSaTCazNpvNPoYVbh6O1YKGRF0u13sNDQ27QMzfpiAAKj0lnU6/gBVfAZW2WWpwwVzy0IgP3G73FpjI6REhAGA9qVRqA1b9mVoBVyIC2tDi8Xg24+dUzQiAbS/s7Ox8G2o/3mKCC+Zw0efzPQEfcVjYrARX3dbV1bUtHo8fMgt42f+Mp0yUTVQbdWsAHVsikdiHkHaPxcQXQufXgUBgMRxme9U0AAxfH4vFvjM7eF6UkbJS5qoQwEQGA57Ac5JllFyUVZZ5ckUEgMVxsK2jlSYzI+QXJsiyjzNEAJyJAzb/KQa41jJKL8pODMQiTEAymXw5n8/P0IjD3bh7Rgog59aanxiIRTVvV/oj0tnHca/WMrVwODwB3raTGxzkBg/gnZVapFV62Wy2n5AO70HM/5wbJ0QnXyQSaVPDIuNZzY0V3ntHMwxiwHA0Gj2Np7ecIBDgaDAYXKCQJM1DhrgJ3nhulcPbl8j4NmHe46X/g60fwbz3aewjkqFQaAqebWU1AOqyQwt8Id6qEHMc97zu7u7FGGsn7HAiVuosVw7P35C1nccdgSCxop1dHeZswmfHMnxBo6ZTk+jN8dl/vF7vWofDsa+MLN9oEUBMxOb3+1eoEsBVw6Zmua49r8YmhAKDiEPcMwBsxMiqQ+ixzPFxZyqRpXARG/YOr1ObFJ0gUskXBbamcR1OKmMUvDxHRAu8/LmY3jFLMUpFqz9HxG65smYJdyKyECOxDiEAe/p1gjF2oonivZAsxVgl2daa4EQWCW6J55qFAFFZiJWYLxNQy2qOSUzGRsyXCUDIeliwAHEO4WSlWQBRFoZakXcKmCXmyXAKs0Ve9vl8q42WoIYpJU4hV3hKcNs8m9gl7p/xQ73eF5kB4j5mNrWmTJRNwAzqiV1CxjVTZCIkEq+Z1bZFZSN2CenmVAFVy4Plz8xKAGWjjAKFk6lCBMDR/MJjLLMSQNm43xAiQKTaA+9/wewhDjL+JVI1kkTSSOTcKbMTwPqESAot6dn6Fr1gHwVJju6IRuyiByPuUUBAg5DGkAgBmxlvdgIEK9gDkohdY/BJo4CAG0R8miRSsGABkgVQs4KXu098IgUXSSRsFAoKZiVAVDY2WUiiPTjYRi41KwGisrGsLtlsth8Fiwnz2fBkQvWfRtlE3iF2yW63/yCacXZ1dW02GwGyTFaRd4idJnCKHRaCxYRHoG5LTKT6SyiToP1fJHbmAYPYRR0UnZQtMnA6s0zg+GZBlt0Gdo7EPHgpE3Q6nZ8YyLhc8Xj8MJh/aKTAY+5FPAKHLE7RdwuYJZmNwzyCMkBCYyKROJBMJl9B/PXXCjjmCmDOVzH3fiPpObEWGqoKe4EBl8v1hlqsdLvd23mkxHM9pc9kMpmno9HoeTii7ewbHEZPPx1ztLS1tV3AnGuMjiNjvbQFuHw6zDo5By7dTPAQNBgMLrRarTkSls1mnwT7uwp9virx9QzbW/HuV/j5d/b+6jniKlllP8lkeONJDk+dq9GsQTnC4fB1heO0K47Hwe7WdDr9nAKgXwOBwHI+C45Htj1d6sd429TUNEcmUdc+PRaLHcvn87dXW4ugzdsaGxufL94NFv9zi1J7GVbhlvb2dnaJ3SVrxfc+n2+NTsZ7/H7/Mr3g5XdSIHyJSH1PZ+7fToyl2+ErqilgZ4NaLYB9goVGaHjR93Hv1ZrU4XDsFT20kH3PObzbWk0CgG1jacVIUnAQb9F+VexyLMzkpcLv0IJV7AHQIOCAUYHx7v5qgScmYHtTqSAyZLEJTK22Bie4iq3xsqpm4SAf9Hq9a2DnJ4uLK3SEULcdRvp3i3zHySqpficxEdsQc1NrlYXXvR+O7qASSezXB+h1SuUomgg9LL8BUoV4749EIolKh+EiqWmqVEZlDgHks2pxHw7xTqUQw9J5NcAXOK10AGIoZ6Zli6JY6Z1Q461KoZ4NiKLHarW+KDsxlDUPHZ5zPQZqUVDPJsTqb5n9malbpAh8C2XXDLl62+WZIDFRUlNVOiwencnNU3aQEkL+cDMSoLvZo2fQB7AJssNAuFuvorlDVVkkg2I87+jo2K2QAVphDrfyViK5VqtO34OkaxXCp+7drdDBCAdubm6eidX+2WwqT5komwh4YQLk+H4aE93h8Xg2gvHekQZOGSgLZTLyDTLJ4Lx9/KZWKBSainT4Iy3FqQBfnUZR42PKQFksBr9QKVXCPusD3OiA/RkQ5kP8qV/Jl1WywAp/6+dcmPM2zL1UrUahe4JqfnWWKXIul3uUbfP8njAFLW1OFr3gdFtZ72cNH+PtQT7/brW+NXqJAHh0y9V8/U/A1U7AfwIMAD7mS3pCbuWJAAAAAElFTkSuQmCC"> | ||||||
|  |       </a> | ||||||
|  |     </div> | ||||||
|  |   </body> | ||||||
|  | </html> | ||||||
		Reference in New Issue
	
	Block a user