├── .editorconfig ├── .eslintrc.json ├── .github └── ISSUE_TEMPLATE │ ├── bug_report.md │ ├── config.yml │ ├── feature_request.md │ └── question.md ├── .gitignore ├── LICENSE ├── README.md ├── declarations.d.ts ├── demo ├── app.js ├── electron-tabs.html ├── mark-github.svg ├── package.json └── page.html ├── dist ├── electron-tabs.d.ts ├── electron-tabs.d.ts.map ├── electron-tabs.js └── electron-tabs.js.map ├── image.jpg ├── package-lock.json ├── package.json ├── src ├── index.ts └── style.css └── tsconfig.json /.editorconfig: -------------------------------------------------------------------------------- 1 | # https://editorconfig.org 2 | 3 | root = true 4 | 5 | [*] 6 | charset = utf-8 7 | indent_style = space 8 | indent_size = 2 9 | end_of_line = lf 10 | insert_final_newline = true 11 | trim_trailing_whitespace = true -------------------------------------------------------------------------------- /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "root": true, 3 | "env": { 4 | "browser": true, 5 | "es2021": true 6 | }, 7 | "extends": "eslint:recommended", 8 | "parserOptions": { 9 | "sourceType": "module", 10 | "ecmaVersion": 12 11 | }, 12 | "rules": { 13 | "indent": [ 14 | "error", 15 | 2 16 | ], 17 | "no-unused-vars": [ 18 | "error", 19 | { 20 | "args": "none" 21 | } 22 | ], 23 | "quotes": [ 24 | "error", 25 | "double" 26 | ], 27 | "semi": [ 28 | "error", 29 | "always" 30 | ] 31 | } 32 | } 33 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/bug_report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Bug report 3 | about: Create a report to help us improve 4 | title: '' 5 | labels: bug 6 | assignees: '' 7 | 8 | --- 9 | 10 | 11 | 12 | **Describe the bug** 13 | A clear and concise description of what the bug is. 14 | 15 | **To Reproduce** 16 | Steps to reproduce the behavior. 17 | 18 | **Expected behavior** 19 | A clear and concise description of what you expected to happen. 20 | 21 | **Please complete the following information** 22 | - OS: [e.g. iOS] 23 | - electron-tabs version [e.g. 0.12.0] 24 | - Electron version [e.g. 8.0.0] 25 | 26 | **Code** 27 | If your issue involves code, please consider creating a sample in a dedicated repository. 28 | 29 | **Screenshots** 30 | If applicable, add screenshots to help explain your problem. 31 | 32 | **Additional information** 33 | Add any other context about the problem here. 34 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/config.yml: -------------------------------------------------------------------------------- 1 | blank_issues_enabled: false 2 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/feature_request.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Feature request 3 | about: Suggest an idea for this project 4 | title: '' 5 | labels: enhancement 6 | assignees: '' 7 | 8 | --- 9 | 10 | **Is your feature request related to a problem? Please describe.** 11 | A clear and concise description of what the problem is. 12 | 13 | **Describe the solution you'd like** 14 | A clear and concise description of what you want to happen. 15 | 16 | **Additional context** 17 | Add any other context or screenshots about the feature request here. 18 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/question.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Question 3 | about: Ask a specific question about electron-tabs 4 | title: '' 5 | labels: question 6 | assignees: '' 7 | 8 | --- 9 | 10 | 17 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | 6 | # Runtime data 7 | pids 8 | *.pid 9 | *.seed 10 | 11 | # Directory for instrumented libs generated by jscoverage/JSCover 12 | lib-cov 13 | 14 | # Coverage directory used by tools like istanbul 15 | coverage 16 | 17 | # nyc test coverage 18 | .nyc_output 19 | 20 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 21 | .grunt 22 | 23 | # node-waf configuration 24 | .lock-wscript 25 | 26 | # Compiled binary addons (http://nodejs.org/api/addons.html) 27 | build/Release 28 | 29 | # Dependency directories 30 | node_modules 31 | jspm_packages 32 | 33 | # Optional npm cache directory 34 | .npm 35 | 36 | # Optional REPL history 37 | .node_repl_history 38 | 39 | # Parcel 40 | .parcel-cache 41 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2016 Thomas Brouard 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # electron-tabs is discontinued 2 | 3 | Thank you for your support and contributions all these years. 4 | 5 | --- 6 | 7 | # electron-tabs 8 | 9 | > Simple tabs for Electron applications 10 | 11 | ![Electron Tab Demo](image.jpg) 12 | 13 | ## Features 14 | 15 | * :electron: Compatible with Electron ≥ 17. 16 | * :lock: Compliant with [Electron security recommendations](https://www.electronjs.org/docs/latest/tutorial/security) (works without `nodeIntegration: true`). 17 | * :toolbox: Written with TypeScript and Web Components. 18 | * :hand: Supports drag and drop out of the box. 19 | * :art: Easily customizable. 20 | 21 | ## Installation 22 | 23 | ```bash 24 | npm install --save electron-tabs 25 | ``` 26 | 27 | ## Getting started 28 | 29 | Define the following `webPreferences` options in the main process: 30 | 31 | ```js 32 | const mainWindow = new electron.BrowserWindow({ 33 | webPreferences: { 34 | webviewTag: true 35 | } 36 | }); 37 | ``` 38 | 39 | Then add the following markup where you want the tabs to display: 40 | 41 | ```html 42 | 43 | 44 | 45 | ``` 46 | 47 | ## Options 48 | 49 | You can add options by setting `` element attributes: 50 | 51 | ```html 52 | 53 | ``` 54 | 55 | The following attributes are supported: 56 | 57 | * `close-button-text` (string): text of the tabs "Close" button. 58 | * `new-tab-button` (boolean): set it to true to display the "New Tab" button. 59 | * `new-tab-button-text` (string): text of the "New Tab" button. 60 | * `sortable` (boolean): set it to true to make the tabs sortable by drag and drop. 61 | * `visibility-threshold` (number): the minimum number of tabs necessary for the tab bar to be displayed. 0 (default) means that it will always remain visible. 62 | 63 | ## Methods 64 | 65 | Use `TabGroup` methods and manipulate tabs in a script after calling `electron-tabs.js`. 66 | 67 | ```html 68 | 69 | 70 | 71 | 72 | 91 | ``` 92 | 93 | ### TabGroup 94 | 95 | #### `tabGroup.addTab(options)` 96 | 97 | Add a new tab and returns the related `Tab` instance. 98 | 99 | * `title`: tab title. 100 | * `src`: URL to the page which will be loaded into the view. This is actually the same than `options.webview.src`. 101 | * `badge`: optional text to put into a badge, badge will be hidden if false. 102 | * `iconURL`: optional URL to the tab icon. 103 | * `icon`: optional code for a tab icon. Can be used with symbol libraries (example with Font Awesome: `icon: 'fa fa-icon-name'`). This attribute is ignored if an `iconURL` was given. 104 | * `closable` (default: `true`): if set to `true` the close button won't be displayed and the user won't be able to close the tab. See also `tab.close()`. 105 | * `visible` (default: `true`): set this to `false` if you don't want to display the tab once it is loaded. If set to `false` then you will need to call `tab.show()` to display the tab. 106 | * `active` (default: `false`): set this to `true` if you want to activate the tab once it is loaded. Otherwise you will need to call `tab.activate()`. 107 | * `ready`: a callback function to call once the tab is ready. The `Tab` instance is passed as the only parameter. 108 | * `webviewAttributes`: attributes to add to the webview tag. See [webview documentation](http://electron.atom.io/docs/api/web-view-tag/#tag-attributes). 109 | 110 | #### `tabGroup.setDefaultTab(options)` 111 | 112 | Define default options to use for creating the tab when the "New Tab" button is clicked or when calling `tabGroup.addTab()` with no parameter. 113 | 114 | ```javascript 115 | tabGroup.setDefaultTab({ 116 | title: "New Page", 117 | src: "path/to/new-page.html", 118 | active: true 119 | }); 120 | ``` 121 | 122 | #### `tabGroup.getTab(id)` 123 | 124 | Retrieve an instance of `Tab` from this `id` (return `null` if not found). 125 | 126 | #### `tabGroup.getTabByPosition(position)` 127 | 128 | Retrieve an instance of `Tab` from this `position` (return `null` if not found). A negative value is an offset from the right. 129 | 130 | To get the tab in the leftmost position: 131 | 132 | ```javascript 133 | tabGroup.getTabByPosition(1); 134 | ``` 135 | 136 | To get the tab in the rightmost position: 137 | 138 | ```javascript 139 | tabGroup.getTabByPosition(-1); 140 | ``` 141 | 142 | #### `tabGroup.getTabByRelPosition(position)` 143 | 144 | Retrieve an instance of `Tab` from this `position` relative to the active tab (return `null` if not found). 145 | 146 | `tabGroup.getNextTab()` is an alias to `tabGroup.getTabByRelPosition(1)`. 147 | 148 | `tabGroup.getPreviousTab()` is an alias to `tabGroup.getTabByRelPosition(-1)`. 149 | 150 | #### `tabGroup.getActiveTab()` 151 | 152 | Return the active tab (return `null` if none). 153 | 154 | #### `tabGroup.getTabs()` 155 | 156 | Return all registered tabs. 157 | 158 | #### `tabGroup.eachTab(fn, thisArg)` 159 | 160 | Loop through the list of tabs in `tabGroup` and execute the `fn` function for each tab. `fn` is called with the following parameters: 161 | 162 | * `currentTab`: the current tab object. 163 | * `index`: the index of the current tab being processed. 164 | * `tabs`: the full array of tabs (similar to `tabGroup.getTabs()`). 165 | 166 | `thisArg` (optional) is the value to use as `this` when executing `fn`. 167 | 168 | ### Tab 169 | 170 | Instances of `Tab` are returned by the `tabGroup.addTab()` method. 171 | 172 | #### `tab.setTitle(title)` 173 | 174 | Set tab title. 175 | 176 | #### `tab.getTitle()` 177 | 178 | Get current tab title. 179 | 180 | #### `tab.setBadge(badge)` 181 | 182 | Set tab badge. 183 | 184 | #### `tab.getBadge()` 185 | 186 | Get current tab badge. 187 | 188 | #### `tab.setIcon (iconURL, icon)` 189 | 190 | Set tab icon (a iconURL or an icon must be given). 191 | 192 | #### `tab.getIcon()` 193 | 194 | Get current tab icon URL / icon. 195 | 196 | #### `tab.setPosition(newPosition)` 197 | 198 | Move tab to the specified position. See [`tabGroup.getTabByPosition`](#tabgroupgettabbypositionposition) for information about positions. 199 | 200 | #### `tab.getPosition(fromRight)` 201 | 202 | Get the tab position. If `fromRight` is true the index returned is negative and is the offset from the right. 203 | 204 | #### `tab.activate()` 205 | 206 | Activate this tab. The class "active" is added to the active tab. 207 | 208 | #### `tab.show(flag)` 209 | 210 | Toggle the "visible" class on the tab. `tab.hide()` is an alias to `tab.show(false)`. 211 | 212 | #### `tab.hasClass(classname)` 213 | 214 | Return `true` if the tab element has the specified classname. Useful for checking if a tab is "active" or "visible". 215 | 216 | #### `tab.close(force)` 217 | 218 | Close the tab (and activate another tab if relevant). When `force` is set to `true` the tab will be closed even if it is not `closable`. 219 | 220 | ## Events 221 | 222 | The following events are emitted: 223 | 224 | * `tabGroup.on("tab-added", (tab, tabGroup) => { ... });` 225 | * `tabGroup.on("tab-removed", (tab, tabGroup) => { ... });` 226 | * `tabGroup.on("tab-active", (tab, tabGroup) => { ... });` 227 | * `tab.on("webview-ready", (tab) => { ... });` 228 | * `tab.on("webview-dom-ready", (tab) => { ... });` 229 | * `tab.on("title-changed", (title, tab) => { ... });` 230 | * `tab.on("badge-changed", (badge, tab) => { ... });` 231 | * `tab.on("icon-changed", (icon, tab) => { ... });` 232 | * `tab.on("active", (tab) => { ... });` 233 | * `tab.on("inactive", (tab) => { ... });` 234 | * `tab.on("visible", (tab) => { ... });` 235 | * `tab.on("hidden", (tab) => { ... });` 236 | * `tab.on("close", (tab) => { ... });` 237 | * `tab.on("closing", (tab, abort) => { ... });` (Use `abort()` function to cancel closing) 238 | 239 | You can also use `tab.once` to automatically remove the listener when invoked: 240 | 241 | * `tab.once("webview-ready", (tab) => { ... });` 242 | * `tab.once("webview-dom-ready", (tab) => { ... });` 243 | 244 | ## Access Electron webview element 245 | 246 | You can access the webview element and use its methods with through the `Tab.webview` attribute. See [webview documentation](https://electronjs.org/docs/api/webview-tag#methods). 247 | 248 | ```javascript 249 | let webview = tab.webview; 250 | webview.loadURL("file://path/to/new/page.html"); 251 | ``` 252 | 253 | ## Custom styles 254 | 255 | To customize tab-group styles, set new values to [electron-tabs CSS variables](https://github.com/brrd/electron-tabs/blob/master/src/style.css) in your application stylesheet. 256 | 257 | Since `TabGroup` is a Web Component you won't be able to change its styles directly from your app stylesheet. If you need more control over it then you can add a ` 264 | 265 | ``` 266 | 267 | This method is particularly useful when you need to define custom badges or tab styles: 268 | 269 | ```html 270 | 271 | 281 | 282 | 283 | 284 | 285 | 305 | ``` 306 | 307 | ## Development 308 | 309 | `electron-tabs` uses TypeScript and Parcel under the hood. 310 | 311 | ### Requirements 312 | 313 | Git and Node 12+. 314 | 315 | ### Build 316 | 317 | ```bash 318 | # Clone this repo 319 | git clone git@github.com:brrd/electron-tabs.git 320 | cd electron-tabs 321 | 322 | # Install dependencies 323 | npm install 324 | 325 | # Build 326 | npm run build 327 | 328 | # ...or watch 329 | npm run watch 330 | ``` 331 | 332 | ### Demo 333 | 334 | ```bash 335 | npm run demo 336 | ``` 337 | 338 | ## License 339 | 340 | The MIT License (MIT) - Copyright (c) 2022 Thomas Brouard 341 | -------------------------------------------------------------------------------- /declarations.d.ts: -------------------------------------------------------------------------------- 1 | declare module "*.css"; 2 | -------------------------------------------------------------------------------- /demo/app.js: -------------------------------------------------------------------------------- 1 | const electron = require("electron"); 2 | const app = electron.app; 3 | 4 | app.on("ready", function () { 5 | const mainWindow = new electron.BrowserWindow({ 6 | webPreferences: { 7 | webviewTag: true 8 | } 9 | }); 10 | mainWindow.loadURL("file://" + __dirname + "/electron-tabs.html"); 11 | mainWindow.on("ready-to-show", function () { 12 | mainWindow.show(); 13 | mainWindow.focus(); 14 | }); 15 | }); 16 | -------------------------------------------------------------------------------- /demo/electron-tabs.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | electron-tabs-demo 6 | 7 | 8 | 9 | 10 | 21 | 22 | 23 | 24 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /demo/mark-github.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /demo/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "electron-tabs-demo", 3 | "main": "app.js" 4 | } 5 | -------------------------------------------------------------------------------- /demo/page.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | Local Page 8 | 9 | 10 |

Hello World!

11 |

This is a local page.

12 | 13 | 14 | -------------------------------------------------------------------------------- /dist/electron-tabs.d.ts: -------------------------------------------------------------------------------- 1 | import Sortable from "sortablejs"; 2 | interface TabGroupOptions { 3 | closeButtonText: string; 4 | defaultTab: TabOptions | ((tabGroup: TabGroup) => TabOptions); 5 | newTabButton: boolean; 6 | newTabButtonText: string; 7 | sortable: boolean; 8 | sortableOptions?: Sortable.Options; 9 | visibilityThreshold: number; 10 | } 11 | interface TabOptions { 12 | active?: boolean; 13 | badge?: Badge; 14 | closable?: boolean; 15 | icon?: string; 16 | iconURL?: string; 17 | ready?: ((tab: Tab) => void); 18 | src?: string; 19 | title?: string; 20 | visible?: boolean; 21 | webviewAttributes?: { 22 | [key: string]: any; 23 | }; 24 | } 25 | interface Badge { 26 | text: string; 27 | classname: string; 28 | } 29 | export class TabGroup extends HTMLElement { 30 | buttonContainer: HTMLDivElement; 31 | isReady: boolean; 32 | newTabId: number; 33 | options: TabGroupOptions; 34 | shadow: ShadowRoot; 35 | tabContainer: HTMLDivElement; 36 | tabs: Array; 37 | viewContainer: HTMLDivElement; 38 | constructor(); 39 | emit(type: string, ...args: any[]): void; 40 | on(type: string, fn: (...detail: any[]) => void): void; 41 | once(type: string, fn: (detail: string) => void): void; 42 | connectedCallback(): void; 43 | initSortable(): void; 44 | setDefaultTab(tab: TabOptions): void; 45 | addTab(args?: TabOptions | ((tabGroup: TabGroup) => TabOptions)): Tab; 46 | getTab(id: number): Tab; 47 | getTabByPosition(position: number): Tab; 48 | getTabByRelPosition(position: number): Tab; 49 | getNextTab(): Tab; 50 | getPreviousTab(): Tab; 51 | getTabs(): Tab[]; 52 | eachTab(fn: (tab: Tab) => void): void; 53 | getActiveTab(): Tab; 54 | setActiveTab(tab: Tab): void; 55 | removeTab(tab: Tab, triggerEvent?: boolean): void; 56 | activateRecentTab(): void; 57 | } 58 | export class Tab extends EventTarget { 59 | badge: Badge; 60 | closable: boolean; 61 | element: HTMLDivElement; 62 | icon: string; 63 | iconURL: string; 64 | id: number; 65 | isClosed: boolean; 66 | isReady: boolean; 67 | spans: { 68 | [key: string]: HTMLSpanElement; 69 | }; 70 | tabGroup: TabGroup; 71 | title: string; 72 | webview: HTMLElement; 73 | webviewAttributes: { 74 | [key: string]: any; 75 | }; 76 | constructor(tabGroup: TabGroup, id: number, args: TabOptions); 77 | emit(type: string, ...args: any[]): void; 78 | on(type: string, fn: (...detail: any[]) => void): void; 79 | once(type: string, fn: (detail: string) => void): void; 80 | initWebview(): void; 81 | setTitle(title: string): this; 82 | getTitle(): string; 83 | setBadge(badge?: Badge): void; 84 | getBadge(): Badge; 85 | setIcon(iconURL: string, icon: string): this; 86 | getIcon(): string; 87 | setPosition(newPosition: number): this; 88 | getPosition(fromRight?: boolean): number; 89 | activate(): this; 90 | show(flag?: boolean): this; 91 | hide(): this; 92 | hasClass(classname: string): boolean; 93 | close(force: boolean): void; 94 | } 95 | 96 | //# sourceMappingURL=electron-tabs.d.ts.map 97 | -------------------------------------------------------------------------------- /dist/electron-tabs.d.ts.map: -------------------------------------------------------------------------------- 1 | {"mappings":";AAQA;IACE,eAAe,EAAE,MAAM,CAAC;IACxB,UAAU,EAAE,UAAU,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,KAAK,UAAU,CAAC,CAAC;IAC9D,YAAY,EAAE,OAAO,CAAC;IACtB,gBAAgB,EAAE,MAAM,CAAC;IACzB,QAAQ,EAAE,OAAO,CAAC;IAClB,eAAe,CAAC,EAAE,SAAS,OAAO,CAAA;IAClC,mBAAmB,EAAE,MAAM,CAAC;CAC7B;AAED;IACE,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,KAAK,CAAC,EAAE,KAAK,CAAC;IACd,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,CAAC,CAAC,GAAG,EAAE,GAAG,KAAK,IAAI,CAAC,CAAC;IAC7B,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,iBAAiB,CAAC,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;KAAE,CAAC;CAC5C;AAED;IACE,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAA;CAClB;AA0BD,qBAAe,SAAQ,WAAW;IAChC,eAAe,EAAE,cAAc,CAAC;IAChC,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,eAAe,CAAC;IACzB,MAAM,EAAE,UAAU,CAAC;IACnB,YAAY,EAAE,cAAc,CAAC;IAC7B,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC;IACjB,aAAa,EAAE,cAAc,CAAC;;IA6B9B,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIjC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,GAAG,MAAM,EAAE,GAAG,EAAE,KAAK,IAAI;IAI/C,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI;IAI/C,iBAAiB;IA+DjB,YAAY;IAiBZ,aAAa,CAAC,GAAG,EAAE,UAAU;IAI7B,MAAM,CAAC,IAAI,oDAA0B;IAgBrC,MAAM,CAAC,EAAE,EAAE,MAAM;IASjB,gBAAgB,CAAC,QAAQ,EAAE,MAAM;IAUjC,mBAAmB,CAAC,QAAQ,EAAE,MAAM;IAQpC,UAAU;IAIV,cAAc;IAId,OAAO;IAIP,OAAO,CAAC,EAAE,EAAE,CAAC,GAAG,EAAE,GAAG,KAAK,IAAI;IAI9B,YAAY;IAKZ,YAAY,CAAC,GAAG,EAAE,GAAG;IAMrB,SAAS,CAAC,GAAG,EAAE,GAAG,EAAE,YAAY,UAAQ;IASxC,iBAAiB;CAKlB;AAED,gBAAU,SAAQ,WAAW;IAC3B,KAAK,EAAE,KAAK,CAAC;IACb,QAAQ,EAAE,OAAO,CAAC;IAClB,OAAO,EAAE,cAAc,CAAC;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,EAAE,EAAE,MAAM,CAAC;IACX,QAAQ,EAAE,OAAO,CAAC;IAClB,OAAO,EAAE,OAAO,CAAC;IACjB,KAAK,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,eAAe,CAAA;KAAE,CAAC;IAC1C,QAAQ,EAAE,QAAQ,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,WAAW,CAAC;IACrB,iBAAiB,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;KAAE,CAAC;gBAE9B,QAAQ,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,UAAU;IA4B5D,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,IAAI,EAAE,GAAG,EAAE;IAIjC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,GAAG,MAAM,EAAE,GAAG,EAAE,KAAK,IAAI;IAI/C,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI;IAmD/C,WAAW;IA+BX,QAAQ,CAAC,KAAK,EAAE,MAAM;IAUtB,QAAQ;IAKR,QAAQ,CAAC,KAAK,CAAC,EAAE,KAAK;IAgBtB,QAAQ;IAKR,OAAO,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;IAgBrC,OAAO;IAMP,WAAW,CAAC,WAAW,EAAE,MAAM;IAuB/B,WAAW,CAAC,SAAS,UAAQ;IAY7B,QAAQ;IAgBR,IAAI,CAAC,IAAI,UAAO;IAYhB,IAAI;IAIJ,QAAQ,CAAC,SAAS,EAAE,MAAM;IAI1B,KAAK,CAAC,KAAK,EAAE,OAAO;CAqBrB","sources":["src/src/index.ts","src/index.ts"],"sourcesContent":[null,"import Sortable from \"sortablejs\";\n// @ts-ignore\nimport styles from \"bundle-text:./style.css\";\n\nif (!document) {\n throw Error(\"electron-tabs module must be called in renderer process\");\n}\n\ninterface TabGroupOptions {\n closeButtonText: string,\n defaultTab: TabOptions | ((tabGroup: TabGroup) => TabOptions),\n newTabButton: boolean,\n newTabButtonText: string,\n sortable: boolean,\n sortableOptions?: Sortable.Options\n visibilityThreshold: number,\n}\n\ninterface TabOptions {\n active?: boolean;\n badge?: Badge;\n closable?: boolean;\n icon?: string;\n iconURL?: string;\n ready?: ((tab: Tab) => void);\n src?: string;\n title?: string;\n visible?: boolean;\n webviewAttributes?: { [key: string]: any };\n}\n\ninterface Badge {\n text: string,\n classname: string\n}\n\nconst CLASSNAMES = {\n ROOT: \"etabs\",\n NAV: \"nav\",\n TABS: \"tabs\",\n TAB: \"tab\",\n BUTTONS: \"buttons\",\n VIEWS: \"views\",\n VIEW: \"view\"\n}\n\nfunction emit(emitter: TabGroup | Tab, type: string, args: any[]) {\n if (type === \"ready\") {\n emitter.isReady = true;\n }\n emitter.dispatchEvent(new CustomEvent(type, { detail: args }));\n}\n\nfunction on(emitter: TabGroup | Tab, type: string, fn: (detail: string) => void, options?: { [key: string]: any }) {\n if (type === \"ready\" && emitter.isReady === true) {\n fn.apply(emitter, [emitter]);\n }\n emitter.addEventListener(type, ((e: CustomEvent) => fn.apply(emitter, e.detail)) as EventListener, options);\n}\n\nclass TabGroup extends HTMLElement {\n buttonContainer: HTMLDivElement;\n isReady: boolean;\n newTabId: number;\n options: TabGroupOptions;\n shadow: ShadowRoot;\n tabContainer: HTMLDivElement;\n tabs: Array;\n viewContainer: HTMLDivElement;\n\n constructor() {\n super();\n\n this.isReady = false;\n\n // Options\n this.options = {\n closeButtonText: this.getAttribute(\"close-button-text\") || \"×\",\n defaultTab: { title: \"New Tab\", active: true },\n newTabButton: !!this.getAttribute(\"new-tab-button\") === true || false,\n newTabButtonText: this.getAttribute(\"new-tab-button-text\") || \"+\",\n sortable: !!this.getAttribute(\"sortable\") === true || false,\n visibilityThreshold: Number(this.getAttribute(\"visibility-threshold\")) || 0\n };\n\n this.tabs = [];\n this.newTabId = 0;\n\n this.createComponent();\n this.initVisibility();\n if (this.options.sortable) {\n this.initSortable();\n }\n\n this.emit(\"ready\", this);\n }\n\n emit(type: string, ...args: any[]) {\n return emit(this, type, args);\n }\n\n on(type: string, fn: (...detail: any[]) => void) {\n return on(this, type, fn);\n }\n\n once(type: string, fn: (detail: string) => void) {\n return on(this, type, fn, { once: true });\n }\n\n connectedCallback() {\n // Support custom styles\n const style = this.querySelector(\"style\");\n if (style) {\n this.shadow.appendChild(style);\n }\n }\n\n private createComponent() {\n const shadow = this.attachShadow({mode: \"open\"});\n this.shadow = shadow;\n\n const wrapper = document.createElement(\"div\");\n wrapper.setAttribute(\"class\", CLASSNAMES.ROOT);\n\n const tabgroup = document.createElement(\"nav\");\n tabgroup.setAttribute(\"class\", CLASSNAMES.NAV);\n wrapper.appendChild(tabgroup);\n\n const tabContainer = document.createElement(\"div\");\n tabContainer.setAttribute(\"class\", CLASSNAMES.TABS);\n tabgroup.appendChild(tabContainer);\n this.tabContainer = tabContainer;\n\n const buttonContainer = document.createElement(\"div\");\n buttonContainer.setAttribute(\"class\", CLASSNAMES.BUTTONS);\n tabgroup.appendChild(buttonContainer);\n this.buttonContainer = buttonContainer;\n\n if (this.options.newTabButton) {\n const button = this.buttonContainer.appendChild(document.createElement(\"button\"));\n button.innerHTML = this.options.newTabButtonText;\n button.addEventListener(\"click\", this.addTab.bind(this, undefined), false);\n }\n\n const viewContainer = document.createElement(\"div\");\n viewContainer.setAttribute(\"class\", CLASSNAMES.VIEWS);\n wrapper.appendChild(viewContainer);\n this.viewContainer = viewContainer;\n\n const style = document.createElement(\"style\");\n style.textContent = styles;\n\n shadow.appendChild(style);\n shadow.appendChild(wrapper);\n }\n\n private initVisibility() {\n function toggleTabsVisibility(tab: Tab, tabGroup: TabGroup) {\n const visibilityThreshold = this.options.visibilityThreshold;\n const el = tabGroup.tabContainer.parentElement;\n if (this.tabs.length >= visibilityThreshold) {\n el.classList.add(\"visible\");\n } else {\n el.classList.remove(\"visible\");\n }\n }\n\n this.on(\"tab-added\", toggleTabsVisibility);\n this.on(\"tab-removed\", toggleTabsVisibility);\n toggleTabsVisibility(null, this);\n }\n\n initSortable() {\n const createNewSortable = () => {\n const options = Object.assign({\n direction: \"horizontal\",\n animation: 150,\n swapThreshold: 0.20\n }, this.options.sortableOptions);\n new Sortable(this.tabContainer, options);\n };\n\n if (Sortable) {\n createNewSortable();\n } else {\n document.addEventListener(\"DOMContentLoaded\", createNewSortable);\n }\n }\n\n setDefaultTab(tab: TabOptions) {\n this.options.defaultTab = tab;\n }\n\n addTab(args = this.options.defaultTab) {\n if (typeof args === \"function\") {\n args = args(this);\n }\n const id = this.newTabId;\n this.newTabId++;\n const tab = new Tab(this, id, args);\n this.tabs.push(tab);\n // Don't call tab.activate() before a tab is referenced in this.tabs\n if (args.active === true) {\n tab.activate();\n }\n this.emit(\"tab-added\", tab, this);\n return tab;\n }\n\n getTab(id: number) {\n for (let i in this.tabs) {\n if (this.tabs[i].id === id) {\n return this.tabs[i];\n }\n }\n return null;\n }\n\n getTabByPosition(position: number) {\n const fromRight = position < 0;\n for (let i in this.tabs) {\n if (this.tabs[i].getPosition(fromRight) === position) {\n return this.tabs[i];\n }\n }\n return null;\n }\n\n getTabByRelPosition(position: number) {\n position = this.getActiveTab().getPosition() + position;\n if (position <= 0) {\n return null;\n }\n return this.getTabByPosition(position);\n }\n\n getNextTab() {\n return this.getTabByRelPosition(1);\n }\n\n getPreviousTab() {\n return this.getTabByRelPosition(-1);\n }\n\n getTabs() {\n return this.tabs.slice();\n }\n\n eachTab(fn: (tab: Tab) => void) {\n this.getTabs().forEach(fn);\n }\n\n getActiveTab() {\n if (this.tabs.length === 0) return null;\n return this.tabs[0];\n }\n\n setActiveTab(tab: Tab) {\n this.removeTab(tab);\n this.tabs.unshift(tab);\n this.emit(\"tab-active\", tab, this);\n }\n\n removeTab(tab: Tab, triggerEvent = false) {\n const id = tab.id;\n const index = this.tabs.findIndex((t: Tab) => t.id === id);\n this.tabs.splice(index, 1);\n if (triggerEvent) {\n this.emit(\"tab-removed\", tab, this);\n }\n }\n\n activateRecentTab() {\n if (this.tabs.length > 0) {\n this.tabs[0].activate();\n }\n }\n}\n\nclass Tab extends EventTarget {\n badge: Badge;\n closable: boolean;\n element: HTMLDivElement;\n icon: string;\n iconURL: string;\n id: number;\n isClosed: boolean;\n isReady: boolean;\n spans: { [key: string]: HTMLSpanElement };\n tabGroup: TabGroup;\n title: string;\n webview: HTMLElement;\n webviewAttributes: { [key: string]: any };\n\n constructor(tabGroup: TabGroup, id: number, args: TabOptions) {\n super();\n this.badge = args.badge;\n this.closable = args.closable === false ? false : true;\n this.icon = args.icon;\n this.iconURL = args.iconURL;\n this.id = id;\n this.isClosed = false;\n this.isReady = false;\n this.spans = {};\n this.tabGroup = tabGroup;\n this.title = args.title;\n this.webviewAttributes = args.webviewAttributes || {};\n this.webviewAttributes.src = args.src;\n\n this.initTab();\n this.initWebview();\n\n if (args.visible !== false) {\n this.show();\n }\n if (typeof args.ready === \"function\") {\n args.ready(this);\n } else {\n this.emit(\"ready\", this);\n }\n }\n\n emit(type: string, ...args: any[]) {\n return emit(this, type, args);\n }\n\n on(type: string, fn: (...detail: any[]) => void) {\n return on(this, type, fn);\n }\n\n once(type: string, fn: (detail: string) => void) {\n return on(this, type, fn, { once: true });\n }\n\n private initTab() {\n const tab = this.element = document.createElement(\"div\");\n tab.classList.add(CLASSNAMES.TAB);\n for (let el of [\"icon\", \"title\", \"badge\", \"close\"]) {\n const span = tab.appendChild(document.createElement(\"span\"));\n span.classList.add(`${CLASSNAMES.TAB}-${el}`);\n this.spans[el] = span;\n }\n\n this.setTitle(this.title);\n this.setBadge(this.badge);\n this.setIcon(this.iconURL, this.icon);\n this.initTabCloseButton();\n this.initTabClickHandler();\n\n this.tabGroup.tabContainer.appendChild(this.element);\n }\n\n private initTabCloseButton() {\n const container = this.spans.close;\n if (this.closable) {\n const button = container.appendChild(document.createElement(\"button\"));\n button.innerHTML = this.tabGroup.options.closeButtonText;\n button.addEventListener(\"click\", this.close.bind(this, false), false);\n }\n }\n\n private initTabClickHandler() {\n // Mouse up\n const tabClickHandler = function(e: KeyboardEvent) {\n if (this.isClosed) return;\n if (e.which === 2) {\n this.close();\n }\n };\n this.element.addEventListener(\"mouseup\", tabClickHandler.bind(this), false);\n // Mouse down\n const tabMouseDownHandler = function(e: KeyboardEvent) {\n if (this.isClosed) return;\n if (e.which === 1) {\n if ((e.target as HTMLElement).matches(\"button\")) return;\n this.activate();\n }\n };\n this.element.addEventListener(\"mousedown\", tabMouseDownHandler.bind(this), false);\n }\n\n initWebview() {\n const webview = this.webview = document.createElement(\"webview\");\n\n const tabWebviewDidFinishLoadHandler = function(e: Event) {\n this.emit(\"webview-ready\", this);\n };\n\n this.webview.addEventListener(\"did-finish-load\", tabWebviewDidFinishLoadHandler.bind(this), false);\n\n const tabWebviewDomReadyHandler = function(e: Event) {\n // Remove this once https://github.com/electron/electron/issues/14474 is fixed\n webview.blur();\n webview.focus();\n this.emit(\"webview-dom-ready\", this);\n };\n\n this.webview.addEventListener(\"dom-ready\", tabWebviewDomReadyHandler.bind(this), false);\n\n this.webview.classList.add(CLASSNAMES.VIEW);\n if (this.webviewAttributes) {\n const attrs = this.webviewAttributes;\n for (let key in attrs) {\n const attr = attrs[key];\n if (attr === false) continue;\n this.webview.setAttribute(key, attr);\n }\n }\n\n this.tabGroup.viewContainer.appendChild(this.webview);\n }\n\n setTitle(title: string) {\n if (this.isClosed) return;\n const span = this.spans.title;\n span.innerHTML = title;\n span.title = title;\n this.title = title;\n this.emit(\"title-changed\", title, this);\n return this;\n }\n\n getTitle() {\n if (this.isClosed) return;\n return this.title;\n }\n\n setBadge(badge?: Badge) {\n if (this.isClosed) return;\n const span = this.spans.badge;\n this.badge = badge;\n\n if (badge) {\n span.innerHTML = badge.text;\n span.classList.add(badge.classname);\n span.classList.remove(\"hidden\");\n } else {\n span.classList.add(\"hidden\");\n }\n\n this.emit(\"badge-changed\", badge, this);\n }\n\n getBadge() {\n if (this.isClosed) return;\n return this.badge;\n }\n\n setIcon(iconURL: string, icon: string) {\n if (this.isClosed) return;\n this.iconURL = iconURL;\n this.icon = icon;\n const span = this.spans.icon;\n if (iconURL) {\n span.innerHTML = ``;\n this.emit(\"icon-changed\", iconURL, this);\n } else if (icon) {\n span.innerHTML = ``;\n this.emit(\"icon-changed\", icon, this);\n }\n\n return this;\n }\n\n getIcon() {\n if (this.isClosed) return;\n if (this.iconURL) return this.iconURL;\n return this.icon;\n }\n\n setPosition(newPosition: number) {\n const tabContainer = this.tabGroup.tabContainer;\n const length = tabContainer.childElementCount;\n const thisPosition = this.getPosition();\n const tabs = Array.from(tabContainer.children)\n tabs.splice(thisPosition, 1);\n\n if (newPosition < 0) {\n newPosition += length;\n if (newPosition < 0) {\n newPosition = 0;\n }\n }\n\n if (newPosition < length) {\n tabContainer.insertBefore(this.element, tabs[newPosition]);\n } else {\n tabContainer.appendChild(this.element);\n }\n\n return this;\n }\n\n getPosition(fromRight = false) {\n let position = 0;\n let tab = this.element;\n while ((tab = tab.previousSibling as HTMLDivElement) != null) position++;\n\n if (fromRight === true) {\n position -= this.tabGroup.tabContainer.childElementCount;\n }\n\n return position;\n }\n\n activate() {\n if (this.isClosed) return;\n const activeTab = this.tabGroup.getActiveTab();\n if (activeTab) {\n activeTab.element.classList.remove(\"active\");\n activeTab.webview.classList.remove(\"visible\");\n activeTab.emit(\"inactive\", activeTab);\n }\n this.tabGroup.setActiveTab(this);\n this.element.classList.add(\"active\");\n this.webview.classList.add(\"visible\");\n this.webview.focus();\n this.emit(\"active\", this);\n return this;\n }\n\n show(flag = true) {\n if (this.isClosed) return;\n if (flag) {\n this.element.classList.add(\"visible\");\n this.emit(\"visible\", this);\n } else {\n this.element.classList.remove(\"visible\");\n this.emit(\"hidden\", this);\n }\n return this;\n }\n\n hide() {\n return this.show(false);\n }\n\n hasClass(classname: string) {\n return this.element.classList.contains(classname);\n }\n\n close(force: boolean) {\n const abortController = new AbortController();\n const abort = () => abortController.abort();\n this.emit(\"closing\", this, abort);\n\n const abortSignal = abortController.signal;\n if (this.isClosed || (!this.closable && !force) || abortSignal.aborted) return;\n\n this.isClosed = true;\n const tabGroup = this.tabGroup;\n tabGroup.tabContainer.removeChild(this.element);\n tabGroup.viewContainer.removeChild(this.webview);\n const activeTab = this.tabGroup.getActiveTab();\n tabGroup.removeTab(this, true);\n\n this.emit(\"close\", this);\n\n if (activeTab.id === this.id) {\n tabGroup.activateRecentTab();\n }\n }\n}\n\ncustomElements.define(\"tab-group\", TabGroup);\n\nexport type { TabGroup, Tab };\n"],"names":[],"version":3,"file":"electron-tabs.d.ts.map"} -------------------------------------------------------------------------------- /image.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/brrd/electron-tabs/d43a1569e031f38918c21867add2cd0844535408/image.jpg -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "electron-tabs", 3 | "version": "1.0.3", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "electron-tabs", 9 | "version": "1.0.3", 10 | "license": "MIT", 11 | "devDependencies": { 12 | "@parcel/packager-ts": "^2.5.0", 13 | "@parcel/transformer-inline-string": "^2.5.0", 14 | "@parcel/transformer-typescript-types": "^2.5.0", 15 | "@types/sortablejs": "^1.13.0", 16 | "electron": "^17.1.2", 17 | "parcel": "^2.5.0", 18 | "postcss": "^8.4.14", 19 | "sortablejs": "^1.15.0", 20 | "typescript": "^4.7.2" 21 | } 22 | }, 23 | "node_modules/@babel/code-frame": { 24 | "version": "7.16.7", 25 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.16.7.tgz", 26 | "integrity": "sha512-iAXqUn8IIeBTNd72xsFlgaXHkMBMt6y4HJp1tIaK465CWLT/fG1aqB7ykr95gHHmlBdGbFeWWfyB4NJJ0nmeIg==", 27 | "dev": true, 28 | "dependencies": { 29 | "@babel/highlight": "^7.16.7" 30 | }, 31 | "engines": { 32 | "node": ">=6.9.0" 33 | } 34 | }, 35 | "node_modules/@babel/helper-validator-identifier": { 36 | "version": "7.16.7", 37 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.16.7.tgz", 38 | "integrity": "sha512-hsEnFemeiW4D08A5gUAZxLBTXpZ39P+a+DGDsHw1yxqyQ/jzFEnxf5uTEGp+3bzAbNOxU1paTgYS4ECU/IgfDw==", 39 | "dev": true, 40 | "engines": { 41 | "node": ">=6.9.0" 42 | } 43 | }, 44 | "node_modules/@babel/highlight": { 45 | "version": "7.17.12", 46 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.17.12.tgz", 47 | "integrity": "sha512-7yykMVF3hfZY2jsHZEEgLc+3x4o1O+fYyULu11GynEUQNwB6lua+IIQn1FiJxNucd5UlyJryrwsOh8PL9Sn8Qg==", 48 | "dev": true, 49 | "dependencies": { 50 | "@babel/helper-validator-identifier": "^7.16.7", 51 | "chalk": "^2.0.0", 52 | "js-tokens": "^4.0.0" 53 | }, 54 | "engines": { 55 | "node": ">=6.9.0" 56 | } 57 | }, 58 | "node_modules/@babel/highlight/node_modules/ansi-styles": { 59 | "version": "3.2.1", 60 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 61 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 62 | "dev": true, 63 | "dependencies": { 64 | "color-convert": "^1.9.0" 65 | }, 66 | "engines": { 67 | "node": ">=4" 68 | } 69 | }, 70 | "node_modules/@babel/highlight/node_modules/chalk": { 71 | "version": "2.4.2", 72 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 73 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 74 | "dev": true, 75 | "dependencies": { 76 | "ansi-styles": "^3.2.1", 77 | "escape-string-regexp": "^1.0.5", 78 | "supports-color": "^5.3.0" 79 | }, 80 | "engines": { 81 | "node": ">=4" 82 | } 83 | }, 84 | "node_modules/@babel/highlight/node_modules/color-convert": { 85 | "version": "1.9.3", 86 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 87 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 88 | "dev": true, 89 | "dependencies": { 90 | "color-name": "1.1.3" 91 | } 92 | }, 93 | "node_modules/@babel/highlight/node_modules/color-name": { 94 | "version": "1.1.3", 95 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 96 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 97 | "dev": true 98 | }, 99 | "node_modules/@babel/highlight/node_modules/escape-string-regexp": { 100 | "version": "1.0.5", 101 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 102 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 103 | "dev": true, 104 | "engines": { 105 | "node": ">=0.8.0" 106 | } 107 | }, 108 | "node_modules/@babel/highlight/node_modules/has-flag": { 109 | "version": "3.0.0", 110 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 111 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 112 | "dev": true, 113 | "engines": { 114 | "node": ">=4" 115 | } 116 | }, 117 | "node_modules/@babel/highlight/node_modules/supports-color": { 118 | "version": "5.5.0", 119 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 120 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 121 | "dev": true, 122 | "dependencies": { 123 | "has-flag": "^3.0.0" 124 | }, 125 | "engines": { 126 | "node": ">=4" 127 | } 128 | }, 129 | "node_modules/@electron/get": { 130 | "version": "1.14.1", 131 | "resolved": "https://registry.npmjs.org/@electron/get/-/get-1.14.1.tgz", 132 | "integrity": "sha512-BrZYyL/6m0ZXz/lDxy/nlVhQz+WF+iPS6qXolEU8atw7h6v1aYkjwJZ63m+bJMBTxDE66X+r2tPS4a/8C82sZw==", 133 | "dev": true, 134 | "dependencies": { 135 | "debug": "^4.1.1", 136 | "env-paths": "^2.2.0", 137 | "fs-extra": "^8.1.0", 138 | "got": "^9.6.0", 139 | "progress": "^2.0.3", 140 | "semver": "^6.2.0", 141 | "sumchecker": "^3.0.1" 142 | }, 143 | "engines": { 144 | "node": ">=8.6" 145 | }, 146 | "optionalDependencies": { 147 | "global-agent": "^3.0.0", 148 | "global-tunnel-ng": "^2.7.1" 149 | } 150 | }, 151 | "node_modules/@lezer/common": { 152 | "version": "0.15.12", 153 | "resolved": "https://registry.npmjs.org/@lezer/common/-/common-0.15.12.tgz", 154 | "integrity": "sha512-edfwCxNLnzq5pBA/yaIhwJ3U3Kz8VAUOTRg0hhxaizaI1N+qxV7EXDv/kLCkLeq2RzSFvxexlaj5Mzfn2kY0Ig==", 155 | "dev": true 156 | }, 157 | "node_modules/@lezer/lr": { 158 | "version": "0.15.8", 159 | "resolved": "https://registry.npmjs.org/@lezer/lr/-/lr-0.15.8.tgz", 160 | "integrity": "sha512-bM6oE6VQZ6hIFxDNKk8bKPa14hqFrV07J/vHGOeiAbJReIaQXmkVb6xQu4MR+JBTLa5arGRyAAjJe1qaQt3Uvg==", 161 | "dev": true, 162 | "dependencies": { 163 | "@lezer/common": "^0.15.0" 164 | } 165 | }, 166 | "node_modules/@mischnic/json-sourcemap": { 167 | "version": "0.1.0", 168 | "resolved": "https://registry.npmjs.org/@mischnic/json-sourcemap/-/json-sourcemap-0.1.0.tgz", 169 | "integrity": "sha512-dQb3QnfNqmQNYA4nFSN/uLaByIic58gOXq4Y4XqLOWmOrw73KmJPt/HLyG0wvn1bnR6mBKs/Uwvkh+Hns1T0XA==", 170 | "dev": true, 171 | "dependencies": { 172 | "@lezer/common": "^0.15.7", 173 | "@lezer/lr": "^0.15.4", 174 | "json5": "^2.2.1" 175 | }, 176 | "engines": { 177 | "node": ">=12.0.0" 178 | } 179 | }, 180 | "node_modules/@msgpackr-extract/msgpackr-extract-darwin-arm64": { 181 | "version": "2.0.2", 182 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-darwin-arm64/-/msgpackr-extract-darwin-arm64-2.0.2.tgz", 183 | "integrity": "sha512-FMX5i7a+ojIguHpWbzh5MCsCouJkwf4z4ejdUY/fsgB9Vkdak4ZnoIEskOyOUMMB4lctiZFGszFQJXUeFL8tRg==", 184 | "cpu": [ 185 | "arm64" 186 | ], 187 | "dev": true, 188 | "optional": true, 189 | "os": [ 190 | "darwin" 191 | ] 192 | }, 193 | "node_modules/@msgpackr-extract/msgpackr-extract-darwin-x64": { 194 | "version": "2.0.2", 195 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-darwin-x64/-/msgpackr-extract-darwin-x64-2.0.2.tgz", 196 | "integrity": "sha512-DznYtF3lHuZDSRaIOYeif4JgO0NtO2Xf8DsngAugMx/bUdTFbg86jDTmkVJBNmV+cxszz6OjGvinnS8AbJ342g==", 197 | "cpu": [ 198 | "x64" 199 | ], 200 | "dev": true, 201 | "optional": true, 202 | "os": [ 203 | "darwin" 204 | ] 205 | }, 206 | "node_modules/@msgpackr-extract/msgpackr-extract-linux-arm": { 207 | "version": "2.0.2", 208 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-arm/-/msgpackr-extract-linux-arm-2.0.2.tgz", 209 | "integrity": "sha512-Gy9+c3Wj+rUlD3YvCZTi92gs+cRX7ZQogtwq0IhRenloTTlsbpezNgk6OCkt59V4ATEWSic9rbU92H/l7XsRvA==", 210 | "cpu": [ 211 | "arm" 212 | ], 213 | "dev": true, 214 | "optional": true, 215 | "os": [ 216 | "linux" 217 | ] 218 | }, 219 | "node_modules/@msgpackr-extract/msgpackr-extract-linux-arm64": { 220 | "version": "2.0.2", 221 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-arm64/-/msgpackr-extract-linux-arm64-2.0.2.tgz", 222 | "integrity": "sha512-b0jMEo566YdM2K+BurSed7bswjo3a6bcdw5ETqoIfSuxKuRLPfAiOjVbZyZBgx3J/TAM/QrvEQ/VN89A0ZAxSg==", 223 | "cpu": [ 224 | "arm64" 225 | ], 226 | "dev": true, 227 | "optional": true, 228 | "os": [ 229 | "linux" 230 | ] 231 | }, 232 | "node_modules/@msgpackr-extract/msgpackr-extract-linux-x64": { 233 | "version": "2.0.2", 234 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-x64/-/msgpackr-extract-linux-x64-2.0.2.tgz", 235 | "integrity": "sha512-zrBHaePwcv4cQXxzYgNj0+A8I1uVN97E7/3LmkRocYZ+rMwUsnPpp4RuTAHSRoKlTQV3nSdCQW4Qdt4MXw/iHw==", 236 | "cpu": [ 237 | "x64" 238 | ], 239 | "dev": true, 240 | "optional": true, 241 | "os": [ 242 | "linux" 243 | ] 244 | }, 245 | "node_modules/@msgpackr-extract/msgpackr-extract-win32-x64": { 246 | "version": "2.0.2", 247 | "resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-win32-x64/-/msgpackr-extract-win32-x64-2.0.2.tgz", 248 | "integrity": "sha512-fpnI00dt+yO1cKx9qBXelKhPBdEgvc8ZPav1+0r09j0woYQU2N79w/jcGawSY5UGlgQ3vjaJsFHnGbGvvqdLzg==", 249 | "cpu": [ 250 | "x64" 251 | ], 252 | "dev": true, 253 | "optional": true, 254 | "os": [ 255 | "win32" 256 | ] 257 | }, 258 | "node_modules/@parcel/bundler-default": { 259 | "version": "2.5.0", 260 | "resolved": "https://registry.npmjs.org/@parcel/bundler-default/-/bundler-default-2.5.0.tgz", 261 | "integrity": "sha512-7CJzE17SirCXjcRgBcnqWO/5EOA1raq/3OIKtT4cxbjpDQGHZpjpEEZiMNRpEpdNMxDSlsG8mAkXTYGL2VVWRw==", 262 | "dev": true, 263 | "dependencies": { 264 | "@parcel/diagnostic": "2.5.0", 265 | "@parcel/hash": "2.5.0", 266 | "@parcel/plugin": "2.5.0", 267 | "@parcel/utils": "2.5.0", 268 | "nullthrows": "^1.1.1" 269 | }, 270 | "engines": { 271 | "node": ">= 12.0.0", 272 | "parcel": "^2.5.0" 273 | }, 274 | "funding": { 275 | "type": "opencollective", 276 | "url": "https://opencollective.com/parcel" 277 | } 278 | }, 279 | "node_modules/@parcel/cache": { 280 | "version": "2.5.0", 281 | "resolved": "https://registry.npmjs.org/@parcel/cache/-/cache-2.5.0.tgz", 282 | "integrity": "sha512-3kOO3cZQv0FAKhrMHGLdb4Qtzpmy78Q6jPN3u8eCY4yqeDTnyQBZvWNHoyCm5WlmL8y6Q6REYMbETLxSH1ggAQ==", 283 | "dev": true, 284 | "dependencies": { 285 | "@parcel/fs": "2.5.0", 286 | "@parcel/logger": "2.5.0", 287 | "@parcel/utils": "2.5.0", 288 | "lmdb": "2.2.4" 289 | }, 290 | "engines": { 291 | "node": ">= 12.0.0" 292 | }, 293 | "funding": { 294 | "type": "opencollective", 295 | "url": "https://opencollective.com/parcel" 296 | }, 297 | "peerDependencies": { 298 | "@parcel/core": "^2.5.0" 299 | } 300 | }, 301 | "node_modules/@parcel/codeframe": { 302 | "version": "2.5.0", 303 | "resolved": "https://registry.npmjs.org/@parcel/codeframe/-/codeframe-2.5.0.tgz", 304 | "integrity": "sha512-qafqL8Vu2kr932cCWESoDEEoAeKVi7/xdzTBuhzEJng1AfmRT0rCbt/P4ao3RjiDyozPSjXsHOqM6GDZcto4eQ==", 305 | "dev": true, 306 | "dependencies": { 307 | "chalk": "^4.1.0" 308 | }, 309 | "engines": { 310 | "node": ">= 12.0.0" 311 | }, 312 | "funding": { 313 | "type": "opencollective", 314 | "url": "https://opencollective.com/parcel" 315 | } 316 | }, 317 | "node_modules/@parcel/compressor-raw": { 318 | "version": "2.5.0", 319 | "resolved": "https://registry.npmjs.org/@parcel/compressor-raw/-/compressor-raw-2.5.0.tgz", 320 | "integrity": "sha512-I5Zs+2f1ue4sTPdfT8BNsLfTZl48sMWLk2Io3elUJjH/SS9kO7ut5ChkuJtt77ZS35m0OF+ZCt3ICTJdnDG8eA==", 321 | "dev": true, 322 | "dependencies": { 323 | "@parcel/plugin": "2.5.0" 324 | }, 325 | "engines": { 326 | "node": ">= 12.0.0", 327 | "parcel": "^2.5.0" 328 | }, 329 | "funding": { 330 | "type": "opencollective", 331 | "url": "https://opencollective.com/parcel" 332 | } 333 | }, 334 | "node_modules/@parcel/config-default": { 335 | "version": "2.5.0", 336 | "resolved": "https://registry.npmjs.org/@parcel/config-default/-/config-default-2.5.0.tgz", 337 | "integrity": "sha512-r30V61958SONvP9I8KV8s44ZOFq0H219VyFjPysraSabHjZ+KMaCTQOuqaDtUMa272sHUQkBcZxKYj5jYPJlZg==", 338 | "dev": true, 339 | "dependencies": { 340 | "@parcel/bundler-default": "2.5.0", 341 | "@parcel/compressor-raw": "2.5.0", 342 | "@parcel/namer-default": "2.5.0", 343 | "@parcel/optimizer-css": "2.5.0", 344 | "@parcel/optimizer-htmlnano": "2.5.0", 345 | "@parcel/optimizer-image": "2.5.0", 346 | "@parcel/optimizer-svgo": "2.5.0", 347 | "@parcel/optimizer-terser": "2.5.0", 348 | "@parcel/packager-css": "2.5.0", 349 | "@parcel/packager-html": "2.5.0", 350 | "@parcel/packager-js": "2.5.0", 351 | "@parcel/packager-raw": "2.5.0", 352 | "@parcel/packager-svg": "2.5.0", 353 | "@parcel/reporter-dev-server": "2.5.0", 354 | "@parcel/resolver-default": "2.5.0", 355 | "@parcel/runtime-browser-hmr": "2.5.0", 356 | "@parcel/runtime-js": "2.5.0", 357 | "@parcel/runtime-react-refresh": "2.5.0", 358 | "@parcel/runtime-service-worker": "2.5.0", 359 | "@parcel/transformer-babel": "2.5.0", 360 | "@parcel/transformer-css": "2.5.0", 361 | "@parcel/transformer-html": "2.5.0", 362 | "@parcel/transformer-image": "2.5.0", 363 | "@parcel/transformer-js": "2.5.0", 364 | "@parcel/transformer-json": "2.5.0", 365 | "@parcel/transformer-postcss": "2.5.0", 366 | "@parcel/transformer-posthtml": "2.5.0", 367 | "@parcel/transformer-raw": "2.5.0", 368 | "@parcel/transformer-react-refresh-wrap": "2.5.0", 369 | "@parcel/transformer-svg": "2.5.0" 370 | }, 371 | "funding": { 372 | "type": "opencollective", 373 | "url": "https://opencollective.com/parcel" 374 | }, 375 | "peerDependencies": { 376 | "@parcel/core": "^2.5.0" 377 | } 378 | }, 379 | "node_modules/@parcel/core": { 380 | "version": "2.5.0", 381 | "resolved": "https://registry.npmjs.org/@parcel/core/-/core-2.5.0.tgz", 382 | "integrity": "sha512-dygDmPsfAYJKTnUftcbEzjCik7AAaPbFvJW8ETYz8diyjkAG9y6hvCAZIrJE5pNOjFzg32en4v4UWv8Sqlzl9g==", 383 | "dev": true, 384 | "dependencies": { 385 | "@mischnic/json-sourcemap": "^0.1.0", 386 | "@parcel/cache": "2.5.0", 387 | "@parcel/diagnostic": "2.5.0", 388 | "@parcel/events": "2.5.0", 389 | "@parcel/fs": "2.5.0", 390 | "@parcel/graph": "2.5.0", 391 | "@parcel/hash": "2.5.0", 392 | "@parcel/logger": "2.5.0", 393 | "@parcel/package-manager": "2.5.0", 394 | "@parcel/plugin": "2.5.0", 395 | "@parcel/source-map": "^2.0.0", 396 | "@parcel/types": "2.5.0", 397 | "@parcel/utils": "2.5.0", 398 | "@parcel/workers": "2.5.0", 399 | "abortcontroller-polyfill": "^1.1.9", 400 | "base-x": "^3.0.8", 401 | "browserslist": "^4.6.6", 402 | "clone": "^2.1.1", 403 | "dotenv": "^7.0.0", 404 | "dotenv-expand": "^5.1.0", 405 | "json5": "^2.2.0", 406 | "msgpackr": "^1.5.4", 407 | "nullthrows": "^1.1.1", 408 | "semver": "^5.7.1" 409 | }, 410 | "engines": { 411 | "node": ">= 12.0.0" 412 | }, 413 | "funding": { 414 | "type": "opencollective", 415 | "url": "https://opencollective.com/parcel" 416 | } 417 | }, 418 | "node_modules/@parcel/core/node_modules/semver": { 419 | "version": "5.7.1", 420 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 421 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 422 | "dev": true, 423 | "bin": { 424 | "semver": "bin/semver" 425 | } 426 | }, 427 | "node_modules/@parcel/css": { 428 | "version": "1.8.3", 429 | "resolved": "https://registry.npmjs.org/@parcel/css/-/css-1.8.3.tgz", 430 | "integrity": "sha512-6qUN4iicr8f9Q6UUZttwwHMzrb65BRX46PHWq0icA4KEmvmfR9cSYlp/hJH8F4stg3Wncx12Bnw+EuPf5OAEPQ==", 431 | "dev": true, 432 | "dependencies": { 433 | "detect-libc": "^1.0.3" 434 | }, 435 | "engines": { 436 | "node": ">= 12.0.0" 437 | }, 438 | "funding": { 439 | "type": "opencollective", 440 | "url": "https://opencollective.com/parcel" 441 | }, 442 | "optionalDependencies": { 443 | "@parcel/css-darwin-arm64": "1.8.3", 444 | "@parcel/css-darwin-x64": "1.8.3", 445 | "@parcel/css-linux-arm-gnueabihf": "1.8.3", 446 | "@parcel/css-linux-arm64-gnu": "1.8.3", 447 | "@parcel/css-linux-arm64-musl": "1.8.3", 448 | "@parcel/css-linux-x64-gnu": "1.8.3", 449 | "@parcel/css-linux-x64-musl": "1.8.3", 450 | "@parcel/css-win32-x64-msvc": "1.8.3" 451 | } 452 | }, 453 | "node_modules/@parcel/css-darwin-arm64": { 454 | "version": "1.8.3", 455 | "resolved": "https://registry.npmjs.org/@parcel/css-darwin-arm64/-/css-darwin-arm64-1.8.3.tgz", 456 | "integrity": "sha512-qh/Ig6GfVjGoiGSWjIYDo6Ghwmyy/9BXvYS1l3R+Bp50F300cq84Czfl6wxaL+aFmghdHzhjJuGfWmZlcYliPA==", 457 | "cpu": [ 458 | "arm64" 459 | ], 460 | "dev": true, 461 | "optional": true, 462 | "os": [ 463 | "darwin" 464 | ], 465 | "engines": { 466 | "node": ">= 12.0.0" 467 | }, 468 | "funding": { 469 | "type": "opencollective", 470 | "url": "https://opencollective.com/parcel" 471 | } 472 | }, 473 | "node_modules/@parcel/css-darwin-x64": { 474 | "version": "1.8.3", 475 | "resolved": "https://registry.npmjs.org/@parcel/css-darwin-x64/-/css-darwin-x64-1.8.3.tgz", 476 | "integrity": "sha512-gTUIoRgwyYr4UuH7sSn3gOuMlIshJBOJLmjL+E/mR5lqdYabguiKiRORvkrnb/gHBmOUF9re0RcTaFmJ2VOAlg==", 477 | "cpu": [ 478 | "x64" 479 | ], 480 | "dev": true, 481 | "optional": true, 482 | "os": [ 483 | "darwin" 484 | ], 485 | "engines": { 486 | "node": ">= 12.0.0" 487 | }, 488 | "funding": { 489 | "type": "opencollective", 490 | "url": "https://opencollective.com/parcel" 491 | } 492 | }, 493 | "node_modules/@parcel/css-linux-arm-gnueabihf": { 494 | "version": "1.8.3", 495 | "resolved": "https://registry.npmjs.org/@parcel/css-linux-arm-gnueabihf/-/css-linux-arm-gnueabihf-1.8.3.tgz", 496 | "integrity": "sha512-4P1r0BvL9dPz70py2xLg/jEvWJmKNyokPgafyrDP+GbpPTfH5NYJJkVRGo/TkKsp3Rv8SJhV9fdlpFKC6BI92A==", 497 | "cpu": [ 498 | "arm" 499 | ], 500 | "dev": true, 501 | "optional": true, 502 | "os": [ 503 | "linux" 504 | ], 505 | "engines": { 506 | "node": ">= 12.0.0" 507 | }, 508 | "funding": { 509 | "type": "opencollective", 510 | "url": "https://opencollective.com/parcel" 511 | } 512 | }, 513 | "node_modules/@parcel/css-linux-arm64-gnu": { 514 | "version": "1.8.3", 515 | "resolved": "https://registry.npmjs.org/@parcel/css-linux-arm64-gnu/-/css-linux-arm64-gnu-1.8.3.tgz", 516 | "integrity": "sha512-1fUy94eaqdzum+C7bsYVF2AgxjLGR/qppArn/4HTQyydHR5QeV+Uoyqo5vdnO5Vclj8eQwlgR9OyAOlmzXxFDA==", 517 | "cpu": [ 518 | "arm64" 519 | ], 520 | "dev": true, 521 | "optional": true, 522 | "os": [ 523 | "linux" 524 | ], 525 | "engines": { 526 | "node": ">= 12.0.0" 527 | }, 528 | "funding": { 529 | "type": "opencollective", 530 | "url": "https://opencollective.com/parcel" 531 | } 532 | }, 533 | "node_modules/@parcel/css-linux-arm64-musl": { 534 | "version": "1.8.3", 535 | "resolved": "https://registry.npmjs.org/@parcel/css-linux-arm64-musl/-/css-linux-arm64-musl-1.8.3.tgz", 536 | "integrity": "sha512-ct1QRK5gAP8sO22NZ7RULZQB7dbHpou+WMa4z0LJb+Fho13a1JNw931vNHbeI5cRr1fCTDq76pz/+Valgetzcw==", 537 | "cpu": [ 538 | "arm64" 539 | ], 540 | "dev": true, 541 | "optional": true, 542 | "os": [ 543 | "linux" 544 | ], 545 | "engines": { 546 | "node": ">= 12.0.0" 547 | }, 548 | "funding": { 549 | "type": "opencollective", 550 | "url": "https://opencollective.com/parcel" 551 | } 552 | }, 553 | "node_modules/@parcel/css-linux-x64-gnu": { 554 | "version": "1.8.3", 555 | "resolved": "https://registry.npmjs.org/@parcel/css-linux-x64-gnu/-/css-linux-x64-gnu-1.8.3.tgz", 556 | "integrity": "sha512-pg/mahoogzjbaZcW76rrTZ64tEu8Wok4Gm0sW/dXHJEJD2QVJ6GxLP4UVNBuhaV0GrNFHggp9pcdhTtLGkKl/g==", 557 | "cpu": [ 558 | "x64" 559 | ], 560 | "dev": true, 561 | "optional": true, 562 | "os": [ 563 | "linux" 564 | ], 565 | "engines": { 566 | "node": ">= 12.0.0" 567 | }, 568 | "funding": { 569 | "type": "opencollective", 570 | "url": "https://opencollective.com/parcel" 571 | } 572 | }, 573 | "node_modules/@parcel/css-linux-x64-musl": { 574 | "version": "1.8.3", 575 | "resolved": "https://registry.npmjs.org/@parcel/css-linux-x64-musl/-/css-linux-x64-musl-1.8.3.tgz", 576 | "integrity": "sha512-4Iwawy28HQ2yAgbuyR60bgO+8oE+OiWpE02eNjbgqnDpTsfmXFMt4l5OYgZwJJ7DlaZqm+/yO8RPMd+EzwtNzg==", 577 | "cpu": [ 578 | "x64" 579 | ], 580 | "dev": true, 581 | "optional": true, 582 | "os": [ 583 | "linux" 584 | ], 585 | "engines": { 586 | "node": ">= 12.0.0" 587 | }, 588 | "funding": { 589 | "type": "opencollective", 590 | "url": "https://opencollective.com/parcel" 591 | } 592 | }, 593 | "node_modules/@parcel/css-win32-x64-msvc": { 594 | "version": "1.8.3", 595 | "resolved": "https://registry.npmjs.org/@parcel/css-win32-x64-msvc/-/css-win32-x64-msvc-1.8.3.tgz", 596 | "integrity": "sha512-vnHUdzIVjqONa5ALFzMJ3ZHt6NiaYTHW/lqzP+AR4l+bq+UTXD2Q75/RgirY5NYwdfy1VPy/jI82jAtLOCymkw==", 597 | "cpu": [ 598 | "x64" 599 | ], 600 | "dev": true, 601 | "optional": true, 602 | "os": [ 603 | "win32" 604 | ], 605 | "engines": { 606 | "node": ">= 12.0.0" 607 | }, 608 | "funding": { 609 | "type": "opencollective", 610 | "url": "https://opencollective.com/parcel" 611 | } 612 | }, 613 | "node_modules/@parcel/diagnostic": { 614 | "version": "2.5.0", 615 | "resolved": "https://registry.npmjs.org/@parcel/diagnostic/-/diagnostic-2.5.0.tgz", 616 | "integrity": "sha512-KiMGGRpEV7wl5gjcxBKcgX84a+cG+IEn94gwy5LK3lENR09nuKShqqgKGAmj/17CobJgw1QNP94/H4Md+oxIWg==", 617 | "dev": true, 618 | "dependencies": { 619 | "@mischnic/json-sourcemap": "^0.1.0", 620 | "nullthrows": "^1.1.1" 621 | }, 622 | "engines": { 623 | "node": ">= 12.0.0" 624 | }, 625 | "funding": { 626 | "type": "opencollective", 627 | "url": "https://opencollective.com/parcel" 628 | } 629 | }, 630 | "node_modules/@parcel/events": { 631 | "version": "2.5.0", 632 | "resolved": "https://registry.npmjs.org/@parcel/events/-/events-2.5.0.tgz", 633 | "integrity": "sha512-Gc2LPwL1H34Ony5MENbKZg7wvCscZ4x9y7Fu92sfbdWpLo3K13hVtsX3TMIIgYt3B7R7OmO8yR880U2T+JfVkQ==", 634 | "dev": true, 635 | "engines": { 636 | "node": ">= 12.0.0" 637 | }, 638 | "funding": { 639 | "type": "opencollective", 640 | "url": "https://opencollective.com/parcel" 641 | } 642 | }, 643 | "node_modules/@parcel/fs": { 644 | "version": "2.5.0", 645 | "resolved": "https://registry.npmjs.org/@parcel/fs/-/fs-2.5.0.tgz", 646 | "integrity": "sha512-YYr14BWtx/bJ+hu6PPQQ6G/3omOTWgVqEw+UFI3iQH3P6+e0LRXW/Ja1yAcJeepGcTwIP0opnXZBQOm8PBQ2SA==", 647 | "dev": true, 648 | "dependencies": { 649 | "@parcel/fs-search": "2.5.0", 650 | "@parcel/types": "2.5.0", 651 | "@parcel/utils": "2.5.0", 652 | "@parcel/watcher": "^2.0.0", 653 | "@parcel/workers": "2.5.0" 654 | }, 655 | "engines": { 656 | "node": ">= 12.0.0" 657 | }, 658 | "funding": { 659 | "type": "opencollective", 660 | "url": "https://opencollective.com/parcel" 661 | }, 662 | "peerDependencies": { 663 | "@parcel/core": "^2.5.0" 664 | } 665 | }, 666 | "node_modules/@parcel/fs-search": { 667 | "version": "2.5.0", 668 | "resolved": "https://registry.npmjs.org/@parcel/fs-search/-/fs-search-2.5.0.tgz", 669 | "integrity": "sha512-uBONkz9ZCNSOqbPGWJY3MNl+pqBTfvzHH9+4UhzHEHPArvK2oD0+syYPVE60+zGrxybXTESYMCJp4bHvH6Z2hA==", 670 | "dev": true, 671 | "dependencies": { 672 | "detect-libc": "^1.0.3" 673 | }, 674 | "engines": { 675 | "node": ">= 12.0.0" 676 | }, 677 | "funding": { 678 | "type": "opencollective", 679 | "url": "https://opencollective.com/parcel" 680 | } 681 | }, 682 | "node_modules/@parcel/graph": { 683 | "version": "2.5.0", 684 | "resolved": "https://registry.npmjs.org/@parcel/graph/-/graph-2.5.0.tgz", 685 | "integrity": "sha512-qa2VtG08dJyTaWrxYAkMIlkoDRSPoiqLDNxxHKplkcxAjXBUw0/AkWaz82VO5r1G6jfOj+nM30ajH9uygZYwbw==", 686 | "dev": true, 687 | "dependencies": { 688 | "@parcel/utils": "2.5.0", 689 | "nullthrows": "^1.1.1" 690 | }, 691 | "engines": { 692 | "node": ">= 12.0.0" 693 | }, 694 | "funding": { 695 | "type": "opencollective", 696 | "url": "https://opencollective.com/parcel" 697 | } 698 | }, 699 | "node_modules/@parcel/hash": { 700 | "version": "2.5.0", 701 | "resolved": "https://registry.npmjs.org/@parcel/hash/-/hash-2.5.0.tgz", 702 | "integrity": "sha512-47JL0XpB7UvIW6Ijf8vv+yVMt9dLvB/lRlBHFmAkmovisueVMVbYD7smxVZnCSehD8UH8BcymKbMzyL5dimgoQ==", 703 | "dev": true, 704 | "dependencies": { 705 | "detect-libc": "^1.0.3", 706 | "xxhash-wasm": "^0.4.2" 707 | }, 708 | "engines": { 709 | "node": ">= 12.0.0" 710 | }, 711 | "funding": { 712 | "type": "opencollective", 713 | "url": "https://opencollective.com/parcel" 714 | } 715 | }, 716 | "node_modules/@parcel/logger": { 717 | "version": "2.5.0", 718 | "resolved": "https://registry.npmjs.org/@parcel/logger/-/logger-2.5.0.tgz", 719 | "integrity": "sha512-pT1L3ceH6trL1N3I3r2HawPjz/PCubOo/Kazu7IeXsMsKVjj1a6AeieZHzkNZIbhiGPtm/cHbBNLz2zTWDLeOA==", 720 | "dev": true, 721 | "dependencies": { 722 | "@parcel/diagnostic": "2.5.0", 723 | "@parcel/events": "2.5.0" 724 | }, 725 | "engines": { 726 | "node": ">= 12.0.0" 727 | }, 728 | "funding": { 729 | "type": "opencollective", 730 | "url": "https://opencollective.com/parcel" 731 | } 732 | }, 733 | "node_modules/@parcel/markdown-ansi": { 734 | "version": "2.5.0", 735 | "resolved": "https://registry.npmjs.org/@parcel/markdown-ansi/-/markdown-ansi-2.5.0.tgz", 736 | "integrity": "sha512-ixkNF3KWIqxMlfxTe9Gb2cp/uNmklQev8VEUxujMVxmUfGyQs4859zdJIQlIinabWYhArhsXATkVf3MzCUN6TQ==", 737 | "dev": true, 738 | "dependencies": { 739 | "chalk": "^4.1.0" 740 | }, 741 | "engines": { 742 | "node": ">= 12.0.0" 743 | }, 744 | "funding": { 745 | "type": "opencollective", 746 | "url": "https://opencollective.com/parcel" 747 | } 748 | }, 749 | "node_modules/@parcel/namer-default": { 750 | "version": "2.5.0", 751 | "resolved": "https://registry.npmjs.org/@parcel/namer-default/-/namer-default-2.5.0.tgz", 752 | "integrity": "sha512-ahGQqHJzsWE5Qux8zXMAU+lyNBOl+ZpcOFzRGE2DWOsmAlytsHl7DBVCQvzUyNBFg1/HmIj+7D4efv2kjR7rTg==", 753 | "dev": true, 754 | "dependencies": { 755 | "@parcel/diagnostic": "2.5.0", 756 | "@parcel/plugin": "2.5.0", 757 | "nullthrows": "^1.1.1" 758 | }, 759 | "engines": { 760 | "node": ">= 12.0.0", 761 | "parcel": "^2.5.0" 762 | }, 763 | "funding": { 764 | "type": "opencollective", 765 | "url": "https://opencollective.com/parcel" 766 | } 767 | }, 768 | "node_modules/@parcel/node-resolver-core": { 769 | "version": "2.5.0", 770 | "resolved": "https://registry.npmjs.org/@parcel/node-resolver-core/-/node-resolver-core-2.5.0.tgz", 771 | "integrity": "sha512-XQvpguiIwQcu75cscLDFOVhjsjuPzXbuMaaZ7XxxUEl0PscIgu/GfKYxTfTruN3cRl+CaQH6qBAMfjLaFng6lQ==", 772 | "dev": true, 773 | "dependencies": { 774 | "@parcel/diagnostic": "2.5.0", 775 | "@parcel/utils": "2.5.0", 776 | "nullthrows": "^1.1.1" 777 | }, 778 | "engines": { 779 | "node": ">= 12.0.0" 780 | }, 781 | "funding": { 782 | "type": "opencollective", 783 | "url": "https://opencollective.com/parcel" 784 | } 785 | }, 786 | "node_modules/@parcel/optimizer-css": { 787 | "version": "2.5.0", 788 | "resolved": "https://registry.npmjs.org/@parcel/optimizer-css/-/optimizer-css-2.5.0.tgz", 789 | "integrity": "sha512-J00bLF+4SsnKc+YbYrNuBr44/zz3cg++CoXteXhH27PxP1rScGQx36Rui8WORgil5mlX2VYN79DuqJC7V3Ynbg==", 790 | "dev": true, 791 | "dependencies": { 792 | "@parcel/css": "^1.8.1", 793 | "@parcel/diagnostic": "2.5.0", 794 | "@parcel/plugin": "2.5.0", 795 | "@parcel/source-map": "^2.0.0", 796 | "@parcel/utils": "2.5.0", 797 | "browserslist": "^4.6.6", 798 | "nullthrows": "^1.1.1" 799 | }, 800 | "engines": { 801 | "node": ">= 12.0.0", 802 | "parcel": "^2.5.0" 803 | }, 804 | "funding": { 805 | "type": "opencollective", 806 | "url": "https://opencollective.com/parcel" 807 | } 808 | }, 809 | "node_modules/@parcel/optimizer-htmlnano": { 810 | "version": "2.5.0", 811 | "resolved": "https://registry.npmjs.org/@parcel/optimizer-htmlnano/-/optimizer-htmlnano-2.5.0.tgz", 812 | "integrity": "sha512-Fr0zPqgxoNaOVdROAjNGDWCts3+wByNQ82Mxhu8Tzc25A2cPjcr1H2sa/TE3hf79c92DxdKf2FaC1ZOgR5YPdg==", 813 | "dev": true, 814 | "dependencies": { 815 | "@parcel/plugin": "2.5.0", 816 | "htmlnano": "^2.0.0", 817 | "nullthrows": "^1.1.1", 818 | "posthtml": "^0.16.5", 819 | "svgo": "^2.4.0" 820 | }, 821 | "engines": { 822 | "node": ">= 12.0.0", 823 | "parcel": "^2.5.0" 824 | }, 825 | "funding": { 826 | "type": "opencollective", 827 | "url": "https://opencollective.com/parcel" 828 | } 829 | }, 830 | "node_modules/@parcel/optimizer-image": { 831 | "version": "2.5.0", 832 | "resolved": "https://registry.npmjs.org/@parcel/optimizer-image/-/optimizer-image-2.5.0.tgz", 833 | "integrity": "sha512-nbo2pdnAt21WLGjzTpsE8ZEL0xNoP7c3wBj9y70Pysmasg1SrRVCbfE8jTy+lHBQwq2yjC6lV/Usv+9lfA7S/w==", 834 | "dev": true, 835 | "dependencies": { 836 | "@parcel/diagnostic": "2.5.0", 837 | "@parcel/plugin": "2.5.0", 838 | "@parcel/utils": "2.5.0", 839 | "@parcel/workers": "2.5.0", 840 | "detect-libc": "^1.0.3" 841 | }, 842 | "engines": { 843 | "node": ">= 12.0.0", 844 | "parcel": "^2.5.0" 845 | }, 846 | "funding": { 847 | "type": "opencollective", 848 | "url": "https://opencollective.com/parcel" 849 | } 850 | }, 851 | "node_modules/@parcel/optimizer-svgo": { 852 | "version": "2.5.0", 853 | "resolved": "https://registry.npmjs.org/@parcel/optimizer-svgo/-/optimizer-svgo-2.5.0.tgz", 854 | "integrity": "sha512-pgZqwU0RLc/wr4WcQY/W1GJmddnEANDEpz1mdppUOqBz1EfTQ7zh5NgUA3hV1i05Hbecp3mHSvXJPV0mhNOl5Q==", 855 | "dev": true, 856 | "dependencies": { 857 | "@parcel/diagnostic": "2.5.0", 858 | "@parcel/plugin": "2.5.0", 859 | "@parcel/utils": "2.5.0", 860 | "svgo": "^2.4.0" 861 | }, 862 | "engines": { 863 | "node": ">= 12.0.0", 864 | "parcel": "^2.5.0" 865 | }, 866 | "funding": { 867 | "type": "opencollective", 868 | "url": "https://opencollective.com/parcel" 869 | } 870 | }, 871 | "node_modules/@parcel/optimizer-terser": { 872 | "version": "2.5.0", 873 | "resolved": "https://registry.npmjs.org/@parcel/optimizer-terser/-/optimizer-terser-2.5.0.tgz", 874 | "integrity": "sha512-PZ3UHBGfjE49/Jloopsd38Hxg4qzsrdepWP53mCuVP7Aw605Y4QtYuB1ho3VV0oXfKQVq+uI7lVIBsuW4K6vqA==", 875 | "dev": true, 876 | "dependencies": { 877 | "@parcel/diagnostic": "2.5.0", 878 | "@parcel/plugin": "2.5.0", 879 | "@parcel/source-map": "^2.0.0", 880 | "@parcel/utils": "2.5.0", 881 | "nullthrows": "^1.1.1", 882 | "terser": "^5.2.0" 883 | }, 884 | "engines": { 885 | "node": ">= 12.0.0", 886 | "parcel": "^2.5.0" 887 | }, 888 | "funding": { 889 | "type": "opencollective", 890 | "url": "https://opencollective.com/parcel" 891 | } 892 | }, 893 | "node_modules/@parcel/package-manager": { 894 | "version": "2.5.0", 895 | "resolved": "https://registry.npmjs.org/@parcel/package-manager/-/package-manager-2.5.0.tgz", 896 | "integrity": "sha512-zTuF55/lITUjw9dUU/X0HiF++589xbPXw/zUiG9T6s8BQThLvrxAhYP89S719pw7cTqDimGkTxnIuK+a0djEkg==", 897 | "dev": true, 898 | "dependencies": { 899 | "@parcel/diagnostic": "2.5.0", 900 | "@parcel/fs": "2.5.0", 901 | "@parcel/logger": "2.5.0", 902 | "@parcel/types": "2.5.0", 903 | "@parcel/utils": "2.5.0", 904 | "@parcel/workers": "2.5.0", 905 | "semver": "^5.7.1" 906 | }, 907 | "engines": { 908 | "node": ">= 12.0.0" 909 | }, 910 | "funding": { 911 | "type": "opencollective", 912 | "url": "https://opencollective.com/parcel" 913 | }, 914 | "peerDependencies": { 915 | "@parcel/core": "^2.5.0" 916 | } 917 | }, 918 | "node_modules/@parcel/package-manager/node_modules/semver": { 919 | "version": "5.7.1", 920 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 921 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 922 | "dev": true, 923 | "bin": { 924 | "semver": "bin/semver" 925 | } 926 | }, 927 | "node_modules/@parcel/packager-css": { 928 | "version": "2.5.0", 929 | "resolved": "https://registry.npmjs.org/@parcel/packager-css/-/packager-css-2.5.0.tgz", 930 | "integrity": "sha512-c0mGBFdVSPhAxaX3+zN8KEIqOOUhkIPKbZex1pnGYfy03Qe2/Mb4nyt5DAGlw9gjka1UCHIN/wszLmKC8YyUeg==", 931 | "dev": true, 932 | "dependencies": { 933 | "@parcel/plugin": "2.5.0", 934 | "@parcel/source-map": "^2.0.0", 935 | "@parcel/utils": "2.5.0", 936 | "nullthrows": "^1.1.1" 937 | }, 938 | "engines": { 939 | "node": ">= 12.0.0", 940 | "parcel": "^2.5.0" 941 | }, 942 | "funding": { 943 | "type": "opencollective", 944 | "url": "https://opencollective.com/parcel" 945 | } 946 | }, 947 | "node_modules/@parcel/packager-html": { 948 | "version": "2.5.0", 949 | "resolved": "https://registry.npmjs.org/@parcel/packager-html/-/packager-html-2.5.0.tgz", 950 | "integrity": "sha512-ZFGUPRMWKrm8kQHdkEJ5S22C05qpSymx+o+57EfuNjCrGyj3M59WyGYYXYJ175bFYZ/jp5yy+VxMh6fZefe+Pw==", 951 | "dev": true, 952 | "dependencies": { 953 | "@parcel/plugin": "2.5.0", 954 | "@parcel/types": "2.5.0", 955 | "@parcel/utils": "2.5.0", 956 | "nullthrows": "^1.1.1", 957 | "posthtml": "^0.16.5" 958 | }, 959 | "engines": { 960 | "node": ">= 12.0.0", 961 | "parcel": "^2.5.0" 962 | }, 963 | "funding": { 964 | "type": "opencollective", 965 | "url": "https://opencollective.com/parcel" 966 | } 967 | }, 968 | "node_modules/@parcel/packager-js": { 969 | "version": "2.5.0", 970 | "resolved": "https://registry.npmjs.org/@parcel/packager-js/-/packager-js-2.5.0.tgz", 971 | "integrity": "sha512-aJAKOTgXdxO3V9O7+2DCVOtne128WwXmUAOVThnMRo7f3zMVSAR7Mxc9pEsuTzPfj8UBXgFBRfdJUSCgsMxiSw==", 972 | "dev": true, 973 | "dependencies": { 974 | "@parcel/diagnostic": "2.5.0", 975 | "@parcel/hash": "2.5.0", 976 | "@parcel/plugin": "2.5.0", 977 | "@parcel/source-map": "^2.0.0", 978 | "@parcel/utils": "2.5.0", 979 | "globals": "^13.2.0", 980 | "nullthrows": "^1.1.1" 981 | }, 982 | "engines": { 983 | "node": ">= 12.0.0", 984 | "parcel": "^2.5.0" 985 | }, 986 | "funding": { 987 | "type": "opencollective", 988 | "url": "https://opencollective.com/parcel" 989 | } 990 | }, 991 | "node_modules/@parcel/packager-raw": { 992 | "version": "2.5.0", 993 | "resolved": "https://registry.npmjs.org/@parcel/packager-raw/-/packager-raw-2.5.0.tgz", 994 | "integrity": "sha512-aHV0oogeiqxhxS1lsttw15EvG3DDWK3FV7+F+7hoaAy+xg89K56NTp6j43Jtw9iyU1/HnZRGBE2hF3C7N73oKw==", 995 | "dev": true, 996 | "dependencies": { 997 | "@parcel/plugin": "2.5.0" 998 | }, 999 | "engines": { 1000 | "node": ">= 12.0.0", 1001 | "parcel": "^2.5.0" 1002 | }, 1003 | "funding": { 1004 | "type": "opencollective", 1005 | "url": "https://opencollective.com/parcel" 1006 | } 1007 | }, 1008 | "node_modules/@parcel/packager-svg": { 1009 | "version": "2.5.0", 1010 | "resolved": "https://registry.npmjs.org/@parcel/packager-svg/-/packager-svg-2.5.0.tgz", 1011 | "integrity": "sha512-XSMFn30K/kpjcPpQqt88GmPJsNUSVL3RNeigXkIAcLpfO6Tb2eV4iOt4yVCagaDrRJ19alXut0TxjMm5bm41/g==", 1012 | "dev": true, 1013 | "dependencies": { 1014 | "@parcel/plugin": "2.5.0", 1015 | "@parcel/types": "2.5.0", 1016 | "@parcel/utils": "2.5.0", 1017 | "posthtml": "^0.16.4" 1018 | }, 1019 | "engines": { 1020 | "node": ">= 12.0.0", 1021 | "parcel": "^2.5.0" 1022 | }, 1023 | "funding": { 1024 | "type": "opencollective", 1025 | "url": "https://opencollective.com/parcel" 1026 | } 1027 | }, 1028 | "node_modules/@parcel/packager-ts": { 1029 | "version": "2.5.0", 1030 | "resolved": "https://registry.npmjs.org/@parcel/packager-ts/-/packager-ts-2.5.0.tgz", 1031 | "integrity": "sha512-BrH2Gum5EKlWJEJ92dFrH7QTSc7A7LxyElv6c2LPc5sI3z52JDdjQsUMEHqm5Fz25D79Ca/xzVvTWQaYA7XyRA==", 1032 | "dev": true, 1033 | "dependencies": { 1034 | "@parcel/plugin": "2.5.0" 1035 | }, 1036 | "engines": { 1037 | "node": ">= 12.0.0", 1038 | "parcel": "^2.5.0" 1039 | }, 1040 | "funding": { 1041 | "type": "opencollective", 1042 | "url": "https://opencollective.com/parcel" 1043 | } 1044 | }, 1045 | "node_modules/@parcel/plugin": { 1046 | "version": "2.5.0", 1047 | "resolved": "https://registry.npmjs.org/@parcel/plugin/-/plugin-2.5.0.tgz", 1048 | "integrity": "sha512-obtb6/Gql6YFQ86bdv75A2Noabx8679reFZeyfKKf0L7Lppx4DFQetXwM9XVy7Gx6hJ1Ekm3UMuuIyVJk33YHQ==", 1049 | "dev": true, 1050 | "dependencies": { 1051 | "@parcel/types": "2.5.0" 1052 | }, 1053 | "engines": { 1054 | "node": ">= 12.0.0" 1055 | }, 1056 | "funding": { 1057 | "type": "opencollective", 1058 | "url": "https://opencollective.com/parcel" 1059 | } 1060 | }, 1061 | "node_modules/@parcel/reporter-cli": { 1062 | "version": "2.5.0", 1063 | "resolved": "https://registry.npmjs.org/@parcel/reporter-cli/-/reporter-cli-2.5.0.tgz", 1064 | "integrity": "sha512-miJt2YbRJBmYSVeoUWUj8YL85Pwj1CmGQB0/btqhulGLH/Fvkbv6T4sJ4gl4l5xIt9mJQsZ70pOWwa8BId3rWw==", 1065 | "dev": true, 1066 | "dependencies": { 1067 | "@parcel/plugin": "2.5.0", 1068 | "@parcel/types": "2.5.0", 1069 | "@parcel/utils": "2.5.0", 1070 | "chalk": "^4.1.0", 1071 | "term-size": "^2.2.1" 1072 | }, 1073 | "engines": { 1074 | "node": ">= 12.0.0", 1075 | "parcel": "^2.5.0" 1076 | }, 1077 | "funding": { 1078 | "type": "opencollective", 1079 | "url": "https://opencollective.com/parcel" 1080 | } 1081 | }, 1082 | "node_modules/@parcel/reporter-dev-server": { 1083 | "version": "2.5.0", 1084 | "resolved": "https://registry.npmjs.org/@parcel/reporter-dev-server/-/reporter-dev-server-2.5.0.tgz", 1085 | "integrity": "sha512-wvxAiW42AxJ3B8jtvowJcP4/cTV8zY48SfKg61YKYu1yUO+TtyJIjHQzDW2XuT34cIGFY97Gr0i+AVu44RyUuQ==", 1086 | "dev": true, 1087 | "dependencies": { 1088 | "@parcel/plugin": "2.5.0", 1089 | "@parcel/utils": "2.5.0" 1090 | }, 1091 | "engines": { 1092 | "node": ">= 12.0.0", 1093 | "parcel": "^2.5.0" 1094 | }, 1095 | "funding": { 1096 | "type": "opencollective", 1097 | "url": "https://opencollective.com/parcel" 1098 | } 1099 | }, 1100 | "node_modules/@parcel/resolver-default": { 1101 | "version": "2.5.0", 1102 | "resolved": "https://registry.npmjs.org/@parcel/resolver-default/-/resolver-default-2.5.0.tgz", 1103 | "integrity": "sha512-39PkZpVr/+iYS11u+lA84vIsKm/yisltTVmUjlYsDnExiuV1c8OSbSdYZ3JMx+7CYPE0bWbosX2AGilIwIMWpQ==", 1104 | "dev": true, 1105 | "dependencies": { 1106 | "@parcel/node-resolver-core": "2.5.0", 1107 | "@parcel/plugin": "2.5.0" 1108 | }, 1109 | "engines": { 1110 | "node": ">= 12.0.0", 1111 | "parcel": "^2.5.0" 1112 | }, 1113 | "funding": { 1114 | "type": "opencollective", 1115 | "url": "https://opencollective.com/parcel" 1116 | } 1117 | }, 1118 | "node_modules/@parcel/runtime-browser-hmr": { 1119 | "version": "2.5.0", 1120 | "resolved": "https://registry.npmjs.org/@parcel/runtime-browser-hmr/-/runtime-browser-hmr-2.5.0.tgz", 1121 | "integrity": "sha512-oPAo8Zf06gXCpt41nyvK7kv2HH1RrHAGgOqttyjStwAFlm5MZKs7BgtJzO58LfJN8g3sMY0cNdG17fB/4f8q6Q==", 1122 | "dev": true, 1123 | "dependencies": { 1124 | "@parcel/plugin": "2.5.0", 1125 | "@parcel/utils": "2.5.0" 1126 | }, 1127 | "engines": { 1128 | "node": ">= 12.0.0", 1129 | "parcel": "^2.5.0" 1130 | }, 1131 | "funding": { 1132 | "type": "opencollective", 1133 | "url": "https://opencollective.com/parcel" 1134 | } 1135 | }, 1136 | "node_modules/@parcel/runtime-js": { 1137 | "version": "2.5.0", 1138 | "resolved": "https://registry.npmjs.org/@parcel/runtime-js/-/runtime-js-2.5.0.tgz", 1139 | "integrity": "sha512-gPC2PbNAiooULP71wF5twe4raekuXsR1Hw/ahITDoqsZdXHzG3CkoCjYL3CkmBGiKQgMMocCyN1E2oBzAH8Kyw==", 1140 | "dev": true, 1141 | "dependencies": { 1142 | "@parcel/plugin": "2.5.0", 1143 | "@parcel/utils": "2.5.0", 1144 | "nullthrows": "^1.1.1" 1145 | }, 1146 | "engines": { 1147 | "node": ">= 12.0.0", 1148 | "parcel": "^2.5.0" 1149 | }, 1150 | "funding": { 1151 | "type": "opencollective", 1152 | "url": "https://opencollective.com/parcel" 1153 | } 1154 | }, 1155 | "node_modules/@parcel/runtime-react-refresh": { 1156 | "version": "2.5.0", 1157 | "resolved": "https://registry.npmjs.org/@parcel/runtime-react-refresh/-/runtime-react-refresh-2.5.0.tgz", 1158 | "integrity": "sha512-+8RuDKFdFYIQTrXG4MRhG9XqkkYEHn0zxKyOJ/IkDDfSEhY0na+EyhrneFUwIvDX63gLPkxceXAg0gwBqXPK/Q==", 1159 | "dev": true, 1160 | "dependencies": { 1161 | "@parcel/plugin": "2.5.0", 1162 | "@parcel/utils": "2.5.0", 1163 | "react-refresh": "^0.9.0" 1164 | }, 1165 | "engines": { 1166 | "node": ">= 12.0.0", 1167 | "parcel": "^2.5.0" 1168 | }, 1169 | "funding": { 1170 | "type": "opencollective", 1171 | "url": "https://opencollective.com/parcel" 1172 | } 1173 | }, 1174 | "node_modules/@parcel/runtime-service-worker": { 1175 | "version": "2.5.0", 1176 | "resolved": "https://registry.npmjs.org/@parcel/runtime-service-worker/-/runtime-service-worker-2.5.0.tgz", 1177 | "integrity": "sha512-STuDlU0fPXeWpAmbayY7o04F0eHy6FTOFeT5KQ0PTxtdEa3Ey8QInP/NVE52Yv0aVQtesWukGrNEFCERlkbFRw==", 1178 | "dev": true, 1179 | "dependencies": { 1180 | "@parcel/plugin": "2.5.0", 1181 | "@parcel/utils": "2.5.0", 1182 | "nullthrows": "^1.1.1" 1183 | }, 1184 | "engines": { 1185 | "node": ">= 12.0.0", 1186 | "parcel": "^2.5.0" 1187 | }, 1188 | "funding": { 1189 | "type": "opencollective", 1190 | "url": "https://opencollective.com/parcel" 1191 | } 1192 | }, 1193 | "node_modules/@parcel/source-map": { 1194 | "version": "2.0.5", 1195 | "resolved": "https://registry.npmjs.org/@parcel/source-map/-/source-map-2.0.5.tgz", 1196 | "integrity": "sha512-DRVlCFKLpqBSIbMxUoVlHgfiv12HTW/U7nnhzw52YgzDVXUX9OA41dXS1PU0pJ1si+D1k8msATUC+AoldN43mg==", 1197 | "dev": true, 1198 | "dependencies": { 1199 | "detect-libc": "^1.0.3" 1200 | }, 1201 | "engines": { 1202 | "node": "^12.18.3 || >=14" 1203 | } 1204 | }, 1205 | "node_modules/@parcel/transformer-babel": { 1206 | "version": "2.5.0", 1207 | "resolved": "https://registry.npmjs.org/@parcel/transformer-babel/-/transformer-babel-2.5.0.tgz", 1208 | "integrity": "sha512-EFb866C9jCoBHIcebWF7goAcYj1wkObx0GDxshlazFtvym1RM27xSWWjRYyqb5+HNOxB3voaNvQOVjcD+DXjCA==", 1209 | "dev": true, 1210 | "dependencies": { 1211 | "@parcel/diagnostic": "2.5.0", 1212 | "@parcel/plugin": "2.5.0", 1213 | "@parcel/source-map": "^2.0.0", 1214 | "@parcel/utils": "2.5.0", 1215 | "browserslist": "^4.6.6", 1216 | "json5": "^2.2.0", 1217 | "nullthrows": "^1.1.1", 1218 | "semver": "^5.7.0" 1219 | }, 1220 | "engines": { 1221 | "node": ">= 12.0.0", 1222 | "parcel": "^2.5.0" 1223 | }, 1224 | "funding": { 1225 | "type": "opencollective", 1226 | "url": "https://opencollective.com/parcel" 1227 | } 1228 | }, 1229 | "node_modules/@parcel/transformer-babel/node_modules/semver": { 1230 | "version": "5.7.1", 1231 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1232 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1233 | "dev": true, 1234 | "bin": { 1235 | "semver": "bin/semver" 1236 | } 1237 | }, 1238 | "node_modules/@parcel/transformer-css": { 1239 | "version": "2.5.0", 1240 | "resolved": "https://registry.npmjs.org/@parcel/transformer-css/-/transformer-css-2.5.0.tgz", 1241 | "integrity": "sha512-p8FOvKWWSbS6H8PbD9a0KZqyaKNpSD2BUTzSRYnNj3TBUv7/ZXaP6Om295XTQ/MPht1o7XTQzvfpF/7yEhr02Q==", 1242 | "dev": true, 1243 | "dependencies": { 1244 | "@parcel/css": "^1.8.1", 1245 | "@parcel/diagnostic": "2.5.0", 1246 | "@parcel/plugin": "2.5.0", 1247 | "@parcel/source-map": "^2.0.0", 1248 | "@parcel/utils": "2.5.0", 1249 | "browserslist": "^4.6.6", 1250 | "nullthrows": "^1.1.1" 1251 | }, 1252 | "engines": { 1253 | "node": ">= 12.0.0", 1254 | "parcel": "^2.5.0" 1255 | }, 1256 | "funding": { 1257 | "type": "opencollective", 1258 | "url": "https://opencollective.com/parcel" 1259 | } 1260 | }, 1261 | "node_modules/@parcel/transformer-html": { 1262 | "version": "2.5.0", 1263 | "resolved": "https://registry.npmjs.org/@parcel/transformer-html/-/transformer-html-2.5.0.tgz", 1264 | "integrity": "sha512-iEjNyAF0wQmY3DMw7FS+UzoOMng76UsSngh+WWA1E5lv5XyqrP8Mk2QLTJp1nWetUhSLhZr58LGmPYBTB4l9ZQ==", 1265 | "dev": true, 1266 | "dependencies": { 1267 | "@parcel/diagnostic": "2.5.0", 1268 | "@parcel/hash": "2.5.0", 1269 | "@parcel/plugin": "2.5.0", 1270 | "nullthrows": "^1.1.1", 1271 | "posthtml": "^0.16.5", 1272 | "posthtml-parser": "^0.10.1", 1273 | "posthtml-render": "^3.0.0", 1274 | "semver": "^5.7.1" 1275 | }, 1276 | "engines": { 1277 | "node": ">= 12.0.0", 1278 | "parcel": "^2.5.0" 1279 | }, 1280 | "funding": { 1281 | "type": "opencollective", 1282 | "url": "https://opencollective.com/parcel" 1283 | } 1284 | }, 1285 | "node_modules/@parcel/transformer-html/node_modules/posthtml-parser": { 1286 | "version": "0.10.2", 1287 | "resolved": "https://registry.npmjs.org/posthtml-parser/-/posthtml-parser-0.10.2.tgz", 1288 | "integrity": "sha512-PId6zZ/2lyJi9LiKfe+i2xv57oEjJgWbsHGGANwos5AvdQp98i6AtamAl8gzSVFGfQ43Glb5D614cvZf012VKg==", 1289 | "dev": true, 1290 | "dependencies": { 1291 | "htmlparser2": "^7.1.1" 1292 | }, 1293 | "engines": { 1294 | "node": ">=12" 1295 | } 1296 | }, 1297 | "node_modules/@parcel/transformer-html/node_modules/semver": { 1298 | "version": "5.7.1", 1299 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1300 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1301 | "dev": true, 1302 | "bin": { 1303 | "semver": "bin/semver" 1304 | } 1305 | }, 1306 | "node_modules/@parcel/transformer-image": { 1307 | "version": "2.5.0", 1308 | "resolved": "https://registry.npmjs.org/@parcel/transformer-image/-/transformer-image-2.5.0.tgz", 1309 | "integrity": "sha512-vVEXTHZl8m/9yopgK0dWHLOQX2zOnghq6pZnWdWVG6fsvXZln7kP1YN5iwWDoADQYkiKzP+Ymn6UwP9pZpHFzA==", 1310 | "dev": true, 1311 | "dependencies": { 1312 | "@parcel/plugin": "2.5.0", 1313 | "@parcel/workers": "2.5.0", 1314 | "nullthrows": "^1.1.1" 1315 | }, 1316 | "engines": { 1317 | "node": ">= 12.0.0", 1318 | "parcel": "^2.5.0" 1319 | }, 1320 | "peerDependencies": { 1321 | "@parcel/core": "^2.5.0" 1322 | } 1323 | }, 1324 | "node_modules/@parcel/transformer-inline-string": { 1325 | "version": "2.5.0", 1326 | "resolved": "https://registry.npmjs.org/@parcel/transformer-inline-string/-/transformer-inline-string-2.5.0.tgz", 1327 | "integrity": "sha512-nBvzbAEIQ8qTSNEbFrxm+9XyOgGTaLOm1+dmBln75+OML4yn9TRUa8w2VxPKWyImPYAwRRZ2CZXAnGKfhwr+LA==", 1328 | "dev": true, 1329 | "dependencies": { 1330 | "@parcel/plugin": "2.5.0" 1331 | }, 1332 | "engines": { 1333 | "node": ">= 12.0.0", 1334 | "parcel": "^2.5.0" 1335 | }, 1336 | "funding": { 1337 | "type": "opencollective", 1338 | "url": "https://opencollective.com/parcel" 1339 | } 1340 | }, 1341 | "node_modules/@parcel/transformer-js": { 1342 | "version": "2.5.0", 1343 | "resolved": "https://registry.npmjs.org/@parcel/transformer-js/-/transformer-js-2.5.0.tgz", 1344 | "integrity": "sha512-Cp8Ic+Au3OcskCRZszmo47z3bqcZ7rfPv2xZYXpXY2TzEc3IV0bKje57bZektoY8LW9LkYM9iBO/WhkVoT6LIg==", 1345 | "dev": true, 1346 | "dependencies": { 1347 | "@parcel/diagnostic": "2.5.0", 1348 | "@parcel/plugin": "2.5.0", 1349 | "@parcel/source-map": "^2.0.0", 1350 | "@parcel/utils": "2.5.0", 1351 | "@parcel/workers": "2.5.0", 1352 | "@swc/helpers": "^0.3.6", 1353 | "browserslist": "^4.6.6", 1354 | "detect-libc": "^1.0.3", 1355 | "nullthrows": "^1.1.1", 1356 | "regenerator-runtime": "^0.13.7", 1357 | "semver": "^5.7.1" 1358 | }, 1359 | "engines": { 1360 | "node": ">= 12.0.0", 1361 | "parcel": "^2.5.0" 1362 | }, 1363 | "funding": { 1364 | "type": "opencollective", 1365 | "url": "https://opencollective.com/parcel" 1366 | }, 1367 | "peerDependencies": { 1368 | "@parcel/core": "^2.5.0" 1369 | } 1370 | }, 1371 | "node_modules/@parcel/transformer-js/node_modules/semver": { 1372 | "version": "5.7.1", 1373 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1374 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1375 | "dev": true, 1376 | "bin": { 1377 | "semver": "bin/semver" 1378 | } 1379 | }, 1380 | "node_modules/@parcel/transformer-json": { 1381 | "version": "2.5.0", 1382 | "resolved": "https://registry.npmjs.org/@parcel/transformer-json/-/transformer-json-2.5.0.tgz", 1383 | "integrity": "sha512-661sByA7TkR6Lmxt+hqV4h2SAt+7lgc58DzmUYArpEl1fQnMuQuaB0kQeHzi6fDD2+2G6o7EC+DuwBZKa479TA==", 1384 | "dev": true, 1385 | "dependencies": { 1386 | "@parcel/plugin": "2.5.0", 1387 | "json5": "^2.2.0" 1388 | }, 1389 | "engines": { 1390 | "node": ">= 12.0.0", 1391 | "parcel": "^2.5.0" 1392 | }, 1393 | "funding": { 1394 | "type": "opencollective", 1395 | "url": "https://opencollective.com/parcel" 1396 | } 1397 | }, 1398 | "node_modules/@parcel/transformer-postcss": { 1399 | "version": "2.5.0", 1400 | "resolved": "https://registry.npmjs.org/@parcel/transformer-postcss/-/transformer-postcss-2.5.0.tgz", 1401 | "integrity": "sha512-IPNlWElekdQHMTBqhdwJNBCQomuYyo7xgNBdnTrt9VJ+R5ihy6n7ZJSWIAJXAH9VZxETTtunfrzRtgkmtjTeZQ==", 1402 | "dev": true, 1403 | "dependencies": { 1404 | "@parcel/diagnostic": "2.5.0", 1405 | "@parcel/hash": "2.5.0", 1406 | "@parcel/plugin": "2.5.0", 1407 | "@parcel/utils": "2.5.0", 1408 | "clone": "^2.1.1", 1409 | "nullthrows": "^1.1.1", 1410 | "postcss-value-parser": "^4.2.0", 1411 | "semver": "^5.7.1" 1412 | }, 1413 | "engines": { 1414 | "node": ">= 12.0.0", 1415 | "parcel": "^2.5.0" 1416 | }, 1417 | "funding": { 1418 | "type": "opencollective", 1419 | "url": "https://opencollective.com/parcel" 1420 | } 1421 | }, 1422 | "node_modules/@parcel/transformer-postcss/node_modules/semver": { 1423 | "version": "5.7.1", 1424 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1425 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1426 | "dev": true, 1427 | "bin": { 1428 | "semver": "bin/semver" 1429 | } 1430 | }, 1431 | "node_modules/@parcel/transformer-posthtml": { 1432 | "version": "2.5.0", 1433 | "resolved": "https://registry.npmjs.org/@parcel/transformer-posthtml/-/transformer-posthtml-2.5.0.tgz", 1434 | "integrity": "sha512-AZxg1XD8OXOS4bEGEmBBR+X9T9qoFdVsbVUg498zzejYSka1ZQHF7TgLI/+pUnE+ZVYNIp7/G0xXqsRVKMKmdQ==", 1435 | "dev": true, 1436 | "dependencies": { 1437 | "@parcel/plugin": "2.5.0", 1438 | "@parcel/utils": "2.5.0", 1439 | "nullthrows": "^1.1.1", 1440 | "posthtml": "^0.16.5", 1441 | "posthtml-parser": "^0.10.1", 1442 | "posthtml-render": "^3.0.0", 1443 | "semver": "^5.7.1" 1444 | }, 1445 | "engines": { 1446 | "node": ">= 12.0.0", 1447 | "parcel": "^2.5.0" 1448 | }, 1449 | "funding": { 1450 | "type": "opencollective", 1451 | "url": "https://opencollective.com/parcel" 1452 | } 1453 | }, 1454 | "node_modules/@parcel/transformer-posthtml/node_modules/posthtml-parser": { 1455 | "version": "0.10.2", 1456 | "resolved": "https://registry.npmjs.org/posthtml-parser/-/posthtml-parser-0.10.2.tgz", 1457 | "integrity": "sha512-PId6zZ/2lyJi9LiKfe+i2xv57oEjJgWbsHGGANwos5AvdQp98i6AtamAl8gzSVFGfQ43Glb5D614cvZf012VKg==", 1458 | "dev": true, 1459 | "dependencies": { 1460 | "htmlparser2": "^7.1.1" 1461 | }, 1462 | "engines": { 1463 | "node": ">=12" 1464 | } 1465 | }, 1466 | "node_modules/@parcel/transformer-posthtml/node_modules/semver": { 1467 | "version": "5.7.1", 1468 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1469 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1470 | "dev": true, 1471 | "bin": { 1472 | "semver": "bin/semver" 1473 | } 1474 | }, 1475 | "node_modules/@parcel/transformer-raw": { 1476 | "version": "2.5.0", 1477 | "resolved": "https://registry.npmjs.org/@parcel/transformer-raw/-/transformer-raw-2.5.0.tgz", 1478 | "integrity": "sha512-I3zjE1u9+Wj90Qqs1V2FTm6iC6SAyOVUthwVZkZey+qbQG/ok682Ez2XjLu7MyQCo9BJNwF/nfOa1hHr3MaJEQ==", 1479 | "dev": true, 1480 | "dependencies": { 1481 | "@parcel/plugin": "2.5.0" 1482 | }, 1483 | "engines": { 1484 | "node": ">= 12.0.0", 1485 | "parcel": "^2.5.0" 1486 | }, 1487 | "funding": { 1488 | "type": "opencollective", 1489 | "url": "https://opencollective.com/parcel" 1490 | } 1491 | }, 1492 | "node_modules/@parcel/transformer-react-refresh-wrap": { 1493 | "version": "2.5.0", 1494 | "resolved": "https://registry.npmjs.org/@parcel/transformer-react-refresh-wrap/-/transformer-react-refresh-wrap-2.5.0.tgz", 1495 | "integrity": "sha512-VPqVBxhTN4OQwcjsdyxrv+smjAm4s6dbSWAplgPwdOITMv+a0tjhhJU37WnRC+xxTrbEqRcOt96JvGOkPb8i7g==", 1496 | "dev": true, 1497 | "dependencies": { 1498 | "@parcel/plugin": "2.5.0", 1499 | "@parcel/utils": "2.5.0", 1500 | "react-refresh": "^0.9.0" 1501 | }, 1502 | "engines": { 1503 | "node": ">= 12.0.0", 1504 | "parcel": "^2.5.0" 1505 | }, 1506 | "funding": { 1507 | "type": "opencollective", 1508 | "url": "https://opencollective.com/parcel" 1509 | } 1510 | }, 1511 | "node_modules/@parcel/transformer-svg": { 1512 | "version": "2.5.0", 1513 | "resolved": "https://registry.npmjs.org/@parcel/transformer-svg/-/transformer-svg-2.5.0.tgz", 1514 | "integrity": "sha512-zCGJcrCpICFe0Q/dgjQZfW7sYFkbJEC7NGT4zEJnMo8Cm/kq8Qh6+2ApX6c+vv5Q0WZn5Ic+N0OvxIMkvgdC/w==", 1515 | "dev": true, 1516 | "dependencies": { 1517 | "@parcel/diagnostic": "2.5.0", 1518 | "@parcel/hash": "2.5.0", 1519 | "@parcel/plugin": "2.5.0", 1520 | "nullthrows": "^1.1.1", 1521 | "posthtml": "^0.16.5", 1522 | "posthtml-parser": "^0.10.1", 1523 | "posthtml-render": "^3.0.0", 1524 | "semver": "^5.7.1" 1525 | }, 1526 | "engines": { 1527 | "node": ">= 12.0.0", 1528 | "parcel": "^2.5.0" 1529 | }, 1530 | "funding": { 1531 | "type": "opencollective", 1532 | "url": "https://opencollective.com/parcel" 1533 | } 1534 | }, 1535 | "node_modules/@parcel/transformer-svg/node_modules/posthtml-parser": { 1536 | "version": "0.10.2", 1537 | "resolved": "https://registry.npmjs.org/posthtml-parser/-/posthtml-parser-0.10.2.tgz", 1538 | "integrity": "sha512-PId6zZ/2lyJi9LiKfe+i2xv57oEjJgWbsHGGANwos5AvdQp98i6AtamAl8gzSVFGfQ43Glb5D614cvZf012VKg==", 1539 | "dev": true, 1540 | "dependencies": { 1541 | "htmlparser2": "^7.1.1" 1542 | }, 1543 | "engines": { 1544 | "node": ">=12" 1545 | } 1546 | }, 1547 | "node_modules/@parcel/transformer-svg/node_modules/semver": { 1548 | "version": "5.7.1", 1549 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 1550 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", 1551 | "dev": true, 1552 | "bin": { 1553 | "semver": "bin/semver" 1554 | } 1555 | }, 1556 | "node_modules/@parcel/transformer-typescript-types": { 1557 | "version": "2.5.0", 1558 | "resolved": "https://registry.npmjs.org/@parcel/transformer-typescript-types/-/transformer-typescript-types-2.5.0.tgz", 1559 | "integrity": "sha512-O+v+vEvgQDj5U1O8C12nYeU9kYOdYaznobWgE21WYSPEV2JD9ppaJVTDoNTI5Lx58gmjc1hndY169o6N6RaV6A==", 1560 | "dev": true, 1561 | "dependencies": { 1562 | "@parcel/diagnostic": "2.5.0", 1563 | "@parcel/plugin": "2.5.0", 1564 | "@parcel/source-map": "^2.0.0", 1565 | "@parcel/ts-utils": "2.5.0", 1566 | "nullthrows": "^1.1.1" 1567 | }, 1568 | "engines": { 1569 | "node": ">= 12.0.0", 1570 | "parcel": "^2.5.0" 1571 | }, 1572 | "funding": { 1573 | "type": "opencollective", 1574 | "url": "https://opencollective.com/parcel" 1575 | }, 1576 | "peerDependencies": { 1577 | "typescript": ">=3.0.0" 1578 | } 1579 | }, 1580 | "node_modules/@parcel/ts-utils": { 1581 | "version": "2.5.0", 1582 | "resolved": "https://registry.npmjs.org/@parcel/ts-utils/-/ts-utils-2.5.0.tgz", 1583 | "integrity": "sha512-YITx84Olg27PDxvJlXzzPVgqTtW3tEqQFh+wE2g7+Mwk4Q8vd/jL+mjDBF/5LEnGCk2WvjkcuBK/QOv7Y+YDsg==", 1584 | "dev": true, 1585 | "dependencies": { 1586 | "nullthrows": "^1.1.1" 1587 | }, 1588 | "engines": { 1589 | "node": ">= 12.0.0" 1590 | }, 1591 | "funding": { 1592 | "type": "opencollective", 1593 | "url": "https://opencollective.com/parcel" 1594 | }, 1595 | "peerDependencies": { 1596 | "typescript": ">=3.0.0" 1597 | } 1598 | }, 1599 | "node_modules/@parcel/types": { 1600 | "version": "2.5.0", 1601 | "resolved": "https://registry.npmjs.org/@parcel/types/-/types-2.5.0.tgz", 1602 | "integrity": "sha512-bA0fhG6aXSGYEVo5Dt96x6lseUQHeVZVzgmiRdZsvb614Gvx22ItfaKhPmAVbM9vzbObZDHl9l9G2Ovw8Xve4g==", 1603 | "dev": true, 1604 | "dependencies": { 1605 | "@parcel/cache": "2.5.0", 1606 | "@parcel/diagnostic": "2.5.0", 1607 | "@parcel/fs": "2.5.0", 1608 | "@parcel/package-manager": "2.5.0", 1609 | "@parcel/source-map": "^2.0.0", 1610 | "@parcel/workers": "2.5.0", 1611 | "utility-types": "^3.10.0" 1612 | } 1613 | }, 1614 | "node_modules/@parcel/utils": { 1615 | "version": "2.5.0", 1616 | "resolved": "https://registry.npmjs.org/@parcel/utils/-/utils-2.5.0.tgz", 1617 | "integrity": "sha512-kaLGXtQuOOH55KZqXdYDvczhh3mk2eeTVqrrXuuihGjbLKYFlUW2tFDm+5r2s9nCPwTQxOO43ZEOCKSnia+e4w==", 1618 | "dev": true, 1619 | "dependencies": { 1620 | "@parcel/codeframe": "2.5.0", 1621 | "@parcel/diagnostic": "2.5.0", 1622 | "@parcel/hash": "2.5.0", 1623 | "@parcel/logger": "2.5.0", 1624 | "@parcel/markdown-ansi": "2.5.0", 1625 | "@parcel/source-map": "^2.0.0", 1626 | "chalk": "^4.1.0" 1627 | }, 1628 | "engines": { 1629 | "node": ">= 12.0.0" 1630 | }, 1631 | "funding": { 1632 | "type": "opencollective", 1633 | "url": "https://opencollective.com/parcel" 1634 | } 1635 | }, 1636 | "node_modules/@parcel/watcher": { 1637 | "version": "2.0.5", 1638 | "resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-2.0.5.tgz", 1639 | "integrity": "sha512-x0hUbjv891omnkcHD7ZOhiyyUqUUR6MNjq89JhEI3BxppeKWAm6NPQsqqRrAkCJBogdT/o/My21sXtTI9rJIsw==", 1640 | "dev": true, 1641 | "hasInstallScript": true, 1642 | "dependencies": { 1643 | "node-addon-api": "^3.2.1", 1644 | "node-gyp-build": "^4.3.0" 1645 | }, 1646 | "engines": { 1647 | "node": ">= 10.0.0" 1648 | }, 1649 | "funding": { 1650 | "type": "opencollective", 1651 | "url": "https://opencollective.com/parcel" 1652 | } 1653 | }, 1654 | "node_modules/@parcel/workers": { 1655 | "version": "2.5.0", 1656 | "resolved": "https://registry.npmjs.org/@parcel/workers/-/workers-2.5.0.tgz", 1657 | "integrity": "sha512-/Ow5OKJWs+9OzV3Jy4J++VnbNx0j3ls/M1CGVBLiBWyCada9DMtquYoBQ4Sk6Uam50BKkIFYetGOeXPNQyyMjg==", 1658 | "dev": true, 1659 | "dependencies": { 1660 | "@parcel/diagnostic": "2.5.0", 1661 | "@parcel/logger": "2.5.0", 1662 | "@parcel/types": "2.5.0", 1663 | "@parcel/utils": "2.5.0", 1664 | "chrome-trace-event": "^1.0.2", 1665 | "nullthrows": "^1.1.1" 1666 | }, 1667 | "engines": { 1668 | "node": ">= 12.0.0" 1669 | }, 1670 | "funding": { 1671 | "type": "opencollective", 1672 | "url": "https://opencollective.com/parcel" 1673 | }, 1674 | "peerDependencies": { 1675 | "@parcel/core": "^2.5.0" 1676 | } 1677 | }, 1678 | "node_modules/@sindresorhus/is": { 1679 | "version": "0.14.0", 1680 | "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", 1681 | "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", 1682 | "dev": true, 1683 | "engines": { 1684 | "node": ">=6" 1685 | } 1686 | }, 1687 | "node_modules/@swc/helpers": { 1688 | "version": "0.3.13", 1689 | "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.3.13.tgz", 1690 | "integrity": "sha512-A1wswJhnqaLRn8uYVQ8YiNTtY5i/JIPmV08EXXjjTresIkUVUEUaFv/wXVhGXfRNYMvHPkuoMR1Nb6NgpxGjNg==", 1691 | "dev": true, 1692 | "dependencies": { 1693 | "tslib": "^2.4.0" 1694 | } 1695 | }, 1696 | "node_modules/@szmarczak/http-timer": { 1697 | "version": "1.1.2", 1698 | "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", 1699 | "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", 1700 | "dev": true, 1701 | "dependencies": { 1702 | "defer-to-connect": "^1.0.1" 1703 | }, 1704 | "engines": { 1705 | "node": ">=6" 1706 | } 1707 | }, 1708 | "node_modules/@trysound/sax": { 1709 | "version": "0.2.0", 1710 | "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz", 1711 | "integrity": "sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==", 1712 | "dev": true, 1713 | "engines": { 1714 | "node": ">=10.13.0" 1715 | } 1716 | }, 1717 | "node_modules/@types/node": { 1718 | "version": "14.18.12", 1719 | "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.12.tgz", 1720 | "integrity": "sha512-q4jlIR71hUpWTnGhXWcakgkZeHa3CCjcQcnuzU8M891BAWA2jHiziiWEPEkdS5pFsz7H9HJiy8BrK7tBRNrY7A==", 1721 | "dev": true 1722 | }, 1723 | "node_modules/@types/parse-json": { 1724 | "version": "4.0.0", 1725 | "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz", 1726 | "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==", 1727 | "dev": true 1728 | }, 1729 | "node_modules/@types/sortablejs": { 1730 | "version": "1.13.0", 1731 | "resolved": "https://registry.npmjs.org/@types/sortablejs/-/sortablejs-1.13.0.tgz", 1732 | "integrity": "sha512-C3064MH72iEfeGCYEGCt7FCxXoAXaMPG0QPnstcxvPmbl54erpISu06d++FY37Smja64iWy5L8wOyHHBghWbJQ==", 1733 | "dev": true 1734 | }, 1735 | "node_modules/abortcontroller-polyfill": { 1736 | "version": "1.7.3", 1737 | "resolved": "https://registry.npmjs.org/abortcontroller-polyfill/-/abortcontroller-polyfill-1.7.3.tgz", 1738 | "integrity": "sha512-zetDJxd89y3X99Kvo4qFx8GKlt6GsvN3UcRZHwU6iFA/0KiOmhkTVhe8oRoTBiTVPZu09x3vCra47+w8Yz1+2Q==", 1739 | "dev": true 1740 | }, 1741 | "node_modules/acorn": { 1742 | "version": "8.7.1", 1743 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.1.tgz", 1744 | "integrity": "sha512-Xx54uLJQZ19lKygFXOWsscKUbsBZW0CPykPhVQdhIeIwrbPmJzqeASDInc8nKBnp/JT6igTs82qPXz069H8I/A==", 1745 | "dev": true, 1746 | "bin": { 1747 | "acorn": "bin/acorn" 1748 | }, 1749 | "engines": { 1750 | "node": ">=0.4.0" 1751 | } 1752 | }, 1753 | "node_modules/ansi-styles": { 1754 | "version": "4.3.0", 1755 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1756 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1757 | "dev": true, 1758 | "dependencies": { 1759 | "color-convert": "^2.0.1" 1760 | }, 1761 | "engines": { 1762 | "node": ">=8" 1763 | }, 1764 | "funding": { 1765 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 1766 | } 1767 | }, 1768 | "node_modules/base-x": { 1769 | "version": "3.0.9", 1770 | "resolved": "https://registry.npmjs.org/base-x/-/base-x-3.0.9.tgz", 1771 | "integrity": "sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==", 1772 | "dev": true, 1773 | "dependencies": { 1774 | "safe-buffer": "^5.0.1" 1775 | } 1776 | }, 1777 | "node_modules/boolbase": { 1778 | "version": "1.0.0", 1779 | "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", 1780 | "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==", 1781 | "dev": true 1782 | }, 1783 | "node_modules/boolean": { 1784 | "version": "3.2.0", 1785 | "resolved": "https://registry.npmjs.org/boolean/-/boolean-3.2.0.tgz", 1786 | "integrity": "sha512-d0II/GO9uf9lfUHH2BQsjxzRJZBdsjgsBiW4BvhWk/3qoKwQFjIDVN19PfX8F2D/r9PCMTtLWjYVCFrpeYUzsw==", 1787 | "dev": true, 1788 | "optional": true 1789 | }, 1790 | "node_modules/browserslist": { 1791 | "version": "4.20.3", 1792 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.20.3.tgz", 1793 | "integrity": "sha512-NBhymBQl1zM0Y5dQT/O+xiLP9/rzOIQdKM/eMJBAq7yBgaB6krIYLGejrwVYnSHZdqjscB1SPuAjHwxjvN6Wdg==", 1794 | "dev": true, 1795 | "funding": [ 1796 | { 1797 | "type": "opencollective", 1798 | "url": "https://opencollective.com/browserslist" 1799 | }, 1800 | { 1801 | "type": "tidelift", 1802 | "url": "https://tidelift.com/funding/github/npm/browserslist" 1803 | } 1804 | ], 1805 | "dependencies": { 1806 | "caniuse-lite": "^1.0.30001332", 1807 | "electron-to-chromium": "^1.4.118", 1808 | "escalade": "^3.1.1", 1809 | "node-releases": "^2.0.3", 1810 | "picocolors": "^1.0.0" 1811 | }, 1812 | "bin": { 1813 | "browserslist": "cli.js" 1814 | }, 1815 | "engines": { 1816 | "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" 1817 | } 1818 | }, 1819 | "node_modules/buffer-crc32": { 1820 | "version": "0.2.13", 1821 | "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", 1822 | "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=", 1823 | "dev": true, 1824 | "engines": { 1825 | "node": "*" 1826 | } 1827 | }, 1828 | "node_modules/buffer-from": { 1829 | "version": "1.1.1", 1830 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 1831 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", 1832 | "dev": true 1833 | }, 1834 | "node_modules/cacheable-request": { 1835 | "version": "6.1.0", 1836 | "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", 1837 | "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", 1838 | "dev": true, 1839 | "dependencies": { 1840 | "clone-response": "^1.0.2", 1841 | "get-stream": "^5.1.0", 1842 | "http-cache-semantics": "^4.0.0", 1843 | "keyv": "^3.0.0", 1844 | "lowercase-keys": "^2.0.0", 1845 | "normalize-url": "^4.1.0", 1846 | "responselike": "^1.0.2" 1847 | }, 1848 | "engines": { 1849 | "node": ">=8" 1850 | } 1851 | }, 1852 | "node_modules/cacheable-request/node_modules/get-stream": { 1853 | "version": "5.2.0", 1854 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", 1855 | "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", 1856 | "dev": true, 1857 | "dependencies": { 1858 | "pump": "^3.0.0" 1859 | }, 1860 | "engines": { 1861 | "node": ">=8" 1862 | }, 1863 | "funding": { 1864 | "url": "https://github.com/sponsors/sindresorhus" 1865 | } 1866 | }, 1867 | "node_modules/cacheable-request/node_modules/lowercase-keys": { 1868 | "version": "2.0.0", 1869 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", 1870 | "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", 1871 | "dev": true, 1872 | "engines": { 1873 | "node": ">=8" 1874 | } 1875 | }, 1876 | "node_modules/callsites": { 1877 | "version": "3.1.0", 1878 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 1879 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 1880 | "dev": true, 1881 | "engines": { 1882 | "node": ">=6" 1883 | } 1884 | }, 1885 | "node_modules/caniuse-lite": { 1886 | "version": "1.0.30001342", 1887 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001342.tgz", 1888 | "integrity": "sha512-bn6sOCu7L7jcbBbyNhLg0qzXdJ/PMbybZTH/BA6Roet9wxYRm6Tr9D0s0uhLkOZ6MSG+QU6txUgdpr3MXIVqjA==", 1889 | "dev": true, 1890 | "funding": [ 1891 | { 1892 | "type": "opencollective", 1893 | "url": "https://opencollective.com/browserslist" 1894 | }, 1895 | { 1896 | "type": "tidelift", 1897 | "url": "https://tidelift.com/funding/github/npm/caniuse-lite" 1898 | } 1899 | ] 1900 | }, 1901 | "node_modules/chalk": { 1902 | "version": "4.1.2", 1903 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 1904 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 1905 | "dev": true, 1906 | "dependencies": { 1907 | "ansi-styles": "^4.1.0", 1908 | "supports-color": "^7.1.0" 1909 | }, 1910 | "engines": { 1911 | "node": ">=10" 1912 | }, 1913 | "funding": { 1914 | "url": "https://github.com/chalk/chalk?sponsor=1" 1915 | } 1916 | }, 1917 | "node_modules/chrome-trace-event": { 1918 | "version": "1.0.3", 1919 | "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", 1920 | "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", 1921 | "dev": true, 1922 | "engines": { 1923 | "node": ">=6.0" 1924 | } 1925 | }, 1926 | "node_modules/clone": { 1927 | "version": "2.1.2", 1928 | "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", 1929 | "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", 1930 | "dev": true, 1931 | "engines": { 1932 | "node": ">=0.8" 1933 | } 1934 | }, 1935 | "node_modules/clone-response": { 1936 | "version": "1.0.2", 1937 | "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", 1938 | "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", 1939 | "dev": true, 1940 | "dependencies": { 1941 | "mimic-response": "^1.0.0" 1942 | } 1943 | }, 1944 | "node_modules/color-convert": { 1945 | "version": "2.0.1", 1946 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1947 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1948 | "dev": true, 1949 | "dependencies": { 1950 | "color-name": "~1.1.4" 1951 | }, 1952 | "engines": { 1953 | "node": ">=7.0.0" 1954 | } 1955 | }, 1956 | "node_modules/color-name": { 1957 | "version": "1.1.4", 1958 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1959 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1960 | "dev": true 1961 | }, 1962 | "node_modules/commander": { 1963 | "version": "7.2.0", 1964 | "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", 1965 | "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", 1966 | "dev": true, 1967 | "engines": { 1968 | "node": ">= 10" 1969 | } 1970 | }, 1971 | "node_modules/concat-stream": { 1972 | "version": "1.6.2", 1973 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", 1974 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", 1975 | "dev": true, 1976 | "engines": [ 1977 | "node >= 0.8" 1978 | ], 1979 | "dependencies": { 1980 | "buffer-from": "^1.0.0", 1981 | "inherits": "^2.0.3", 1982 | "readable-stream": "^2.2.2", 1983 | "typedarray": "^0.0.6" 1984 | } 1985 | }, 1986 | "node_modules/config-chain": { 1987 | "version": "1.1.13", 1988 | "resolved": "https://registry.npmjs.org/config-chain/-/config-chain-1.1.13.tgz", 1989 | "integrity": "sha512-qj+f8APARXHrM0hraqXYb2/bOVSV4PvJQlNZ/DVj0QrmNM2q2euizkeuVckQ57J+W0mRH6Hvi+k50M4Jul2VRQ==", 1990 | "dev": true, 1991 | "optional": true, 1992 | "dependencies": { 1993 | "ini": "^1.3.4", 1994 | "proto-list": "~1.2.1" 1995 | } 1996 | }, 1997 | "node_modules/core-util-is": { 1998 | "version": "1.0.2", 1999 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 2000 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", 2001 | "dev": true 2002 | }, 2003 | "node_modules/cosmiconfig": { 2004 | "version": "7.0.1", 2005 | "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.0.1.tgz", 2006 | "integrity": "sha512-a1YWNUV2HwGimB7dU2s1wUMurNKjpx60HxBB6xUM8Re+2s1g1IIfJvFR0/iCF+XHdE0GMTKTuLR32UQff4TEyQ==", 2007 | "dev": true, 2008 | "dependencies": { 2009 | "@types/parse-json": "^4.0.0", 2010 | "import-fresh": "^3.2.1", 2011 | "parse-json": "^5.0.0", 2012 | "path-type": "^4.0.0", 2013 | "yaml": "^1.10.0" 2014 | }, 2015 | "engines": { 2016 | "node": ">=10" 2017 | } 2018 | }, 2019 | "node_modules/css-select": { 2020 | "version": "4.3.0", 2021 | "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.3.0.tgz", 2022 | "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", 2023 | "dev": true, 2024 | "dependencies": { 2025 | "boolbase": "^1.0.0", 2026 | "css-what": "^6.0.1", 2027 | "domhandler": "^4.3.1", 2028 | "domutils": "^2.8.0", 2029 | "nth-check": "^2.0.1" 2030 | }, 2031 | "funding": { 2032 | "url": "https://github.com/sponsors/fb55" 2033 | } 2034 | }, 2035 | "node_modules/css-tree": { 2036 | "version": "1.1.3", 2037 | "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.3.tgz", 2038 | "integrity": "sha512-tRpdppF7TRazZrjJ6v3stzv93qxRcSsFmW6cX0Zm2NVKpxE1WV1HblnghVv9TreireHkqI/VDEsfolRF1p6y7Q==", 2039 | "dev": true, 2040 | "dependencies": { 2041 | "mdn-data": "2.0.14", 2042 | "source-map": "^0.6.1" 2043 | }, 2044 | "engines": { 2045 | "node": ">=8.0.0" 2046 | } 2047 | }, 2048 | "node_modules/css-what": { 2049 | "version": "6.1.0", 2050 | "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz", 2051 | "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", 2052 | "dev": true, 2053 | "engines": { 2054 | "node": ">= 6" 2055 | }, 2056 | "funding": { 2057 | "url": "https://github.com/sponsors/fb55" 2058 | } 2059 | }, 2060 | "node_modules/csso": { 2061 | "version": "4.2.0", 2062 | "resolved": "https://registry.npmjs.org/csso/-/csso-4.2.0.tgz", 2063 | "integrity": "sha512-wvlcdIbf6pwKEk7vHj8/Bkc0B4ylXZruLvOgs9doS5eOsOpuodOV2zJChSpkp+pRpYQLQMeF04nr3Z68Sta9jA==", 2064 | "dev": true, 2065 | "dependencies": { 2066 | "css-tree": "^1.1.2" 2067 | }, 2068 | "engines": { 2069 | "node": ">=8.0.0" 2070 | } 2071 | }, 2072 | "node_modules/debug": { 2073 | "version": "4.3.4", 2074 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", 2075 | "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", 2076 | "dev": true, 2077 | "dependencies": { 2078 | "ms": "2.1.2" 2079 | }, 2080 | "engines": { 2081 | "node": ">=6.0" 2082 | }, 2083 | "peerDependenciesMeta": { 2084 | "supports-color": { 2085 | "optional": true 2086 | } 2087 | } 2088 | }, 2089 | "node_modules/decompress-response": { 2090 | "version": "3.3.0", 2091 | "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", 2092 | "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", 2093 | "dev": true, 2094 | "dependencies": { 2095 | "mimic-response": "^1.0.0" 2096 | }, 2097 | "engines": { 2098 | "node": ">=4" 2099 | } 2100 | }, 2101 | "node_modules/defer-to-connect": { 2102 | "version": "1.1.3", 2103 | "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", 2104 | "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", 2105 | "dev": true 2106 | }, 2107 | "node_modules/define-properties": { 2108 | "version": "1.1.3", 2109 | "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", 2110 | "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", 2111 | "dev": true, 2112 | "optional": true, 2113 | "dependencies": { 2114 | "object-keys": "^1.0.12" 2115 | }, 2116 | "engines": { 2117 | "node": ">= 0.4" 2118 | } 2119 | }, 2120 | "node_modules/detect-libc": { 2121 | "version": "1.0.3", 2122 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", 2123 | "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups=", 2124 | "dev": true, 2125 | "bin": { 2126 | "detect-libc": "bin/detect-libc.js" 2127 | }, 2128 | "engines": { 2129 | "node": ">=0.10" 2130 | } 2131 | }, 2132 | "node_modules/detect-node": { 2133 | "version": "2.1.0", 2134 | "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.1.0.tgz", 2135 | "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==", 2136 | "dev": true, 2137 | "optional": true 2138 | }, 2139 | "node_modules/dom-serializer": { 2140 | "version": "1.4.1", 2141 | "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.4.1.tgz", 2142 | "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", 2143 | "dev": true, 2144 | "dependencies": { 2145 | "domelementtype": "^2.0.1", 2146 | "domhandler": "^4.2.0", 2147 | "entities": "^2.0.0" 2148 | }, 2149 | "funding": { 2150 | "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" 2151 | } 2152 | }, 2153 | "node_modules/dom-serializer/node_modules/entities": { 2154 | "version": "2.2.0", 2155 | "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", 2156 | "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", 2157 | "dev": true, 2158 | "funding": { 2159 | "url": "https://github.com/fb55/entities?sponsor=1" 2160 | } 2161 | }, 2162 | "node_modules/domelementtype": { 2163 | "version": "2.3.0", 2164 | "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", 2165 | "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", 2166 | "dev": true, 2167 | "funding": [ 2168 | { 2169 | "type": "github", 2170 | "url": "https://github.com/sponsors/fb55" 2171 | } 2172 | ] 2173 | }, 2174 | "node_modules/domhandler": { 2175 | "version": "4.3.1", 2176 | "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.3.1.tgz", 2177 | "integrity": "sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==", 2178 | "dev": true, 2179 | "dependencies": { 2180 | "domelementtype": "^2.2.0" 2181 | }, 2182 | "engines": { 2183 | "node": ">= 4" 2184 | }, 2185 | "funding": { 2186 | "url": "https://github.com/fb55/domhandler?sponsor=1" 2187 | } 2188 | }, 2189 | "node_modules/domutils": { 2190 | "version": "2.8.0", 2191 | "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", 2192 | "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", 2193 | "dev": true, 2194 | "dependencies": { 2195 | "dom-serializer": "^1.0.1", 2196 | "domelementtype": "^2.2.0", 2197 | "domhandler": "^4.2.0" 2198 | }, 2199 | "funding": { 2200 | "url": "https://github.com/fb55/domutils?sponsor=1" 2201 | } 2202 | }, 2203 | "node_modules/dotenv": { 2204 | "version": "7.0.0", 2205 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-7.0.0.tgz", 2206 | "integrity": "sha512-M3NhsLbV1i6HuGzBUH8vXrtxOk+tWmzWKDMbAVSUp3Zsjm7ywFeuwrUXhmhQyRK1q5B5GGy7hcXPbj3bnfZg2g==", 2207 | "dev": true, 2208 | "engines": { 2209 | "node": ">=6" 2210 | } 2211 | }, 2212 | "node_modules/dotenv-expand": { 2213 | "version": "5.1.0", 2214 | "resolved": "https://registry.npmjs.org/dotenv-expand/-/dotenv-expand-5.1.0.tgz", 2215 | "integrity": "sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==", 2216 | "dev": true 2217 | }, 2218 | "node_modules/duplexer3": { 2219 | "version": "0.1.4", 2220 | "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", 2221 | "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", 2222 | "dev": true 2223 | }, 2224 | "node_modules/electron": { 2225 | "version": "17.1.2", 2226 | "resolved": "https://registry.npmjs.org/electron/-/electron-17.1.2.tgz", 2227 | "integrity": "sha512-hqKQaUIRWX5Y2eAD8FZINWD/e5TKdpkbBYbkcZmJS4Bd1PKQsaDVc9h5xoA8zZQkPymE9rss+swjRpAFurOPGQ==", 2228 | "dev": true, 2229 | "hasInstallScript": true, 2230 | "dependencies": { 2231 | "@electron/get": "^1.13.0", 2232 | "@types/node": "^14.6.2", 2233 | "extract-zip": "^1.0.3" 2234 | }, 2235 | "bin": { 2236 | "electron": "cli.js" 2237 | }, 2238 | "engines": { 2239 | "node": ">= 8.6" 2240 | } 2241 | }, 2242 | "node_modules/electron-to-chromium": { 2243 | "version": "1.4.137", 2244 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.137.tgz", 2245 | "integrity": "sha512-0Rcpald12O11BUogJagX3HsCN3FE83DSqWjgXoHo5a72KUKMSfI39XBgJpgNNxS9fuGzytaFjE06kZkiVFy2qA==", 2246 | "dev": true 2247 | }, 2248 | "node_modules/encodeurl": { 2249 | "version": "1.0.2", 2250 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 2251 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", 2252 | "dev": true, 2253 | "optional": true, 2254 | "engines": { 2255 | "node": ">= 0.8" 2256 | } 2257 | }, 2258 | "node_modules/end-of-stream": { 2259 | "version": "1.4.4", 2260 | "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", 2261 | "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", 2262 | "dev": true, 2263 | "dependencies": { 2264 | "once": "^1.4.0" 2265 | } 2266 | }, 2267 | "node_modules/entities": { 2268 | "version": "3.0.1", 2269 | "resolved": "https://registry.npmjs.org/entities/-/entities-3.0.1.tgz", 2270 | "integrity": "sha512-WiyBqoomrwMdFG1e0kqvASYfnlb0lp8M5o5Fw2OFq1hNZxxcNk8Ik0Xm7LxzBhuidnZB/UtBqVCgUz3kBOP51Q==", 2271 | "dev": true, 2272 | "engines": { 2273 | "node": ">=0.12" 2274 | }, 2275 | "funding": { 2276 | "url": "https://github.com/fb55/entities?sponsor=1" 2277 | } 2278 | }, 2279 | "node_modules/env-paths": { 2280 | "version": "2.2.1", 2281 | "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", 2282 | "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", 2283 | "dev": true, 2284 | "engines": { 2285 | "node": ">=6" 2286 | } 2287 | }, 2288 | "node_modules/error-ex": { 2289 | "version": "1.3.2", 2290 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 2291 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 2292 | "dev": true, 2293 | "dependencies": { 2294 | "is-arrayish": "^0.2.1" 2295 | } 2296 | }, 2297 | "node_modules/es6-error": { 2298 | "version": "4.1.1", 2299 | "resolved": "https://registry.npmjs.org/es6-error/-/es6-error-4.1.1.tgz", 2300 | "integrity": "sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg==", 2301 | "dev": true, 2302 | "optional": true 2303 | }, 2304 | "node_modules/escalade": { 2305 | "version": "3.1.1", 2306 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", 2307 | "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", 2308 | "dev": true, 2309 | "engines": { 2310 | "node": ">=6" 2311 | } 2312 | }, 2313 | "node_modules/escape-string-regexp": { 2314 | "version": "4.0.0", 2315 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 2316 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 2317 | "dev": true, 2318 | "optional": true, 2319 | "engines": { 2320 | "node": ">=10" 2321 | }, 2322 | "funding": { 2323 | "url": "https://github.com/sponsors/sindresorhus" 2324 | } 2325 | }, 2326 | "node_modules/extract-zip": { 2327 | "version": "1.7.0", 2328 | "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-1.7.0.tgz", 2329 | "integrity": "sha512-xoh5G1W/PB0/27lXgMQyIhP5DSY/LhoCsOyZgb+6iMmRtCwVBo55uKaMoEYrDCKQhWvqEip5ZPKAc6eFNyf/MA==", 2330 | "dev": true, 2331 | "dependencies": { 2332 | "concat-stream": "^1.6.2", 2333 | "debug": "^2.6.9", 2334 | "mkdirp": "^0.5.4", 2335 | "yauzl": "^2.10.0" 2336 | }, 2337 | "bin": { 2338 | "extract-zip": "cli.js" 2339 | } 2340 | }, 2341 | "node_modules/extract-zip/node_modules/debug": { 2342 | "version": "2.6.9", 2343 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 2344 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 2345 | "dev": true, 2346 | "dependencies": { 2347 | "ms": "2.0.0" 2348 | } 2349 | }, 2350 | "node_modules/extract-zip/node_modules/ms": { 2351 | "version": "2.0.0", 2352 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 2353 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", 2354 | "dev": true 2355 | }, 2356 | "node_modules/fd-slicer": { 2357 | "version": "1.1.0", 2358 | "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.1.0.tgz", 2359 | "integrity": "sha1-JcfInLH5B3+IkbvmHY85Dq4lbx4=", 2360 | "dev": true, 2361 | "dependencies": { 2362 | "pend": "~1.2.0" 2363 | } 2364 | }, 2365 | "node_modules/fs-extra": { 2366 | "version": "8.1.0", 2367 | "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", 2368 | "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", 2369 | "dev": true, 2370 | "dependencies": { 2371 | "graceful-fs": "^4.2.0", 2372 | "jsonfile": "^4.0.0", 2373 | "universalify": "^0.1.0" 2374 | }, 2375 | "engines": { 2376 | "node": ">=6 <7 || >=8" 2377 | } 2378 | }, 2379 | "node_modules/get-port": { 2380 | "version": "4.2.0", 2381 | "resolved": "https://registry.npmjs.org/get-port/-/get-port-4.2.0.tgz", 2382 | "integrity": "sha512-/b3jarXkH8KJoOMQc3uVGHASwGLPq3gSFJ7tgJm2diza+bydJPTGOibin2steecKeOylE8oY2JERlVWkAJO6yw==", 2383 | "dev": true, 2384 | "engines": { 2385 | "node": ">=6" 2386 | } 2387 | }, 2388 | "node_modules/get-stream": { 2389 | "version": "4.1.0", 2390 | "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", 2391 | "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", 2392 | "dev": true, 2393 | "dependencies": { 2394 | "pump": "^3.0.0" 2395 | }, 2396 | "engines": { 2397 | "node": ">=6" 2398 | } 2399 | }, 2400 | "node_modules/global-agent": { 2401 | "version": "3.0.0", 2402 | "resolved": "https://registry.npmjs.org/global-agent/-/global-agent-3.0.0.tgz", 2403 | "integrity": "sha512-PT6XReJ+D07JvGoxQMkT6qji/jVNfX/h364XHZOWeRzy64sSFr+xJ5OX7LI3b4MPQzdL4H8Y8M0xzPpsVMwA8Q==", 2404 | "dev": true, 2405 | "optional": true, 2406 | "dependencies": { 2407 | "boolean": "^3.0.1", 2408 | "es6-error": "^4.1.1", 2409 | "matcher": "^3.0.0", 2410 | "roarr": "^2.15.3", 2411 | "semver": "^7.3.2", 2412 | "serialize-error": "^7.0.1" 2413 | }, 2414 | "engines": { 2415 | "node": ">=10.0" 2416 | } 2417 | }, 2418 | "node_modules/global-agent/node_modules/semver": { 2419 | "version": "7.3.5", 2420 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", 2421 | "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", 2422 | "dev": true, 2423 | "optional": true, 2424 | "dependencies": { 2425 | "lru-cache": "^6.0.0" 2426 | }, 2427 | "bin": { 2428 | "semver": "bin/semver.js" 2429 | }, 2430 | "engines": { 2431 | "node": ">=10" 2432 | } 2433 | }, 2434 | "node_modules/global-tunnel-ng": { 2435 | "version": "2.7.1", 2436 | "resolved": "https://registry.npmjs.org/global-tunnel-ng/-/global-tunnel-ng-2.7.1.tgz", 2437 | "integrity": "sha512-4s+DyciWBV0eK148wqXxcmVAbFVPqtc3sEtUE/GTQfuU80rySLcMhUmHKSHI7/LDj8q0gDYI1lIhRRB7ieRAqg==", 2438 | "dev": true, 2439 | "optional": true, 2440 | "dependencies": { 2441 | "encodeurl": "^1.0.2", 2442 | "lodash": "^4.17.10", 2443 | "npm-conf": "^1.1.3", 2444 | "tunnel": "^0.0.6" 2445 | }, 2446 | "engines": { 2447 | "node": ">=0.10" 2448 | } 2449 | }, 2450 | "node_modules/globals": { 2451 | "version": "13.15.0", 2452 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.15.0.tgz", 2453 | "integrity": "sha512-bpzcOlgDhMG070Av0Vy5Owklpv1I6+j96GhUI7Rh7IzDCKLzboflLrrfqMu8NquDbiR4EOQk7XzJwqVJxicxog==", 2454 | "dev": true, 2455 | "dependencies": { 2456 | "type-fest": "^0.20.2" 2457 | }, 2458 | "engines": { 2459 | "node": ">=8" 2460 | }, 2461 | "funding": { 2462 | "url": "https://github.com/sponsors/sindresorhus" 2463 | } 2464 | }, 2465 | "node_modules/globals/node_modules/type-fest": { 2466 | "version": "0.20.2", 2467 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", 2468 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", 2469 | "dev": true, 2470 | "engines": { 2471 | "node": ">=10" 2472 | }, 2473 | "funding": { 2474 | "url": "https://github.com/sponsors/sindresorhus" 2475 | } 2476 | }, 2477 | "node_modules/globalthis": { 2478 | "version": "1.0.2", 2479 | "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.2.tgz", 2480 | "integrity": "sha512-ZQnSFO1la8P7auIOQECnm0sSuoMeaSq0EEdXMBFF2QJO4uNcwbyhSgG3MruWNbFTqCLmxVwGOl7LZ9kASvHdeQ==", 2481 | "dev": true, 2482 | "optional": true, 2483 | "dependencies": { 2484 | "define-properties": "^1.1.3" 2485 | }, 2486 | "engines": { 2487 | "node": ">= 0.4" 2488 | }, 2489 | "funding": { 2490 | "url": "https://github.com/sponsors/ljharb" 2491 | } 2492 | }, 2493 | "node_modules/got": { 2494 | "version": "9.6.0", 2495 | "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", 2496 | "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", 2497 | "dev": true, 2498 | "dependencies": { 2499 | "@sindresorhus/is": "^0.14.0", 2500 | "@szmarczak/http-timer": "^1.1.2", 2501 | "cacheable-request": "^6.0.0", 2502 | "decompress-response": "^3.3.0", 2503 | "duplexer3": "^0.1.4", 2504 | "get-stream": "^4.1.0", 2505 | "lowercase-keys": "^1.0.1", 2506 | "mimic-response": "^1.0.1", 2507 | "p-cancelable": "^1.0.0", 2508 | "to-readable-stream": "^1.0.0", 2509 | "url-parse-lax": "^3.0.0" 2510 | }, 2511 | "engines": { 2512 | "node": ">=8.6" 2513 | } 2514 | }, 2515 | "node_modules/graceful-fs": { 2516 | "version": "4.2.9", 2517 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.9.tgz", 2518 | "integrity": "sha512-NtNxqUcXgpW2iMrfqSfR73Glt39K+BLwWsPs94yR63v45T0Wbej7eRmL5cWfwEgqXnmjQp3zaJTshdRW/qC2ZQ==", 2519 | "dev": true 2520 | }, 2521 | "node_modules/has-flag": { 2522 | "version": "4.0.0", 2523 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 2524 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 2525 | "dev": true, 2526 | "engines": { 2527 | "node": ">=8" 2528 | } 2529 | }, 2530 | "node_modules/htmlnano": { 2531 | "version": "2.0.2", 2532 | "resolved": "https://registry.npmjs.org/htmlnano/-/htmlnano-2.0.2.tgz", 2533 | "integrity": "sha512-+ZrQFS4Ub+zd+/fWwfvoYCEGNEa0/zrpys6CyXxvZDwtL7Pl+pOtRkiujyvBQ7Lmfp7/iEPxtOFgxWA16Gkj3w==", 2534 | "dev": true, 2535 | "dependencies": { 2536 | "cosmiconfig": "^7.0.1", 2537 | "posthtml": "^0.16.5", 2538 | "timsort": "^0.3.0" 2539 | }, 2540 | "peerDependencies": { 2541 | "cssnano": "^5.0.11", 2542 | "postcss": "^8.3.11", 2543 | "purgecss": "^4.0.3", 2544 | "relateurl": "^0.2.7", 2545 | "srcset": "^5.0.0", 2546 | "svgo": "^2.8.0", 2547 | "terser": "^5.10.0", 2548 | "uncss": "^0.17.3" 2549 | }, 2550 | "peerDependenciesMeta": { 2551 | "cssnano": { 2552 | "optional": true 2553 | }, 2554 | "postcss": { 2555 | "optional": true 2556 | }, 2557 | "purgecss": { 2558 | "optional": true 2559 | }, 2560 | "relateurl": { 2561 | "optional": true 2562 | }, 2563 | "srcset": { 2564 | "optional": true 2565 | }, 2566 | "svgo": { 2567 | "optional": true 2568 | }, 2569 | "terser": { 2570 | "optional": true 2571 | }, 2572 | "uncss": { 2573 | "optional": true 2574 | } 2575 | } 2576 | }, 2577 | "node_modules/htmlparser2": { 2578 | "version": "7.2.0", 2579 | "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-7.2.0.tgz", 2580 | "integrity": "sha512-H7MImA4MS6cw7nbyURtLPO1Tms7C5H602LRETv95z1MxO/7CP7rDVROehUYeYBUYEON94NXXDEPmZuq+hX4sog==", 2581 | "dev": true, 2582 | "funding": [ 2583 | "https://github.com/fb55/htmlparser2?sponsor=1", 2584 | { 2585 | "type": "github", 2586 | "url": "https://github.com/sponsors/fb55" 2587 | } 2588 | ], 2589 | "dependencies": { 2590 | "domelementtype": "^2.0.1", 2591 | "domhandler": "^4.2.2", 2592 | "domutils": "^2.8.0", 2593 | "entities": "^3.0.1" 2594 | } 2595 | }, 2596 | "node_modules/http-cache-semantics": { 2597 | "version": "4.1.0", 2598 | "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", 2599 | "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", 2600 | "dev": true 2601 | }, 2602 | "node_modules/import-fresh": { 2603 | "version": "3.3.0", 2604 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", 2605 | "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", 2606 | "dev": true, 2607 | "dependencies": { 2608 | "parent-module": "^1.0.0", 2609 | "resolve-from": "^4.0.0" 2610 | }, 2611 | "engines": { 2612 | "node": ">=6" 2613 | }, 2614 | "funding": { 2615 | "url": "https://github.com/sponsors/sindresorhus" 2616 | } 2617 | }, 2618 | "node_modules/inherits": { 2619 | "version": "2.0.4", 2620 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 2621 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 2622 | "dev": true 2623 | }, 2624 | "node_modules/ini": { 2625 | "version": "1.3.8", 2626 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", 2627 | "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", 2628 | "dev": true, 2629 | "optional": true 2630 | }, 2631 | "node_modules/is-arrayish": { 2632 | "version": "0.2.1", 2633 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 2634 | "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", 2635 | "dev": true 2636 | }, 2637 | "node_modules/is-json": { 2638 | "version": "2.0.1", 2639 | "resolved": "https://registry.npmjs.org/is-json/-/is-json-2.0.1.tgz", 2640 | "integrity": "sha1-a+Fm0USCihMdaGiRuYPfYsOUkf8=", 2641 | "dev": true 2642 | }, 2643 | "node_modules/isarray": { 2644 | "version": "1.0.0", 2645 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 2646 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", 2647 | "dev": true 2648 | }, 2649 | "node_modules/js-tokens": { 2650 | "version": "4.0.0", 2651 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 2652 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 2653 | "dev": true 2654 | }, 2655 | "node_modules/json-buffer": { 2656 | "version": "3.0.0", 2657 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", 2658 | "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", 2659 | "dev": true 2660 | }, 2661 | "node_modules/json-parse-even-better-errors": { 2662 | "version": "2.3.1", 2663 | "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", 2664 | "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", 2665 | "dev": true 2666 | }, 2667 | "node_modules/json-stringify-safe": { 2668 | "version": "5.0.1", 2669 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 2670 | "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", 2671 | "dev": true, 2672 | "optional": true 2673 | }, 2674 | "node_modules/json5": { 2675 | "version": "2.2.1", 2676 | "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.1.tgz", 2677 | "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", 2678 | "dev": true, 2679 | "bin": { 2680 | "json5": "lib/cli.js" 2681 | }, 2682 | "engines": { 2683 | "node": ">=6" 2684 | } 2685 | }, 2686 | "node_modules/jsonfile": { 2687 | "version": "4.0.0", 2688 | "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", 2689 | "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", 2690 | "dev": true, 2691 | "optionalDependencies": { 2692 | "graceful-fs": "^4.1.6" 2693 | } 2694 | }, 2695 | "node_modules/keyv": { 2696 | "version": "3.1.0", 2697 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", 2698 | "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", 2699 | "dev": true, 2700 | "dependencies": { 2701 | "json-buffer": "3.0.0" 2702 | } 2703 | }, 2704 | "node_modules/lines-and-columns": { 2705 | "version": "1.2.4", 2706 | "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", 2707 | "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", 2708 | "dev": true 2709 | }, 2710 | "node_modules/lmdb": { 2711 | "version": "2.2.4", 2712 | "resolved": "https://registry.npmjs.org/lmdb/-/lmdb-2.2.4.tgz", 2713 | "integrity": "sha512-gto+BB2uEob8qRiTlOq+R3uX0YNHsX9mjxj9Sbdue/LIKqu6IlZjrsjKeGyOMquc/474GEqFyX2pdytpydp0rQ==", 2714 | "dev": true, 2715 | "hasInstallScript": true, 2716 | "dependencies": { 2717 | "msgpackr": "^1.5.4", 2718 | "nan": "^2.14.2", 2719 | "node-gyp-build": "^4.2.3", 2720 | "ordered-binary": "^1.2.4", 2721 | "weak-lru-cache": "^1.2.2" 2722 | } 2723 | }, 2724 | "node_modules/lodash": { 2725 | "version": "4.17.21", 2726 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 2727 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 2728 | "dev": true, 2729 | "optional": true 2730 | }, 2731 | "node_modules/lodash.sortby": { 2732 | "version": "4.7.0", 2733 | "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", 2734 | "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=", 2735 | "dev": true 2736 | }, 2737 | "node_modules/lowercase-keys": { 2738 | "version": "1.0.1", 2739 | "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", 2740 | "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", 2741 | "dev": true, 2742 | "engines": { 2743 | "node": ">=0.10.0" 2744 | } 2745 | }, 2746 | "node_modules/lru-cache": { 2747 | "version": "6.0.0", 2748 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 2749 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 2750 | "dev": true, 2751 | "optional": true, 2752 | "dependencies": { 2753 | "yallist": "^4.0.0" 2754 | }, 2755 | "engines": { 2756 | "node": ">=10" 2757 | } 2758 | }, 2759 | "node_modules/matcher": { 2760 | "version": "3.0.0", 2761 | "resolved": "https://registry.npmjs.org/matcher/-/matcher-3.0.0.tgz", 2762 | "integrity": "sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng==", 2763 | "dev": true, 2764 | "optional": true, 2765 | "dependencies": { 2766 | "escape-string-regexp": "^4.0.0" 2767 | }, 2768 | "engines": { 2769 | "node": ">=10" 2770 | } 2771 | }, 2772 | "node_modules/mdn-data": { 2773 | "version": "2.0.14", 2774 | "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", 2775 | "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==", 2776 | "dev": true 2777 | }, 2778 | "node_modules/mimic-response": { 2779 | "version": "1.0.1", 2780 | "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", 2781 | "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", 2782 | "dev": true, 2783 | "engines": { 2784 | "node": ">=4" 2785 | } 2786 | }, 2787 | "node_modules/minimist": { 2788 | "version": "1.2.6", 2789 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.6.tgz", 2790 | "integrity": "sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==", 2791 | "dev": true 2792 | }, 2793 | "node_modules/mkdirp": { 2794 | "version": "0.5.5", 2795 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 2796 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 2797 | "dev": true, 2798 | "dependencies": { 2799 | "minimist": "^1.2.5" 2800 | }, 2801 | "bin": { 2802 | "mkdirp": "bin/cmd.js" 2803 | } 2804 | }, 2805 | "node_modules/ms": { 2806 | "version": "2.1.2", 2807 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 2808 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", 2809 | "dev": true 2810 | }, 2811 | "node_modules/msgpackr": { 2812 | "version": "1.6.0", 2813 | "resolved": "https://registry.npmjs.org/msgpackr/-/msgpackr-1.6.0.tgz", 2814 | "integrity": "sha512-CJs2OuaIuwpP2iLZx6vl/jfl7WqFNFrYpkp/BC1ctzCbYAACyT9lYMACstgvH4pTcBrCFk4uzOoOZj0gFP/0EA==", 2815 | "dev": true, 2816 | "optionalDependencies": { 2817 | "msgpackr-extract": "^2.0.2" 2818 | } 2819 | }, 2820 | "node_modules/msgpackr-extract": { 2821 | "version": "2.0.2", 2822 | "resolved": "https://registry.npmjs.org/msgpackr-extract/-/msgpackr-extract-2.0.2.tgz", 2823 | "integrity": "sha512-coskCeJG2KDny23zWeu+6tNy7BLnAiOGgiwzlgdm4oeSsTpqEJJPguHIuKZcCdB7tzhZbXNYSg6jZAXkZErkJA==", 2824 | "dev": true, 2825 | "hasInstallScript": true, 2826 | "optional": true, 2827 | "dependencies": { 2828 | "node-gyp-build-optional-packages": "5.0.2" 2829 | }, 2830 | "optionalDependencies": { 2831 | "@msgpackr-extract/msgpackr-extract-darwin-arm64": "2.0.2", 2832 | "@msgpackr-extract/msgpackr-extract-darwin-x64": "2.0.2", 2833 | "@msgpackr-extract/msgpackr-extract-linux-arm": "2.0.2", 2834 | "@msgpackr-extract/msgpackr-extract-linux-arm64": "2.0.2", 2835 | "@msgpackr-extract/msgpackr-extract-linux-x64": "2.0.2", 2836 | "@msgpackr-extract/msgpackr-extract-win32-x64": "2.0.2" 2837 | } 2838 | }, 2839 | "node_modules/msgpackr-extract/node_modules/node-gyp-build-optional-packages": { 2840 | "version": "5.0.2", 2841 | "resolved": "https://registry.npmjs.org/node-gyp-build-optional-packages/-/node-gyp-build-optional-packages-5.0.2.tgz", 2842 | "integrity": "sha512-PiN4NWmlQPqvbEFcH/omQsswWQbe5Z9YK/zdB23irp5j2XibaA2IrGvpSWmVVG4qMZdmPdwPctSy4a86rOMn6g==", 2843 | "dev": true, 2844 | "optional": true, 2845 | "bin": { 2846 | "node-gyp-build-optional": "optional.js", 2847 | "node-gyp-build-optional-packages": "bin.js", 2848 | "node-gyp-build-test": "build-test.js" 2849 | } 2850 | }, 2851 | "node_modules/nan": { 2852 | "version": "2.15.0", 2853 | "resolved": "https://registry.npmjs.org/nan/-/nan-2.15.0.tgz", 2854 | "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==", 2855 | "dev": true 2856 | }, 2857 | "node_modules/nanoid": { 2858 | "version": "3.3.4", 2859 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", 2860 | "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", 2861 | "dev": true, 2862 | "bin": { 2863 | "nanoid": "bin/nanoid.cjs" 2864 | }, 2865 | "engines": { 2866 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 2867 | } 2868 | }, 2869 | "node_modules/node-addon-api": { 2870 | "version": "3.2.1", 2871 | "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-3.2.1.tgz", 2872 | "integrity": "sha512-mmcei9JghVNDYydghQmeDX8KoAm0FAiYyIcUt/N4nhyAipB17pllZQDOJD2fotxABnt4Mdz+dKTO7eftLg4d0A==", 2873 | "dev": true 2874 | }, 2875 | "node_modules/node-gyp-build": { 2876 | "version": "4.4.0", 2877 | "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.4.0.tgz", 2878 | "integrity": "sha512-amJnQCcgtRVw9SvoebO3BKGESClrfXGCUTX9hSn1OuGQTQBOZmVd0Z0OlecpuRksKvbsUqALE8jls/ErClAPuQ==", 2879 | "dev": true, 2880 | "bin": { 2881 | "node-gyp-build": "bin.js", 2882 | "node-gyp-build-optional": "optional.js", 2883 | "node-gyp-build-test": "build-test.js" 2884 | } 2885 | }, 2886 | "node_modules/node-releases": { 2887 | "version": "2.0.4", 2888 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.4.tgz", 2889 | "integrity": "sha512-gbMzqQtTtDz/00jQzZ21PQzdI9PyLYqUSvD0p3naOhX4odFji0ZxYdnVwPTxmSwkmxhcFImpozceidSG+AgoPQ==", 2890 | "dev": true 2891 | }, 2892 | "node_modules/normalize-url": { 2893 | "version": "4.5.1", 2894 | "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.1.tgz", 2895 | "integrity": "sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==", 2896 | "dev": true, 2897 | "engines": { 2898 | "node": ">=8" 2899 | } 2900 | }, 2901 | "node_modules/npm-conf": { 2902 | "version": "1.1.3", 2903 | "resolved": "https://registry.npmjs.org/npm-conf/-/npm-conf-1.1.3.tgz", 2904 | "integrity": "sha512-Yic4bZHJOt9RCFbRP3GgpqhScOY4HH3V2P8yBj6CeYq118Qr+BLXqT2JvpJ00mryLESpgOxf5XlFv4ZjXxLScw==", 2905 | "dev": true, 2906 | "optional": true, 2907 | "dependencies": { 2908 | "config-chain": "^1.1.11", 2909 | "pify": "^3.0.0" 2910 | }, 2911 | "engines": { 2912 | "node": ">=4" 2913 | } 2914 | }, 2915 | "node_modules/nth-check": { 2916 | "version": "2.1.1", 2917 | "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", 2918 | "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", 2919 | "dev": true, 2920 | "dependencies": { 2921 | "boolbase": "^1.0.0" 2922 | }, 2923 | "funding": { 2924 | "url": "https://github.com/fb55/nth-check?sponsor=1" 2925 | } 2926 | }, 2927 | "node_modules/nullthrows": { 2928 | "version": "1.1.1", 2929 | "resolved": "https://registry.npmjs.org/nullthrows/-/nullthrows-1.1.1.tgz", 2930 | "integrity": "sha512-2vPPEi+Z7WqML2jZYddDIfy5Dqb0r2fze2zTxNNknZaFpVHU3mFB3R+DWeJWGVx0ecvttSGlJTI+WG+8Z4cDWw==", 2931 | "dev": true 2932 | }, 2933 | "node_modules/object-keys": { 2934 | "version": "1.1.1", 2935 | "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", 2936 | "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", 2937 | "dev": true, 2938 | "optional": true, 2939 | "engines": { 2940 | "node": ">= 0.4" 2941 | } 2942 | }, 2943 | "node_modules/once": { 2944 | "version": "1.4.0", 2945 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2946 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 2947 | "dev": true, 2948 | "dependencies": { 2949 | "wrappy": "1" 2950 | } 2951 | }, 2952 | "node_modules/ordered-binary": { 2953 | "version": "1.2.5", 2954 | "resolved": "https://registry.npmjs.org/ordered-binary/-/ordered-binary-1.2.5.tgz", 2955 | "integrity": "sha512-djRmZoEpOGvIRW7ufsCDHtvcUa18UC9TxnPbHhSVFZHsoyg0dtut1bWtBZ/fmxdPN62oWXrV6adM7NoWU+CneA==", 2956 | "dev": true 2957 | }, 2958 | "node_modules/p-cancelable": { 2959 | "version": "1.1.0", 2960 | "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", 2961 | "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", 2962 | "dev": true, 2963 | "engines": { 2964 | "node": ">=6" 2965 | } 2966 | }, 2967 | "node_modules/parcel": { 2968 | "version": "2.5.0", 2969 | "resolved": "https://registry.npmjs.org/parcel/-/parcel-2.5.0.tgz", 2970 | "integrity": "sha512-er0mj/BaMjWyzQ/jedLUi/LNAuQcFT8lCvoNqANF+jTaX9rohaBwxIvKVJVAZgyCnmyfbbldp496wPMW0R0+CA==", 2971 | "dev": true, 2972 | "dependencies": { 2973 | "@parcel/config-default": "2.5.0", 2974 | "@parcel/core": "2.5.0", 2975 | "@parcel/diagnostic": "2.5.0", 2976 | "@parcel/events": "2.5.0", 2977 | "@parcel/fs": "2.5.0", 2978 | "@parcel/logger": "2.5.0", 2979 | "@parcel/package-manager": "2.5.0", 2980 | "@parcel/reporter-cli": "2.5.0", 2981 | "@parcel/reporter-dev-server": "2.5.0", 2982 | "@parcel/utils": "2.5.0", 2983 | "chalk": "^4.1.0", 2984 | "commander": "^7.0.0", 2985 | "get-port": "^4.2.0", 2986 | "v8-compile-cache": "^2.0.0" 2987 | }, 2988 | "bin": { 2989 | "parcel": "lib/bin.js" 2990 | }, 2991 | "engines": { 2992 | "node": ">= 12.0.0" 2993 | }, 2994 | "funding": { 2995 | "type": "opencollective", 2996 | "url": "https://opencollective.com/parcel" 2997 | } 2998 | }, 2999 | "node_modules/parent-module": { 3000 | "version": "1.0.1", 3001 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 3002 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 3003 | "dev": true, 3004 | "dependencies": { 3005 | "callsites": "^3.0.0" 3006 | }, 3007 | "engines": { 3008 | "node": ">=6" 3009 | } 3010 | }, 3011 | "node_modules/parse-json": { 3012 | "version": "5.2.0", 3013 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", 3014 | "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", 3015 | "dev": true, 3016 | "dependencies": { 3017 | "@babel/code-frame": "^7.0.0", 3018 | "error-ex": "^1.3.1", 3019 | "json-parse-even-better-errors": "^2.3.0", 3020 | "lines-and-columns": "^1.1.6" 3021 | }, 3022 | "engines": { 3023 | "node": ">=8" 3024 | }, 3025 | "funding": { 3026 | "url": "https://github.com/sponsors/sindresorhus" 3027 | } 3028 | }, 3029 | "node_modules/path-type": { 3030 | "version": "4.0.0", 3031 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", 3032 | "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", 3033 | "dev": true, 3034 | "engines": { 3035 | "node": ">=8" 3036 | } 3037 | }, 3038 | "node_modules/pend": { 3039 | "version": "1.2.0", 3040 | "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", 3041 | "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA=", 3042 | "dev": true 3043 | }, 3044 | "node_modules/picocolors": { 3045 | "version": "1.0.0", 3046 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", 3047 | "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", 3048 | "dev": true 3049 | }, 3050 | "node_modules/pify": { 3051 | "version": "3.0.0", 3052 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 3053 | "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", 3054 | "dev": true, 3055 | "optional": true, 3056 | "engines": { 3057 | "node": ">=4" 3058 | } 3059 | }, 3060 | "node_modules/postcss": { 3061 | "version": "8.4.14", 3062 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.14.tgz", 3063 | "integrity": "sha512-E398TUmfAYFPBSdzgeieK2Y1+1cpdxJx8yXbK/m57nRhKSmk1GB2tO4lbLBtlkfPQTDKfe4Xqv1ASWPpayPEig==", 3064 | "dev": true, 3065 | "funding": [ 3066 | { 3067 | "type": "opencollective", 3068 | "url": "https://opencollective.com/postcss/" 3069 | }, 3070 | { 3071 | "type": "tidelift", 3072 | "url": "https://tidelift.com/funding/github/npm/postcss" 3073 | } 3074 | ], 3075 | "dependencies": { 3076 | "nanoid": "^3.3.4", 3077 | "picocolors": "^1.0.0", 3078 | "source-map-js": "^1.0.2" 3079 | }, 3080 | "engines": { 3081 | "node": "^10 || ^12 || >=14" 3082 | } 3083 | }, 3084 | "node_modules/postcss-value-parser": { 3085 | "version": "4.2.0", 3086 | "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", 3087 | "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", 3088 | "dev": true 3089 | }, 3090 | "node_modules/posthtml": { 3091 | "version": "0.16.6", 3092 | "resolved": "https://registry.npmjs.org/posthtml/-/posthtml-0.16.6.tgz", 3093 | "integrity": "sha512-JcEmHlyLK/o0uGAlj65vgg+7LIms0xKXe60lcDOTU7oVX/3LuEuLwrQpW3VJ7de5TaFKiW4kWkaIpJL42FEgxQ==", 3094 | "dev": true, 3095 | "dependencies": { 3096 | "posthtml-parser": "^0.11.0", 3097 | "posthtml-render": "^3.0.0" 3098 | }, 3099 | "engines": { 3100 | "node": ">=12.0.0" 3101 | } 3102 | }, 3103 | "node_modules/posthtml-parser": { 3104 | "version": "0.11.0", 3105 | "resolved": "https://registry.npmjs.org/posthtml-parser/-/posthtml-parser-0.11.0.tgz", 3106 | "integrity": "sha512-QecJtfLekJbWVo/dMAA+OSwY79wpRmbqS5TeXvXSX+f0c6pW4/SE6inzZ2qkU7oAMCPqIDkZDvd/bQsSFUnKyw==", 3107 | "dev": true, 3108 | "dependencies": { 3109 | "htmlparser2": "^7.1.1" 3110 | }, 3111 | "engines": { 3112 | "node": ">=12" 3113 | } 3114 | }, 3115 | "node_modules/posthtml-render": { 3116 | "version": "3.0.0", 3117 | "resolved": "https://registry.npmjs.org/posthtml-render/-/posthtml-render-3.0.0.tgz", 3118 | "integrity": "sha512-z+16RoxK3fUPgwaIgH9NGnK1HKY9XIDpydky5eQGgAFVXTCSezalv9U2jQuNV+Z9qV1fDWNzldcw4eK0SSbqKA==", 3119 | "dev": true, 3120 | "dependencies": { 3121 | "is-json": "^2.0.1" 3122 | }, 3123 | "engines": { 3124 | "node": ">=12" 3125 | } 3126 | }, 3127 | "node_modules/prepend-http": { 3128 | "version": "2.0.0", 3129 | "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", 3130 | "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", 3131 | "dev": true, 3132 | "engines": { 3133 | "node": ">=4" 3134 | } 3135 | }, 3136 | "node_modules/process-nextick-args": { 3137 | "version": "2.0.1", 3138 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 3139 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", 3140 | "dev": true 3141 | }, 3142 | "node_modules/progress": { 3143 | "version": "2.0.3", 3144 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", 3145 | "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", 3146 | "dev": true, 3147 | "engines": { 3148 | "node": ">=0.4.0" 3149 | } 3150 | }, 3151 | "node_modules/proto-list": { 3152 | "version": "1.2.4", 3153 | "resolved": "https://registry.npmjs.org/proto-list/-/proto-list-1.2.4.tgz", 3154 | "integrity": "sha1-IS1b/hMYMGpCD2QCuOJv85ZHqEk=", 3155 | "dev": true, 3156 | "optional": true 3157 | }, 3158 | "node_modules/pump": { 3159 | "version": "3.0.0", 3160 | "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", 3161 | "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", 3162 | "dev": true, 3163 | "dependencies": { 3164 | "end-of-stream": "^1.1.0", 3165 | "once": "^1.3.1" 3166 | } 3167 | }, 3168 | "node_modules/punycode": { 3169 | "version": "2.1.1", 3170 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 3171 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 3172 | "dev": true, 3173 | "engines": { 3174 | "node": ">=6" 3175 | } 3176 | }, 3177 | "node_modules/react-refresh": { 3178 | "version": "0.9.0", 3179 | "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.9.0.tgz", 3180 | "integrity": "sha512-Gvzk7OZpiqKSkxsQvO/mbTN1poglhmAV7gR/DdIrRrSMXraRQQlfikRJOr3Nb9GTMPC5kof948Zy6jJZIFtDvQ==", 3181 | "dev": true, 3182 | "engines": { 3183 | "node": ">=0.10.0" 3184 | } 3185 | }, 3186 | "node_modules/readable-stream": { 3187 | "version": "2.3.7", 3188 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 3189 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 3190 | "dev": true, 3191 | "dependencies": { 3192 | "core-util-is": "~1.0.0", 3193 | "inherits": "~2.0.3", 3194 | "isarray": "~1.0.0", 3195 | "process-nextick-args": "~2.0.0", 3196 | "safe-buffer": "~5.1.1", 3197 | "string_decoder": "~1.1.1", 3198 | "util-deprecate": "~1.0.1" 3199 | } 3200 | }, 3201 | "node_modules/regenerator-runtime": { 3202 | "version": "0.13.9", 3203 | "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz", 3204 | "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==", 3205 | "dev": true 3206 | }, 3207 | "node_modules/resolve-from": { 3208 | "version": "4.0.0", 3209 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 3210 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 3211 | "dev": true, 3212 | "engines": { 3213 | "node": ">=4" 3214 | } 3215 | }, 3216 | "node_modules/responselike": { 3217 | "version": "1.0.2", 3218 | "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", 3219 | "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", 3220 | "dev": true, 3221 | "dependencies": { 3222 | "lowercase-keys": "^1.0.0" 3223 | } 3224 | }, 3225 | "node_modules/roarr": { 3226 | "version": "2.15.4", 3227 | "resolved": "https://registry.npmjs.org/roarr/-/roarr-2.15.4.tgz", 3228 | "integrity": "sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A==", 3229 | "dev": true, 3230 | "optional": true, 3231 | "dependencies": { 3232 | "boolean": "^3.0.1", 3233 | "detect-node": "^2.0.4", 3234 | "globalthis": "^1.0.1", 3235 | "json-stringify-safe": "^5.0.1", 3236 | "semver-compare": "^1.0.0", 3237 | "sprintf-js": "^1.1.2" 3238 | }, 3239 | "engines": { 3240 | "node": ">=8.0" 3241 | } 3242 | }, 3243 | "node_modules/safe-buffer": { 3244 | "version": "5.1.2", 3245 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 3246 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 3247 | "dev": true 3248 | }, 3249 | "node_modules/semver": { 3250 | "version": "6.3.0", 3251 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", 3252 | "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", 3253 | "dev": true, 3254 | "bin": { 3255 | "semver": "bin/semver.js" 3256 | } 3257 | }, 3258 | "node_modules/semver-compare": { 3259 | "version": "1.0.0", 3260 | "resolved": "https://registry.npmjs.org/semver-compare/-/semver-compare-1.0.0.tgz", 3261 | "integrity": "sha1-De4hahyUGrN+nvsXiPavxf9VN/w=", 3262 | "dev": true, 3263 | "optional": true 3264 | }, 3265 | "node_modules/serialize-error": { 3266 | "version": "7.0.1", 3267 | "resolved": "https://registry.npmjs.org/serialize-error/-/serialize-error-7.0.1.tgz", 3268 | "integrity": "sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw==", 3269 | "dev": true, 3270 | "optional": true, 3271 | "dependencies": { 3272 | "type-fest": "^0.13.1" 3273 | }, 3274 | "engines": { 3275 | "node": ">=10" 3276 | }, 3277 | "funding": { 3278 | "url": "https://github.com/sponsors/sindresorhus" 3279 | } 3280 | }, 3281 | "node_modules/sortablejs": { 3282 | "version": "1.15.0", 3283 | "resolved": "https://registry.npmjs.org/sortablejs/-/sortablejs-1.15.0.tgz", 3284 | "integrity": "sha512-bv9qgVMjUMf89wAvM6AxVvS/4MX3sPeN0+agqShejLU5z5GX4C75ow1O2e5k4L6XItUyAK3gH6AxSbXrOM5e8w==", 3285 | "dev": true 3286 | }, 3287 | "node_modules/source-map": { 3288 | "version": "0.6.1", 3289 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 3290 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 3291 | "dev": true, 3292 | "engines": { 3293 | "node": ">=0.10.0" 3294 | } 3295 | }, 3296 | "node_modules/source-map-js": { 3297 | "version": "1.0.2", 3298 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", 3299 | "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", 3300 | "dev": true, 3301 | "engines": { 3302 | "node": ">=0.10.0" 3303 | } 3304 | }, 3305 | "node_modules/source-map-support": { 3306 | "version": "0.5.21", 3307 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", 3308 | "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", 3309 | "dev": true, 3310 | "dependencies": { 3311 | "buffer-from": "^1.0.0", 3312 | "source-map": "^0.6.0" 3313 | } 3314 | }, 3315 | "node_modules/sprintf-js": { 3316 | "version": "1.1.2", 3317 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", 3318 | "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", 3319 | "dev": true, 3320 | "optional": true 3321 | }, 3322 | "node_modules/stable": { 3323 | "version": "0.1.8", 3324 | "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", 3325 | "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", 3326 | "deprecated": "Modern JS already guarantees Array#sort() is a stable sort, so this library is deprecated. See the compatibility table on MDN: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#browser_compatibility", 3327 | "dev": true 3328 | }, 3329 | "node_modules/string_decoder": { 3330 | "version": "1.1.1", 3331 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 3332 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 3333 | "dev": true, 3334 | "dependencies": { 3335 | "safe-buffer": "~5.1.0" 3336 | } 3337 | }, 3338 | "node_modules/sumchecker": { 3339 | "version": "3.0.1", 3340 | "resolved": "https://registry.npmjs.org/sumchecker/-/sumchecker-3.0.1.tgz", 3341 | "integrity": "sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg==", 3342 | "dev": true, 3343 | "dependencies": { 3344 | "debug": "^4.1.0" 3345 | }, 3346 | "engines": { 3347 | "node": ">= 8.0" 3348 | } 3349 | }, 3350 | "node_modules/supports-color": { 3351 | "version": "7.2.0", 3352 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 3353 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 3354 | "dev": true, 3355 | "dependencies": { 3356 | "has-flag": "^4.0.0" 3357 | }, 3358 | "engines": { 3359 | "node": ">=8" 3360 | } 3361 | }, 3362 | "node_modules/svgo": { 3363 | "version": "2.8.0", 3364 | "resolved": "https://registry.npmjs.org/svgo/-/svgo-2.8.0.tgz", 3365 | "integrity": "sha512-+N/Q9kV1+F+UeWYoSiULYo4xYSDQlTgb+ayMobAXPwMnLvop7oxKMo9OzIrX5x3eS4L4f2UHhc9axXwY8DpChg==", 3366 | "dev": true, 3367 | "dependencies": { 3368 | "@trysound/sax": "0.2.0", 3369 | "commander": "^7.2.0", 3370 | "css-select": "^4.1.3", 3371 | "css-tree": "^1.1.3", 3372 | "csso": "^4.2.0", 3373 | "picocolors": "^1.0.0", 3374 | "stable": "^0.1.8" 3375 | }, 3376 | "bin": { 3377 | "svgo": "bin/svgo" 3378 | }, 3379 | "engines": { 3380 | "node": ">=10.13.0" 3381 | } 3382 | }, 3383 | "node_modules/term-size": { 3384 | "version": "2.2.1", 3385 | "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", 3386 | "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", 3387 | "dev": true, 3388 | "engines": { 3389 | "node": ">=8" 3390 | }, 3391 | "funding": { 3392 | "url": "https://github.com/sponsors/sindresorhus" 3393 | } 3394 | }, 3395 | "node_modules/terser": { 3396 | "version": "5.13.1", 3397 | "resolved": "https://registry.npmjs.org/terser/-/terser-5.13.1.tgz", 3398 | "integrity": "sha512-hn4WKOfwnwbYfe48NgrQjqNOH9jzLqRcIfbYytOXCOv46LBfWr9bDS17MQqOi+BWGD0sJK3Sj5NC/gJjiojaoA==", 3399 | "dev": true, 3400 | "dependencies": { 3401 | "acorn": "^8.5.0", 3402 | "commander": "^2.20.0", 3403 | "source-map": "~0.8.0-beta.0", 3404 | "source-map-support": "~0.5.20" 3405 | }, 3406 | "bin": { 3407 | "terser": "bin/terser" 3408 | }, 3409 | "engines": { 3410 | "node": ">=10" 3411 | } 3412 | }, 3413 | "node_modules/terser/node_modules/commander": { 3414 | "version": "2.20.3", 3415 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 3416 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 3417 | "dev": true 3418 | }, 3419 | "node_modules/terser/node_modules/source-map": { 3420 | "version": "0.8.0-beta.0", 3421 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.8.0-beta.0.tgz", 3422 | "integrity": "sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==", 3423 | "dev": true, 3424 | "dependencies": { 3425 | "whatwg-url": "^7.0.0" 3426 | }, 3427 | "engines": { 3428 | "node": ">= 8" 3429 | } 3430 | }, 3431 | "node_modules/timsort": { 3432 | "version": "0.3.0", 3433 | "resolved": "https://registry.npmjs.org/timsort/-/timsort-0.3.0.tgz", 3434 | "integrity": "sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=", 3435 | "dev": true 3436 | }, 3437 | "node_modules/to-readable-stream": { 3438 | "version": "1.0.0", 3439 | "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", 3440 | "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", 3441 | "dev": true, 3442 | "engines": { 3443 | "node": ">=6" 3444 | } 3445 | }, 3446 | "node_modules/tr46": { 3447 | "version": "1.0.1", 3448 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-1.0.1.tgz", 3449 | "integrity": "sha1-qLE/1r/SSJUZZ0zN5VujaTtwbQk=", 3450 | "dev": true, 3451 | "dependencies": { 3452 | "punycode": "^2.1.0" 3453 | } 3454 | }, 3455 | "node_modules/tslib": { 3456 | "version": "2.4.0", 3457 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.4.0.tgz", 3458 | "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==", 3459 | "dev": true 3460 | }, 3461 | "node_modules/tunnel": { 3462 | "version": "0.0.6", 3463 | "resolved": "https://registry.npmjs.org/tunnel/-/tunnel-0.0.6.tgz", 3464 | "integrity": "sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg==", 3465 | "dev": true, 3466 | "optional": true, 3467 | "engines": { 3468 | "node": ">=0.6.11 <=0.7.0 || >=0.7.3" 3469 | } 3470 | }, 3471 | "node_modules/type-fest": { 3472 | "version": "0.13.1", 3473 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.13.1.tgz", 3474 | "integrity": "sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg==", 3475 | "dev": true, 3476 | "optional": true, 3477 | "engines": { 3478 | "node": ">=10" 3479 | }, 3480 | "funding": { 3481 | "url": "https://github.com/sponsors/sindresorhus" 3482 | } 3483 | }, 3484 | "node_modules/typedarray": { 3485 | "version": "0.0.6", 3486 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 3487 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", 3488 | "dev": true 3489 | }, 3490 | "node_modules/typescript": { 3491 | "version": "4.7.2", 3492 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.7.2.tgz", 3493 | "integrity": "sha512-Mamb1iX2FDUpcTRzltPxgWMKy3fhg0TN378ylbktPGPK/99KbDtMQ4W1hwgsbPAsG3a0xKa1vmw4VKZQbkvz5A==", 3494 | "dev": true, 3495 | "bin": { 3496 | "tsc": "bin/tsc", 3497 | "tsserver": "bin/tsserver" 3498 | }, 3499 | "engines": { 3500 | "node": ">=4.2.0" 3501 | } 3502 | }, 3503 | "node_modules/universalify": { 3504 | "version": "0.1.2", 3505 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 3506 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", 3507 | "dev": true, 3508 | "engines": { 3509 | "node": ">= 4.0.0" 3510 | } 3511 | }, 3512 | "node_modules/url-parse-lax": { 3513 | "version": "3.0.0", 3514 | "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", 3515 | "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", 3516 | "dev": true, 3517 | "dependencies": { 3518 | "prepend-http": "^2.0.0" 3519 | }, 3520 | "engines": { 3521 | "node": ">=4" 3522 | } 3523 | }, 3524 | "node_modules/util-deprecate": { 3525 | "version": "1.0.2", 3526 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 3527 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", 3528 | "dev": true 3529 | }, 3530 | "node_modules/utility-types": { 3531 | "version": "3.10.0", 3532 | "resolved": "https://registry.npmjs.org/utility-types/-/utility-types-3.10.0.tgz", 3533 | "integrity": "sha512-O11mqxmi7wMKCo6HKFt5AhO4BwY3VV68YU07tgxfz8zJTIxr4BpsezN49Ffwy9j3ZpwwJp4fkRwjRzq3uWE6Rg==", 3534 | "dev": true, 3535 | "engines": { 3536 | "node": ">= 4" 3537 | } 3538 | }, 3539 | "node_modules/v8-compile-cache": { 3540 | "version": "2.3.0", 3541 | "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", 3542 | "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==", 3543 | "dev": true 3544 | }, 3545 | "node_modules/weak-lru-cache": { 3546 | "version": "1.2.2", 3547 | "resolved": "https://registry.npmjs.org/weak-lru-cache/-/weak-lru-cache-1.2.2.tgz", 3548 | "integrity": "sha512-DEAoo25RfSYMuTGc9vPJzZcZullwIqRDSI9LOy+fkCJPi6hykCnfKaXTuPBDuXAUcqHXyOgFtHNp/kB2FjYHbw==", 3549 | "dev": true 3550 | }, 3551 | "node_modules/webidl-conversions": { 3552 | "version": "4.0.2", 3553 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-4.0.2.tgz", 3554 | "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==", 3555 | "dev": true 3556 | }, 3557 | "node_modules/whatwg-url": { 3558 | "version": "7.1.0", 3559 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-7.1.0.tgz", 3560 | "integrity": "sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==", 3561 | "dev": true, 3562 | "dependencies": { 3563 | "lodash.sortby": "^4.7.0", 3564 | "tr46": "^1.0.1", 3565 | "webidl-conversions": "^4.0.2" 3566 | } 3567 | }, 3568 | "node_modules/wrappy": { 3569 | "version": "1.0.2", 3570 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 3571 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 3572 | "dev": true 3573 | }, 3574 | "node_modules/xxhash-wasm": { 3575 | "version": "0.4.2", 3576 | "resolved": "https://registry.npmjs.org/xxhash-wasm/-/xxhash-wasm-0.4.2.tgz", 3577 | "integrity": "sha512-/eyHVRJQCirEkSZ1agRSCwriMhwlyUcFkXD5TPVSLP+IPzjsqMVzZwdoczLp1SoQU0R3dxz1RpIK+4YNQbCVOA==", 3578 | "dev": true 3579 | }, 3580 | "node_modules/yallist": { 3581 | "version": "4.0.0", 3582 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 3583 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 3584 | "dev": true, 3585 | "optional": true 3586 | }, 3587 | "node_modules/yaml": { 3588 | "version": "1.10.2", 3589 | "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", 3590 | "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", 3591 | "dev": true, 3592 | "engines": { 3593 | "node": ">= 6" 3594 | } 3595 | }, 3596 | "node_modules/yauzl": { 3597 | "version": "2.10.0", 3598 | "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.10.0.tgz", 3599 | "integrity": "sha1-x+sXyT4RLLEIb6bY5R+wZnt5pfk=", 3600 | "dev": true, 3601 | "dependencies": { 3602 | "buffer-crc32": "~0.2.3", 3603 | "fd-slicer": "~1.1.0" 3604 | } 3605 | } 3606 | } 3607 | } 3608 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "electron-tabs", 3 | "version": "1.0.4", 4 | "description": "Simple tabs for Electron applications", 5 | "main": "dist/electron-tabs.js", 6 | "types": "dist/electron-tabs.d.ts", 7 | "source": "src/index.ts", 8 | "targets": { 9 | "main": { 10 | "includeNodeModules": true 11 | } 12 | }, 13 | "repository": { 14 | "type": "git", 15 | "url": "https://github.com/brrd/electron-tabs" 16 | }, 17 | "scripts": { 18 | "demo": "electron ./demo", 19 | "watch": "parcel watch", 20 | "build": "parcel build" 21 | }, 22 | "keywords": [ 23 | "electron", 24 | "tab", 25 | "tabs", 26 | "webview" 27 | ], 28 | "author": "brrd", 29 | "license": "MIT", 30 | "devDependencies": { 31 | "@parcel/packager-ts": "^2.5.0", 32 | "@parcel/transformer-inline-string": "^2.5.0", 33 | "@parcel/transformer-typescript-types": "^2.5.0", 34 | "@types/sortablejs": "^1.13.0", 35 | "electron": "^17.1.2", 36 | "parcel": "^2.5.0", 37 | "postcss": "^8.4.14", 38 | "sortablejs": "^1.15.0", 39 | "typescript": "^4.7.2" 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import Sortable from "sortablejs"; 2 | // @ts-ignore 3 | import styles from "bundle-text:./style.css"; 4 | 5 | if (!document) { 6 | throw Error("electron-tabs module must be called in renderer process"); 7 | } 8 | 9 | interface TabGroupOptions { 10 | closeButtonText: string, 11 | defaultTab: TabOptions | ((tabGroup: TabGroup) => TabOptions), 12 | newTabButton: boolean, 13 | newTabButtonText: string, 14 | sortable: boolean, 15 | sortableOptions?: Sortable.Options 16 | visibilityThreshold: number, 17 | } 18 | 19 | interface TabOptions { 20 | active?: boolean; 21 | badge?: Badge; 22 | closable?: boolean; 23 | icon?: string; 24 | iconURL?: string; 25 | ready?: ((tab: Tab) => void); 26 | src?: string; 27 | title?: string; 28 | visible?: boolean; 29 | webviewAttributes?: { [key: string]: any }; 30 | } 31 | 32 | interface Badge { 33 | text: string, 34 | classname: string 35 | } 36 | 37 | const CLASSNAMES = { 38 | ROOT: "etabs", 39 | NAV: "nav", 40 | TABS: "tabs", 41 | TAB: "tab", 42 | BUTTONS: "buttons", 43 | VIEWS: "views", 44 | VIEW: "view" 45 | } 46 | 47 | function emit(emitter: TabGroup | Tab, type: string, args: any[]) { 48 | if (type === "ready") { 49 | emitter.isReady = true; 50 | } 51 | emitter.dispatchEvent(new CustomEvent(type, { detail: args })); 52 | } 53 | 54 | function on(emitter: TabGroup | Tab, type: string, fn: (detail: string) => void, options?: { [key: string]: any }) { 55 | if (type === "ready" && emitter.isReady === true) { 56 | fn.apply(emitter, [emitter]); 57 | } 58 | emitter.addEventListener(type, ((e: CustomEvent) => fn.apply(emitter, e.detail)) as EventListener, options); 59 | } 60 | 61 | class TabGroup extends HTMLElement { 62 | buttonContainer: HTMLDivElement; 63 | isReady: boolean; 64 | newTabId: number; 65 | options: TabGroupOptions; 66 | shadow: ShadowRoot; 67 | tabContainer: HTMLDivElement; 68 | tabs: Array; 69 | viewContainer: HTMLDivElement; 70 | 71 | constructor() { 72 | super(); 73 | 74 | this.isReady = false; 75 | 76 | // Options 77 | this.options = { 78 | closeButtonText: this.getAttribute("close-button-text") || "×", 79 | defaultTab: { title: "New Tab", active: true }, 80 | newTabButton: !!this.getAttribute("new-tab-button") === true || false, 81 | newTabButtonText: this.getAttribute("new-tab-button-text") || "+", 82 | sortable: !!this.getAttribute("sortable") === true || false, 83 | visibilityThreshold: Number(this.getAttribute("visibility-threshold")) || 0 84 | }; 85 | 86 | this.tabs = []; 87 | this.newTabId = 0; 88 | 89 | this.createComponent(); 90 | this.initVisibility(); 91 | if (this.options.sortable) { 92 | this.initSortable(); 93 | } 94 | 95 | this.emit("ready", this); 96 | } 97 | 98 | emit(type: string, ...args: any[]) { 99 | return emit(this, type, args); 100 | } 101 | 102 | on(type: string, fn: (...detail: any[]) => void) { 103 | return on(this, type, fn); 104 | } 105 | 106 | once(type: string, fn: (detail: string) => void) { 107 | return on(this, type, fn, { once: true }); 108 | } 109 | 110 | connectedCallback() { 111 | // Support custom styles 112 | const style = this.querySelector("style"); 113 | if (style) { 114 | this.shadow.appendChild(style); 115 | } 116 | } 117 | 118 | private createComponent() { 119 | const shadow = this.attachShadow({mode: "open"}); 120 | this.shadow = shadow; 121 | 122 | const wrapper = document.createElement("div"); 123 | wrapper.setAttribute("class", CLASSNAMES.ROOT); 124 | 125 | const tabgroup = document.createElement("nav"); 126 | tabgroup.setAttribute("class", CLASSNAMES.NAV); 127 | wrapper.appendChild(tabgroup); 128 | 129 | const tabContainer = document.createElement("div"); 130 | tabContainer.setAttribute("class", CLASSNAMES.TABS); 131 | tabgroup.appendChild(tabContainer); 132 | this.tabContainer = tabContainer; 133 | 134 | const buttonContainer = document.createElement("div"); 135 | buttonContainer.setAttribute("class", CLASSNAMES.BUTTONS); 136 | tabgroup.appendChild(buttonContainer); 137 | this.buttonContainer = buttonContainer; 138 | 139 | if (this.options.newTabButton) { 140 | const button = this.buttonContainer.appendChild(document.createElement("button")); 141 | button.innerHTML = this.options.newTabButtonText; 142 | button.addEventListener("click", this.addTab.bind(this, undefined), false); 143 | } 144 | 145 | const viewContainer = document.createElement("div"); 146 | viewContainer.setAttribute("class", CLASSNAMES.VIEWS); 147 | wrapper.appendChild(viewContainer); 148 | this.viewContainer = viewContainer; 149 | 150 | const style = document.createElement("style"); 151 | style.textContent = styles; 152 | 153 | shadow.appendChild(style); 154 | shadow.appendChild(wrapper); 155 | } 156 | 157 | private initVisibility() { 158 | function toggleTabsVisibility(tab: Tab, tabGroup: TabGroup) { 159 | const visibilityThreshold = tabGroup.options.visibilityThreshold; 160 | const el = tabGroup.tabContainer.parentElement; 161 | if (tabGroup.tabs.length >= visibilityThreshold) { 162 | el.classList.add("visible"); 163 | } else { 164 | el.classList.remove("visible"); 165 | } 166 | } 167 | 168 | this.on("tab-added", toggleTabsVisibility); 169 | this.on("tab-removed", toggleTabsVisibility); 170 | toggleTabsVisibility(null, this); 171 | } 172 | 173 | initSortable() { 174 | const createNewSortable = () => { 175 | const options = Object.assign({ 176 | direction: "horizontal", 177 | animation: 150, 178 | swapThreshold: 0.20 179 | }, this.options.sortableOptions); 180 | new Sortable(this.tabContainer, options); 181 | }; 182 | 183 | if (Sortable) { 184 | createNewSortable(); 185 | } else { 186 | document.addEventListener("DOMContentLoaded", createNewSortable); 187 | } 188 | } 189 | 190 | setDefaultTab(tab: TabOptions) { 191 | this.options.defaultTab = tab; 192 | } 193 | 194 | addTab(args = this.options.defaultTab) { 195 | if (typeof args === "function") { 196 | args = args(this); 197 | } 198 | const id = this.newTabId; 199 | this.newTabId++; 200 | const tab = new Tab(this, id, args); 201 | this.tabs.push(tab); 202 | // Don't call tab.activate() before a tab is referenced in this.tabs 203 | if (args.active === true) { 204 | tab.activate(); 205 | } 206 | this.emit("tab-added", tab, this); 207 | return tab; 208 | } 209 | 210 | getTab(id: number) { 211 | for (let i in this.tabs) { 212 | if (this.tabs[i].id === id) { 213 | return this.tabs[i]; 214 | } 215 | } 216 | return null; 217 | } 218 | 219 | getTabByPosition(position: number) { 220 | const fromRight = position < 0; 221 | for (let i in this.tabs) { 222 | if (this.tabs[i].getPosition(fromRight) === position) { 223 | return this.tabs[i]; 224 | } 225 | } 226 | return null; 227 | } 228 | 229 | getTabByRelPosition(position: number) { 230 | position = this.getActiveTab().getPosition() + position; 231 | if (position < 0) { 232 | return null; 233 | } 234 | return this.getTabByPosition(position); 235 | } 236 | 237 | getNextTab() { 238 | return this.getTabByRelPosition(1); 239 | } 240 | 241 | getPreviousTab() { 242 | return this.getTabByRelPosition(-1); 243 | } 244 | 245 | getTabs() { 246 | return this.tabs.slice(); 247 | } 248 | 249 | eachTab(fn: (tab: Tab) => void) { 250 | this.getTabs().forEach(fn); 251 | } 252 | 253 | getActiveTab() { 254 | if (this.tabs.length === 0) return null; 255 | return this.tabs[0]; 256 | } 257 | 258 | setActiveTab(tab: Tab) { 259 | this.removeTab(tab); 260 | this.tabs.unshift(tab); 261 | this.emit("tab-active", tab, this); 262 | } 263 | 264 | removeTab(tab: Tab, triggerEvent = false) { 265 | const id = tab.id; 266 | const index = this.tabs.findIndex((t: Tab) => t.id === id); 267 | this.tabs.splice(index, 1); 268 | if (triggerEvent) { 269 | this.emit("tab-removed", tab, this); 270 | } 271 | } 272 | 273 | activateRecentTab() { 274 | if (this.tabs.length > 0) { 275 | this.tabs[0].activate(); 276 | } 277 | } 278 | } 279 | 280 | class Tab extends EventTarget { 281 | badge: Badge; 282 | closable: boolean; 283 | element: HTMLDivElement; 284 | icon: string; 285 | iconURL: string; 286 | id: number; 287 | isClosed: boolean; 288 | isReady: boolean; 289 | spans: { [key: string]: HTMLSpanElement }; 290 | tabGroup: TabGroup; 291 | title: string; 292 | webview: HTMLElement; 293 | webviewAttributes: { [key: string]: any }; 294 | 295 | constructor(tabGroup: TabGroup, id: number, args: TabOptions) { 296 | super(); 297 | this.badge = args.badge; 298 | this.closable = args.closable === false ? false : true; 299 | this.icon = args.icon; 300 | this.iconURL = args.iconURL; 301 | this.id = id; 302 | this.isClosed = false; 303 | this.isReady = false; 304 | this.spans = {}; 305 | this.tabGroup = tabGroup; 306 | this.title = args.title; 307 | this.webviewAttributes = args.webviewAttributes || {}; 308 | this.webviewAttributes.src = args.src; 309 | 310 | this.initTab(); 311 | this.initWebview(); 312 | 313 | if (args.visible !== false) { 314 | this.show(); 315 | } 316 | if (typeof args.ready === "function") { 317 | args.ready(this); 318 | } else { 319 | this.emit("ready", this); 320 | } 321 | } 322 | 323 | emit(type: string, ...args: any[]) { 324 | return emit(this, type, args); 325 | } 326 | 327 | on(type: string, fn: (...detail: any[]) => void) { 328 | return on(this, type, fn); 329 | } 330 | 331 | once(type: string, fn: (detail: string) => void) { 332 | return on(this, type, fn, { once: true }); 333 | } 334 | 335 | private initTab() { 336 | const tab = this.element = document.createElement("div"); 337 | tab.classList.add(CLASSNAMES.TAB); 338 | for (let el of ["icon", "title", "badge", "close"]) { 339 | const span = tab.appendChild(document.createElement("span")); 340 | span.classList.add(`${CLASSNAMES.TAB}-${el}`); 341 | this.spans[el] = span; 342 | } 343 | 344 | this.setTitle(this.title); 345 | this.setBadge(this.badge); 346 | this.setIcon(this.iconURL, this.icon); 347 | this.initTabCloseButton(); 348 | this.initTabClickHandler(); 349 | 350 | this.tabGroup.tabContainer.appendChild(this.element); 351 | } 352 | 353 | private initTabCloseButton() { 354 | const container = this.spans.close; 355 | if (this.closable) { 356 | const button = container.appendChild(document.createElement("button")); 357 | button.innerHTML = this.tabGroup.options.closeButtonText; 358 | button.addEventListener("click", this.close.bind(this, false), false); 359 | } 360 | } 361 | 362 | private initTabClickHandler() { 363 | // Mouse up 364 | const tabClickHandler = function(e: KeyboardEvent) { 365 | if (this.isClosed) return; 366 | if (e.which === 2) { 367 | this.close(); 368 | } 369 | }; 370 | this.element.addEventListener("mouseup", tabClickHandler.bind(this), false); 371 | // Mouse down 372 | const tabMouseDownHandler = function(e: KeyboardEvent) { 373 | if (this.isClosed) return; 374 | if (e.which === 1) { 375 | if ((e.target as HTMLElement).matches("button")) return; 376 | this.activate(); 377 | } 378 | }; 379 | this.element.addEventListener("mousedown", tabMouseDownHandler.bind(this), false); 380 | } 381 | 382 | initWebview() { 383 | const webview = this.webview = document.createElement("webview"); 384 | 385 | const tabWebviewDidFinishLoadHandler = function(e: Event) { 386 | this.emit("webview-ready", this); 387 | }; 388 | 389 | this.webview.addEventListener("did-finish-load", tabWebviewDidFinishLoadHandler.bind(this), false); 390 | 391 | const tabWebviewDomReadyHandler = function(e: Event) { 392 | // Remove this once https://github.com/electron/electron/issues/14474 is fixed 393 | webview.blur(); 394 | webview.focus(); 395 | this.emit("webview-dom-ready", this); 396 | }; 397 | 398 | this.webview.addEventListener("dom-ready", tabWebviewDomReadyHandler.bind(this), false); 399 | 400 | this.webview.classList.add(CLASSNAMES.VIEW); 401 | if (this.webviewAttributes) { 402 | const attrs = this.webviewAttributes; 403 | for (let key in attrs) { 404 | const attr = attrs[key]; 405 | if (attr === false) continue; 406 | this.webview.setAttribute(key, attr); 407 | } 408 | } 409 | 410 | this.tabGroup.viewContainer.appendChild(this.webview); 411 | } 412 | 413 | setTitle(title: string) { 414 | if (this.isClosed) return; 415 | const span = this.spans.title; 416 | span.innerHTML = title; 417 | span.title = title; 418 | this.title = title; 419 | this.emit("title-changed", title, this); 420 | return this; 421 | } 422 | 423 | getTitle() { 424 | if (this.isClosed) return; 425 | return this.title; 426 | } 427 | 428 | setBadge(badge?: Badge) { 429 | if (this.isClosed) return; 430 | const span = this.spans.badge; 431 | this.badge = badge; 432 | 433 | if (badge) { 434 | span.innerHTML = badge.text; 435 | span.classList.add(badge.classname); 436 | span.classList.remove("hidden"); 437 | } else { 438 | span.classList.add("hidden"); 439 | } 440 | 441 | this.emit("badge-changed", badge, this); 442 | } 443 | 444 | getBadge() { 445 | if (this.isClosed) return; 446 | return this.badge; 447 | } 448 | 449 | setIcon(iconURL: string, icon: string) { 450 | if (this.isClosed) return; 451 | this.iconURL = iconURL; 452 | this.icon = icon; 453 | const span = this.spans.icon; 454 | if (iconURL) { 455 | span.innerHTML = ``; 456 | this.emit("icon-changed", iconURL, this); 457 | } else if (icon) { 458 | span.innerHTML = ``; 459 | this.emit("icon-changed", icon, this); 460 | } 461 | 462 | return this; 463 | } 464 | 465 | getIcon() { 466 | if (this.isClosed) return; 467 | if (this.iconURL) return this.iconURL; 468 | return this.icon; 469 | } 470 | 471 | setPosition(newPosition: number) { 472 | const tabContainer = this.tabGroup.tabContainer; 473 | const length = tabContainer.childElementCount; 474 | const thisPosition = this.getPosition(); 475 | const tabs = Array.from(tabContainer.children) 476 | tabs.splice(thisPosition, 1); 477 | 478 | if (newPosition < 0) { 479 | newPosition += length; 480 | if (newPosition < 0) { 481 | newPosition = 0; 482 | } 483 | } 484 | 485 | if (newPosition < length) { 486 | tabContainer.insertBefore(this.element, tabs[newPosition]); 487 | } else { 488 | tabContainer.appendChild(this.element); 489 | } 490 | 491 | return this; 492 | } 493 | 494 | getPosition(fromRight = false) { 495 | let position = 0; 496 | let tab = this.element; 497 | while ((tab = tab.previousSibling as HTMLDivElement) != null) position++; 498 | 499 | if (fromRight === true) { 500 | position -= this.tabGroup.tabContainer.childElementCount; 501 | } 502 | 503 | return position; 504 | } 505 | 506 | activate() { 507 | if (this.isClosed) return; 508 | const activeTab = this.tabGroup.getActiveTab(); 509 | if (activeTab) { 510 | activeTab.element.classList.remove("active"); 511 | activeTab.webview.classList.remove("visible"); 512 | activeTab.emit("inactive", activeTab); 513 | } 514 | this.tabGroup.setActiveTab(this); 515 | this.element.classList.add("active"); 516 | this.webview.classList.add("visible"); 517 | this.webview.focus(); 518 | this.emit("active", this); 519 | return this; 520 | } 521 | 522 | show(flag = true) { 523 | if (this.isClosed) return; 524 | if (flag) { 525 | this.element.classList.add("visible"); 526 | this.emit("visible", this); 527 | } else { 528 | this.element.classList.remove("visible"); 529 | this.emit("hidden", this); 530 | } 531 | return this; 532 | } 533 | 534 | hide() { 535 | return this.show(false); 536 | } 537 | 538 | hasClass(classname: string) { 539 | return this.element.classList.contains(classname); 540 | } 541 | 542 | close(force: boolean) { 543 | const abortController = new AbortController(); 544 | const abort = () => abortController.abort(); 545 | this.emit("closing", this, abort); 546 | 547 | const abortSignal = abortController.signal; 548 | if (this.isClosed || (!this.closable && !force) || abortSignal.aborted) return; 549 | 550 | this.isClosed = true; 551 | const tabGroup = this.tabGroup; 552 | tabGroup.tabContainer.removeChild(this.element); 553 | tabGroup.viewContainer.removeChild(this.webview); 554 | const activeTab = this.tabGroup.getActiveTab(); 555 | tabGroup.removeTab(this, true); 556 | 557 | this.emit("close", this); 558 | 559 | if (activeTab.id === this.id) { 560 | tabGroup.activateRecentTab(); 561 | } 562 | } 563 | } 564 | 565 | customElements.define("tab-group", TabGroup); 566 | 567 | export type { TabGroup, Tab }; 568 | -------------------------------------------------------------------------------- /src/style.css: -------------------------------------------------------------------------------- 1 | /* CUSTOM PROPERTIES */ 2 | /* ================= */ 3 | 4 | :host { 5 | --tabgroup-background: #E7EAED; 6 | --tab-font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol"; 7 | --tab-font-size: 13px; 8 | --tab-background: #E7EAED; 9 | --tab-color: #696A6C; 10 | --tab-border-color: #DADCE0; 11 | --tab-transition: background-color 200ms ease-out, color 200ms ease-out; 12 | --tab-cursor: pointer; 13 | --tab-active-color: currentcolor; 14 | --tab-active-background: #FFF; 15 | --tag-hover-color: currentcolor; 16 | --tag-hover-background: #F1F3F4; 17 | --button-font-size: 15px; 18 | --button-background: none; 19 | --button-color: #696A6C; 20 | --button-hover-background: #DADCE0; 21 | --button-hover-color: #383a3e; 22 | --button-border-radius: 50%; 23 | --button-cursor: pointer; 24 | --badge-background: #383a3e; 25 | --badge-color: #fff; 26 | --close-button-visibility: visible; 27 | } 28 | 29 | /* STYLES */ 30 | /* ====== */ 31 | 32 | webview { 33 | position: absolute; 34 | visibility: hidden; 35 | width: 100%; 36 | height: 100%; 37 | } 38 | 39 | webview.visible { 40 | visibility: visible; 41 | } 42 | 43 | .etabs { 44 | font-family: var(--tab-font-family); 45 | text-rendering: optimizeLegibility; 46 | font-feature-settings: 'liga', 'clig', 'kern'; 47 | } 48 | 49 | .nav { 50 | background: var(--tabgroup-background); 51 | box-shadow: inset 0 -1px var(--tab-border-color); 52 | border-top: 1px solid var(--tab-border-color); 53 | font-size: var(--tab-font-size); 54 | display: none; 55 | width: 100%; 56 | height: 32px; 57 | cursor: default; 58 | -webkit-user-select: none; 59 | user-select: none; 60 | } 61 | 62 | .nav.visible { 63 | display: flex; 64 | } 65 | 66 | .tabs { 67 | height: 100%; 68 | } 69 | 70 | .tab { 71 | background: var(--tab-background); 72 | box-shadow: inset 0 -1px var(--tab-border-color); 73 | color: var(--tab-color); 74 | cursor: var(--tab-cursor); 75 | font-size: var(--tab-font-size); 76 | transition: var(--tab-transition); 77 | display: none; 78 | position: relative; 79 | box-sizing: border-box; 80 | height: 100%; 81 | padding: 5px 9px; 82 | align-items: center; 83 | } 84 | 85 | .tab:first-child { 86 | border-left: none; 87 | } 88 | 89 | .tab.visible { 90 | display: inline-flex; 91 | } 92 | 93 | .tab.active { 94 | color: var(--tab-active-color); 95 | background: var(--tab-active-background); 96 | border-left: 1px solid var(--tab-border-color); 97 | border-right: 1px solid var(--tab-border-color); 98 | padding-left: 8px; 99 | padding-right: 8px; 100 | box-shadow: none; 101 | } 102 | 103 | .tab.active:last-child { 104 | border-right: none; 105 | } 106 | 107 | .tab.visible:not(.active)+.tab.visible:not(.active) { 108 | border-left: 1px solid var(--tab-border-color); 109 | padding-left: 8px; 110 | } 111 | 112 | .tab:not(.active):hover { 113 | background: var(--tab-hover-background); 114 | color: var(--tab-hover-color); 115 | } 116 | 117 | .tab-badge { 118 | background: var(--badge-background); 119 | color: var(--badge-color); 120 | text-align: center; 121 | border-radius: 5px; 122 | margin-left: 5px; 123 | padding: 1px 4px; 124 | font-size: 8px; 125 | font-weight: bold; 126 | line-height: 1.2; 127 | } 128 | 129 | .tab-badge.hidden { 130 | display: none; 131 | } 132 | 133 | .tab-icon { 134 | display: inline-block; 135 | height: 16px; 136 | } 137 | 138 | .tab-icon img { 139 | max-width: 16px; 140 | max-height: 16px; 141 | } 142 | 143 | .tab-title { 144 | display: inline-block; 145 | margin-left: 10px; 146 | } 147 | 148 | .tab-close { 149 | display: inline-block; 150 | margin-left: 10px; 151 | } 152 | 153 | .tab-close button { 154 | background: var(--button-background); 155 | border-radius: var(--button-border-radius); 156 | color: var(--button-color); 157 | cursor: var(--button-cursor); 158 | font-size: var(--button-font-size); 159 | display: inline-block; 160 | border: none; 161 | width: 20px; 162 | height: 20px; 163 | text-align: center; 164 | padding: 1px 0 0 0; 165 | visibility: var(--close-button-visibility); 166 | } 167 | 168 | .tab.active .tab-close button { 169 | visibility: visible; 170 | } 171 | 172 | .tab-close button:hover { 173 | color: var(--button-hover-color); 174 | background: var(--button-hover-background); 175 | } 176 | 177 | .buttons { 178 | display: flex; 179 | padding: 5px; 180 | border-left: 1px solid var(--tab-border-color); 181 | } 182 | 183 | .buttons button { 184 | color: var(--button-color); 185 | background: var(--button-background); 186 | border-radius: var(--button-border-radius); 187 | cursor: var(--button-cursor); 188 | font-size: var(--button-font-size); 189 | display: block; 190 | font-family: inherit; 191 | border: none; 192 | margin: 0; 193 | width: 20px; 194 | height: 20px; 195 | text-align: center; 196 | line-height: 1; 197 | padding: 1px 0 0 0; 198 | } 199 | 200 | .buttons button:hover { 201 | color: var(--button-hover-color); 202 | background: var(--button-hover-background); 203 | } 204 | 205 | .views { 206 | position: relative; 207 | height: calc(100vh - 33px); 208 | } 209 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es2021", 4 | "noImplicitAny": true, 5 | "noUnusedParameters": false, 6 | "allowSyntheticDefaultImports": true 7 | } 8 | } 9 | --------------------------------------------------------------------------------