├── codecov.yml ├── src ├── index.ts └── SignaturePad.tsx ├── .husky └── pre-commit ├── tsconfig.json ├── .gitignore ├── tsconfig.test.json ├── __tests__ ├── .eslintrc.json ├── helpers │ └── signature.ts └── SignaturePad.test.tsx ├── .editorconfig ├── example ├── app.tsx ├── tsconfig.json ├── index.html ├── webpack.config.mjs ├── package.json ├── Layout.tsx └── package-lock.json ├── prettier.config.mjs ├── jest.config.mjs ├── sonar-project.properties ├── tsconfig.base.json ├── LICENSE ├── eslint.config.mjs ├── rollup.config.ts ├── .github └── workflows │ └── ci.yml ├── package.json └── README.md /codecov.yml: -------------------------------------------------------------------------------- 1 | ignore: 2 | - '__tests__/**/*' 3 | - 'example/**/*' 4 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import SignaturePad from './SignaturePad'; 2 | 3 | export default SignaturePad; 4 | -------------------------------------------------------------------------------- /.husky/pre-commit: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | . "$(dirname "$0")/_/husky.sh" 3 | 4 | npx --no-install lint-staged 5 | npm test 6 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "./tsconfig.base", 3 | "compilerOptions": { 4 | "esModuleInterop": false 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | /.idea 2 | /node_modules 3 | /coverage 4 | /dist 5 | /example/dist 6 | /example/node_modules 7 | .DS_Store 8 | .eslintcache 9 | npm-debug.log 10 | -------------------------------------------------------------------------------- /tsconfig.test.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "./tsconfig.base", 3 | "compilerOptions": { 4 | "verbatimModuleSyntax": false 5 | }, 6 | "include": ["__tests__/**/*"] 7 | } 8 | -------------------------------------------------------------------------------- /__tests__/.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "node": true, 4 | "jest/globals": true 5 | }, 6 | "plugins": ["jest"], 7 | "extends": ["plugin:jest/recommended", "plugin:jest-dom/recommended"] 8 | } 9 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | end_of_line = lf 5 | insert_final_newline = true 6 | 7 | [*.{html,js,ts}] 8 | charset = utf-8 9 | indent_style = space 10 | indent_size = 4 11 | trim_trailing_whitespace = true 12 | 13 | [*.md] 14 | trim_trailing_whitespace = false 15 | 16 | [*.{json,yml,yaml}] 17 | indent_size = 2 18 | -------------------------------------------------------------------------------- /example/app.tsx: -------------------------------------------------------------------------------- 1 | import { StrictMode } from 'react'; 2 | import { createRoot } from 'react-dom/client'; 3 | 4 | import Layout from './Layout'; 5 | 6 | const rootElement = document.getElementById('root'); 7 | 8 | if (!rootElement) { 9 | throw new Error('Failed to find the root element'); 10 | } 11 | 12 | createRoot(rootElement).render( 13 | 14 | 15 | , 16 | ); 17 | -------------------------------------------------------------------------------- /prettier.config.mjs: -------------------------------------------------------------------------------- 1 | /** @type {import('prettier').Config} */ 2 | export default { 3 | semi: true, 4 | trailingComma: 'all', 5 | singleQuote: true, 6 | printWidth: 120, 7 | tabWidth: 4, 8 | endOfLine: 'auto', 9 | overrides: [ 10 | { 11 | files: ['*.json', '*.yml', '*.yaml'], 12 | options: { 13 | tabWidth: 2, 14 | }, 15 | }, 16 | ], 17 | }; 18 | -------------------------------------------------------------------------------- /example/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "moduleResolution": "node", 4 | "target": "es5", 5 | "module": "esnext", 6 | "lib": ["es5", "es6", "es7", "dom"], 7 | "removeComments": true, 8 | "strict": true, 9 | "sourceMap": true, 10 | "declaration": true, 11 | "declarationDir": "./dist", 12 | "outDir": "./dist", 13 | "jsx": "react-jsx" 14 | }, 15 | "include": ["**/*"], 16 | "exclude": ["node_modules", "dist", "example"] 17 | } 18 | -------------------------------------------------------------------------------- /jest.config.mjs: -------------------------------------------------------------------------------- 1 | /** @type {import('jest').Config} */ 2 | export default { 3 | testEnvironment: 'jsdom', 4 | testMatch: ['**/__tests__/**/*.[jt]s?(x)'], 5 | testPathIgnorePatterns: ['/__tests__/shim.ts', '/__tests__/helpers/', '/node_modules/'], 6 | transform: { 7 | '\\.[jt]sx?$': [ 8 | 'ts-jest', 9 | { 10 | tsconfig: 'tsconfig.test.json', 11 | }, 12 | ], 13 | }, 14 | collectCoverage: true, 15 | }; 16 | -------------------------------------------------------------------------------- /sonar-project.properties: -------------------------------------------------------------------------------- 1 | # must be unique in a given SonarQube instance 2 | sonar.projectKey=michaeldzjap_react-signature-pad-wrapper 3 | 4 | # --- optional properties --- 5 | 6 | sonar.organization=michaeldzjap-github 7 | sonar.javascript.lcov.reportPaths=coverage/lcov.info 8 | sonar.exclusions=.prettierrc.js, jest.config.js, rollup.config.ts, __tests__/**, example/** 9 | 10 | # defaults to project key 11 | #sonar.projectName=My project 12 | # defaults to 'not provided' 13 | #sonar.projectVersion=1.0 14 | 15 | # Path is relative to the sonar-project.properties file. Defaults to . 16 | #sonar.sources=. 17 | 18 | # Encoding of the source code. Default is default system encoding 19 | #sonar.sourceEncoding=UTF-8 20 | -------------------------------------------------------------------------------- /tsconfig.base.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "module": "ESNext", 4 | "moduleResolution": "Bundler", 5 | "target": "ES6", 6 | "noEmit": true, 7 | "removeComments": true, 8 | "resolveJsonModule": true, 9 | "esModuleInterop": true, 10 | "verbatimModuleSyntax": true, 11 | "useDefineForClassFields": true, 12 | "forceConsistentCasingInFileNames": true, 13 | "skipLibCheck": true, 14 | "strict": true, 15 | "declaration": true, 16 | "declarationDir": "./dist", 17 | "rootDir": "./src", 18 | "outDir": "./dist", 19 | "jsx": "react" 20 | }, 21 | "include": ["src/**/*", "rollup.config.ts"], 22 | "exclude": ["node_modules", "dist", "example"] 23 | } 24 | -------------------------------------------------------------------------------- /example/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | React-Signature-Pad Example 9 | 10 | 11 | 12 | 17 | 18 | 19 |
20 | 21 | 22 | 23 | 24 | 25 | 26 | -------------------------------------------------------------------------------- /example/webpack.config.mjs: -------------------------------------------------------------------------------- 1 | import { dirname, resolve } from 'path'; 2 | import { fileURLToPath } from 'url'; 3 | 4 | const __filename = fileURLToPath(import.meta.url); 5 | const __dirname = dirname(__filename); 6 | 7 | export default { 8 | mode: 'development', 9 | entry: { 10 | vendor: ['react', 'react-dom'], 11 | bundle: { 12 | import: './app', 13 | dependOn: 'vendor', 14 | }, 15 | }, 16 | output: { 17 | filename: '[name].js', 18 | path: resolve(__dirname, 'dist'), 19 | devtoolModuleFilenameTemplate: 'webpack:///[absolute-resource-path]', 20 | }, 21 | module: { 22 | rules: [ 23 | { 24 | test: /\.tsx?$/, 25 | use: 'ts-loader', 26 | exclude: /node_modules/, 27 | }, 28 | ], 29 | }, 30 | resolve: { 31 | extensions: ['.js', '.ts', '.tsx'], 32 | }, 33 | devtool: 'eval-cheap-module-source-map', 34 | }; 35 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2025 Michael Dzjaparidze 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 | -------------------------------------------------------------------------------- /eslint.config.mjs: -------------------------------------------------------------------------------- 1 | // @ts-check 2 | 3 | import eslint from '@eslint/js'; 4 | import prettierConfig from 'eslint-config-prettier'; 5 | import jest from 'eslint-plugin-jest'; 6 | import jestDom from 'eslint-plugin-jest-dom'; 7 | import react from 'eslint-plugin-react'; 8 | import globals from 'globals'; 9 | import tseslint from 'typescript-eslint'; 10 | 11 | export default [ 12 | { 13 | ignores: ['coverage/**', 'dist/**', 'example/dist/**', '__tests__/**'], 14 | }, 15 | { 16 | plugins: { 17 | react, 18 | }, 19 | languageOptions: { 20 | parserOptions: { 21 | ecmaFeatures: { 22 | jsx: true, 23 | }, 24 | }, 25 | globals: { 26 | ...globals.browser, 27 | }, 28 | }, 29 | }, 30 | ...tseslint.config(eslint.configs.recommended, ...tseslint.configs.recommended, prettierConfig), 31 | ...tseslint.config({ 32 | files: ['__tests__/**'], 33 | plugins: { jest }, 34 | ...jest.configs['flat/recommended'], 35 | ...jestDom.configs['flat/recommended'], 36 | }), 37 | ]; 38 | -------------------------------------------------------------------------------- /example/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "example", 3 | "version": "1.0.0", 4 | "description": "A simple example of react-signature-pad-wrapper", 5 | "main": "dist/app.js", 6 | "jsnext:main": "app.js", 7 | "scripts": { 8 | "build": "./node_modules/.bin/webpack --progress", 9 | "watch": "npm run build -- --watch" 10 | }, 11 | "repository": { 12 | "type": "git", 13 | "url": "git+https://github.com/michaeldzjap/react-signature-pad-wrapper.git" 14 | }, 15 | "keywords": [ 16 | "react", 17 | "signature_pad" 18 | ], 19 | "author": "Michael Dzjaparidze", 20 | "license": "MIT", 21 | "bugs": { 22 | "url": "https://github.com/michaeldzjap/react-signature-pad-wrapper/issues" 23 | }, 24 | "homepage": "https://github.com/michaeldzjap/react-signature-pad-wrapper#readme", 25 | "dependencies": { 26 | "bulma": "^1.0.0", 27 | "react": "^19.0.0", 28 | "react-dom": "^19.0.0" 29 | }, 30 | "devDependencies": { 31 | "@types/react": "^19.0.0", 32 | "@types/react-dom": "^19.0.0", 33 | "ts-loader": "^9.2.6", 34 | "typescript": "^5.0.2", 35 | "webpack": "^5.53.0", 36 | "webpack-cli": "^6.0.1" 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /rollup.config.ts: -------------------------------------------------------------------------------- 1 | import { readFileSync } from 'fs'; 2 | 3 | import commonjs from '@rollup/plugin-commonjs'; 4 | import { nodeResolve } from '@rollup/plugin-node-resolve'; 5 | import typescript from '@rollup/plugin-typescript'; 6 | import terser from '@rollup/plugin-terser'; 7 | 8 | const pkg = JSON.parse(readFileSync('./package.json', 'utf8')); 9 | const local = process.env.NODE_ENV === 'local'; 10 | 11 | export default { 12 | input: 'src/index.ts', 13 | output: [ 14 | { 15 | dir: './', 16 | entryFileNames: pkg.main, 17 | format: 'cjs', 18 | sourcemap: local, 19 | exports: 'named', 20 | }, 21 | { 22 | dir: './', 23 | entryFileNames: pkg.module, 24 | format: 'es', 25 | sourcemap: local, 26 | }, 27 | ], 28 | external: [...Object.keys(pkg.dependencies || {}), ...Object.keys(pkg.peerDependencies || {})], 29 | watch: { 30 | include: 'src/**', 31 | }, 32 | plugins: [ 33 | typescript({ 34 | sourceMap: local, 35 | }), 36 | commonjs(), 37 | nodeResolve(), 38 | terser(), 39 | ], 40 | }; 41 | -------------------------------------------------------------------------------- /.github/workflows/ci.yml: -------------------------------------------------------------------------------- 1 | name: CI 2 | on: [push] 3 | jobs: 4 | ci: 5 | name: CI - OS ${{ matrix.os }}, Node.js ${{ matrix.node-version }} 6 | runs-on: ${{ matrix.os }} 7 | strategy: 8 | matrix: 9 | os: [ubuntu-latest] 10 | node-version: [22.x, 23.x, 24.x] 11 | steps: 12 | - name: Check out repository code 13 | uses: actions/checkout@v5 14 | - name: Use Node.js ${{ matrix.node-version }} 15 | uses: actions/setup-node@v5 16 | with: 17 | node-version: ${{ matrix.node-version }} 18 | cache: 'npm' 19 | - name: Install 20 | run: npm ci 21 | - name: Lint 22 | run: npm run lint 23 | - name: Build 24 | run: npm run prod 25 | - name: Test 26 | run: npm run test 27 | - name: Upload coverage report 28 | uses: codecov/codecov-action@v5 29 | if: ${{ matrix.os == 'ubuntu-latest' && matrix.node-version == '24.x' }} 30 | - name: SonarQube Scan 31 | uses: SonarSource/sonarqube-scan-action@v6 32 | env: 33 | SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} 34 | if: ${{ matrix.os == 'ubuntu-latest' && matrix.node-version == '24.x' }} 35 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "react-signature-pad-wrapper", 3 | "version": "4.3.2", 4 | "description": "A React component wrapper for signature_pad", 5 | "main": "dist/index.js", 6 | "module": "dist/index.es.js", 7 | "files": [ 8 | "dist" 9 | ], 10 | "types": "dist/index.d.ts", 11 | "scripts": { 12 | "test": "cross-env NODE_ENV=test npx jest", 13 | "lint": "npx eslint .", 14 | "format": "npx prettier . --write", 15 | "development": "cross-env NODE_ENV=local npx rollup --config rollup.config.ts --configPlugin @rollup/plugin-typescript", 16 | "dev": "npm run development", 17 | "production": "cross-env NODE_ENV=production npx rollup --config rollup.config.ts --configPlugin @rollup/plugin-typescript", 18 | "prod": "npm run production", 19 | "watch": "cross-env NODE_ENV=local npx rollup --config rollup.config.ts --configPlugin @rollup/plugin-typescript --watch", 20 | "clean": "rm -rf dist" 21 | }, 22 | "keywords": [ 23 | "react", 24 | "signature_pad" 25 | ], 26 | "author": "Michael Dzjaparidze", 27 | "license": "MIT", 28 | "repository": { 29 | "type": "git", 30 | "url": "git+https://github.com/michaeldzjap/react-signature-pad-wrapper.git" 31 | }, 32 | "bugs": { 33 | "url": "https://github.com/michaeldzjap/react-signature-pad-wrapper/issues" 34 | }, 35 | "peerDependencies": { 36 | "react": "17 - 19", 37 | "react-dom": "17 - 19" 38 | }, 39 | "devDependencies": { 40 | "@eslint/js": "^9.9.1", 41 | "@rollup/plugin-commonjs": "^29.0.0", 42 | "@rollup/plugin-node-resolve": "^16.0.1", 43 | "@rollup/plugin-terser": "^0.4.0", 44 | "@rollup/plugin-typescript": "^12.1.1", 45 | "@testing-library/jest-dom": "^6.1.2", 46 | "@testing-library/react": "^16.0.0", 47 | "@types/eslint__js": "^8.42.3", 48 | "@types/jest": "^30.0.0", 49 | "@types/prop-types": "^15.7.14", 50 | "@types/react": "^19.0.0", 51 | "@types/react-dom": "^19.0.0", 52 | "@types/throttle-debounce": "^5.0.0", 53 | "canvas": "^3.1.2", 54 | "cross-env": "^10.0.0", 55 | "eslint": "^9.9.1", 56 | "eslint-config-prettier": "^10.1.5", 57 | "eslint-plugin-jest": "^29.0.1", 58 | "eslint-plugin-jest-dom": "^5.0.1", 59 | "eslint-plugin-prettier": "^5.0.0", 60 | "eslint-plugin-react": "^7.20.6", 61 | "husky": "^9.0.11", 62 | "jest": "^30.0.3", 63 | "jest-environment-jsdom": "^30.0.2", 64 | "lint-staged": "^16.1.2", 65 | "prettier": "^3.0.0", 66 | "react": "^19.0.0", 67 | "react-dom": "^19.0.0", 68 | "rollup": "^4.5.0", 69 | "ts-jest": "^29.0.0", 70 | "tslib": "^2.8.1", 71 | "typescript": "^5.0.2", 72 | "typescript-eslint": "^8.4.0" 73 | }, 74 | "dependencies": { 75 | "prop-types": "^15.8.1", 76 | "signature_pad": "^5.1.0", 77 | "throttle-debounce": "^5.0.0" 78 | }, 79 | "lint-staged": { 80 | "*.ts?(x)": [ 81 | "eslint --cache --fix", 82 | "prettier --write --ignore-unknown" 83 | ] 84 | } 85 | } 86 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ![NPM Version](https://img.shields.io/npm/v/react-signature-pad-wrapper.svg?branch=master) 2 | ![downloads](https://img.shields.io/npm/dt/react-signature-pad-wrapper.svg) 3 | ![GitHub Workflow Status](https://img.shields.io/github/actions/workflow/status/michaeldzjap/react-signature-pad-wrapper/ci.yml?branch=master) 4 | ![Libraries.io dependency status for GitHub repo](https://img.shields.io/librariesio/github/michaeldzjap/react-signature-pad-wrapper) 5 | [![tested with jest](https://img.shields.io/badge/tested_with-jest-99424f.svg)](https://github.com/facebook/jest) 6 | [![codecov](https://codecov.io/gh/michaeldzjap/react-signature-pad-wrapper/branch/master/graph/badge.svg)](https://codecov.io/gh/michaeldzjap/react-signature-pad-wrapper) 7 | [![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=michaeldzjap_react-signature-pad-wrapper&metric=alert_status)](https://sonarcloud.io/dashboard?id=michaeldzjap_react-signature-pad-wrapper) 8 | [![License](https://img.shields.io/npm/l/react-signature-pad-wrapper.svg)](https://github.com/michaeldzjap/react-signature-pad-wrapper/blob/master/LICENSE) 9 | 10 | # react-signature-pad-wrapper 11 | 12 | A React wrapper for [signature pad](https://github.com/szimek/signature_pad). 13 | 14 | ## Installation 15 | 16 | This package is available through npm: 17 | 18 | ``` 19 | npm install --save react-signature-pad-wrapper 20 | ``` 21 | 22 | ## Usage 23 | 24 | This package implements exactly the same interface as the original _signature_pad_ package and adds a couple of extra features that make responsive behaviour a little easier to deal with. For a complete overview of the available options and callables see the documentation for [signature pad](https://github.com/szimek/signature_pad). 25 | 26 | Import the component like (ES6): 27 | 28 | ```javascript 29 | import SignaturePad from 'react-signature-pad-wrapper'; 30 | ``` 31 | 32 | Options may be passed as a component property during initialization: 33 | 34 | ```javascript 35 | ... 36 | render() { 37 | return ; 38 | } 39 | ... 40 | ``` 41 | 42 | or they can be set during runtime: 43 | 44 | ```javascript 45 | ... 46 | render() { 47 | return ; 48 | } 49 | ... 50 | ``` 51 | 52 | then from somewhere else in the code (assuming the ref is defined): 53 | 54 | ```javascript 55 | // Call an instance method 56 | this.signaturePad.clear(); 57 | this.signaturePad.isEmpty(); 58 | 59 | // Get/set an object property 60 | this.signaturePad.minWidth = 5; 61 | this.signaturePad.maxWidth = 10; 62 | this.signaturePad.penColor = 'rgb(66, 133, 244)'; 63 | ``` 64 | 65 | ## Responsiveness 66 | 67 | The HTML canvas object sucks when it comes to responsiveness. The approach taken with this plugin is to use a fixed size canvas when a height and width (in pixels) is explicitly passed in as a component property: 68 | 69 | ```javascript 70 | ... 71 | render() { 72 | return ; 73 | } 74 | ... 75 | ``` 76 | 77 | If you want the component to be responsive, simply ommit the width and height property: 78 | 79 | ```javascript 80 | ... 81 | render() { 82 | return ; 83 | } 84 | ... 85 | ``` 86 | 87 | The canvas width and height will now be updated whenever the window is resized (using a debounced handler). Changing the width and height properties of a HTML canvas object will erase its current content. 88 | 89 | If you'd like to keep what is currently drawn on the canvas you can pass a `redrawOnResize` property to the component and set it to `true` (`redrawOnResize` is `false` by default): 90 | 91 | ```javascript 92 | ... 93 | render() { 94 | return ; 95 | } 96 | ... 97 | ``` 98 | 99 | This will save the current canvas content to a base64 data string before performing the resize operation and load it in the canvas right after the resize operation finishes. **Note**: the repeated saving and loading of image data when resizing often will degrade the quality rapidly. There is no easy solution around this unfortunately. Resampling the image data is imagined to help significantly, but this is a rather costly operation in general and not something you would ideally do with JavaScript in the browser on every resize event (even if throttled/debounced). 100 | 101 | ## Example 102 | 103 | This project includes a simple example that demonstrates a responsive sketch pad. To build the example: 104 | 105 | ```shell 106 | npm install 107 | cd example && npm install && npm run build 108 | ``` 109 | 110 | Then open `example/index.html` in a browser of your choice. 111 | -------------------------------------------------------------------------------- /example/Layout.tsx: -------------------------------------------------------------------------------- 1 | import { useRef, useCallback, useState, useEffect } from 'react'; 2 | 3 | import SignaturePad from '../src/SignaturePad'; 4 | 5 | /** 6 | * Layout component for the signature pad example. 7 | * Demonstrates a responsive signature pad with clear and save functionality. 8 | */ 9 | const Layout = () => { 10 | const signaturePadRef = useRef(null); 11 | const [hasSignature, setHasSignature] = useState(false); 12 | const [savedSignature, setSavedSignature] = useState(null); 13 | 14 | /** 15 | * Set up event listeners for tracking signature state. 16 | */ 17 | useEffect(() => { 18 | const signaturePad = signaturePadRef.current; 19 | 20 | if (!signaturePad) { 21 | return; 22 | } 23 | 24 | const handleBeginStroke = () => { 25 | setHasSignature(true); 26 | }; 27 | 28 | // Add event listeners 29 | signaturePad.instance.addEventListener('beginStroke', handleBeginStroke); 30 | 31 | // Cleanup event listeners 32 | return () => { 33 | signaturePad.instance.removeEventListener('beginStroke', handleBeginStroke); 34 | }; 35 | }, []); 36 | 37 | /** 38 | * Clear the signature pad. 39 | */ 40 | const handleClear = useCallback(() => { 41 | const signaturePad = signaturePadRef.current; 42 | 43 | if (signaturePad) { 44 | signaturePad.instance.clear(); 45 | setHasSignature(false); 46 | } 47 | }, []); 48 | 49 | /** 50 | * Save the signature by opening it in a new window. 51 | */ 52 | const handleSave = useCallback(() => { 53 | const signaturePad = signaturePadRef.current; 54 | 55 | if (!signaturePad) { 56 | return; 57 | } 58 | 59 | if (signaturePad.isEmpty()) { 60 | alert('Please provide a signature first.'); 61 | } else { 62 | const dataURL = signaturePad.toDataURL(); 63 | setSavedSignature(dataURL); 64 | } 65 | }, []); 66 | 67 | return ( 68 |
69 |
70 |
71 |
72 |

Digital Signature Pad

73 |

Create your signature with ease

74 |
75 |
76 | 77 |
78 |
79 |
80 |
81 |
82 |

83 | Sign in the box below using your mouse or touch screen 84 |

85 |
86 | 87 |
88 |
89 |
90 |
91 | 103 |
104 | Sign above 105 |
106 | 117 |
118 |
119 |
120 |
121 | 122 | {savedSignature && ( 123 |
124 |
125 |
126 |
127 |

✅ Saved Signature

128 |
129 | Saved signature 130 |
131 |
132 |
133 |
134 |
135 | )} 136 |
137 |
138 | ); 139 | }; 140 | 141 | export default Layout; 142 | -------------------------------------------------------------------------------- /src/SignaturePad.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import * as PropTypes from 'prop-types'; 3 | import SigPad, { type FromDataUrlOptions, type Options, type PointGroup, type ToSVGOptions } from 'signature_pad'; 4 | import { debounce } from 'throttle-debounce'; 5 | 6 | type Props = { 7 | width?: number; 8 | height?: number; 9 | options?: Options; 10 | canvasProps?: { [key: string]: string | { [key: string]: string } }; 11 | } & DefaultProps; 12 | 13 | type DefaultProps = { 14 | redrawOnResize: boolean; 15 | debounceInterval: number; 16 | }; 17 | 18 | type State = { 19 | canvasWidth: number; 20 | canvasHeight: number; 21 | }; 22 | 23 | /** 24 | * @class 25 | * @classdesc Signature pad component. 26 | * @extends {PureComponent} 27 | */ 28 | class SignaturePad extends React.PureComponent { 29 | static displayName = 'react-signature-pad-wrapper'; 30 | 31 | static propTypes = { 32 | width: PropTypes.number, 33 | height: PropTypes.number, 34 | options: PropTypes.object, 35 | canvasProps: PropTypes.object, 36 | redrawOnResize: PropTypes.bool, 37 | debounceInterval: PropTypes.number, 38 | }; 39 | 40 | static readonly defaultProps: DefaultProps = { 41 | redrawOnResize: false, 42 | debounceInterval: 150, 43 | }; 44 | 45 | private readonly canvasRef = React.createRef(); 46 | 47 | private signaturePad!: SigPad; 48 | 49 | private readonly callResizeHandler!: debounce<() => void>; 50 | 51 | /** 52 | * Create a new signature pad. 53 | * 54 | * @param {Props} props 55 | */ 56 | constructor(props: Props) { 57 | super(props); 58 | 59 | this.state = { canvasWidth: 0, canvasHeight: 0 }; 60 | 61 | this.callResizeHandler = debounce<() => void>(this.props.debounceInterval, this.handleResize.bind(this)); 62 | } 63 | 64 | /** 65 | * Initialise the signature pad once the canvas element is rendered. 66 | * 67 | * @return {void} 68 | */ 69 | componentDidMount(): void { 70 | const canvas = this.canvasRef.current; 71 | 72 | if (canvas) { 73 | if (!this.props.width || !this.props.height) { 74 | canvas.style.width = '100%'; 75 | window.addEventListener('resize', this.callResizeHandler); 76 | } 77 | 78 | this.signaturePad = new SigPad(canvas, this.props.options); 79 | 80 | this.scaleCanvas(canvas); 81 | } 82 | } 83 | 84 | /** 85 | * Remove the resize event listener and switch the signature pad off on 86 | * unmount. 87 | * 88 | * @return {void} 89 | */ 90 | componentWillUnmount(): void { 91 | if (!this.props.width || !this.props.height) { 92 | window.removeEventListener('resize', this.callResizeHandler); 93 | } 94 | 95 | this.signaturePad.off(); 96 | } 97 | 98 | /** 99 | * Get the original signature_pad instance. 100 | * 101 | * @return {SigPad} 102 | */ 103 | get instance(): SigPad { 104 | return this.signaturePad; 105 | } 106 | 107 | /** 108 | * Get the canvas ref. 109 | * 110 | * @return {Object} 111 | */ 112 | get canvas(): React.RefObject { 113 | return this.canvasRef; 114 | } 115 | 116 | /** 117 | * Set the radius of a single dot. 118 | * 119 | * @param {number} dotSize 120 | * @return {void} 121 | */ 122 | set dotSize(dotSize: number) { 123 | this.signaturePad.dotSize = dotSize; 124 | } 125 | 126 | /** 127 | * Get the radius of a single dot. 128 | * 129 | * @return {number} 130 | */ 131 | get dotSize(): number { 132 | return this.signaturePad.dotSize; 133 | } 134 | 135 | /** 136 | * Set the minimum width of a line. 137 | * 138 | * @param {number} minWidth 139 | * @return {void} 140 | */ 141 | set minWidth(minWidth: number) { 142 | this.signaturePad.minWidth = minWidth; 143 | } 144 | 145 | /** 146 | * Get the minimum width of a line. 147 | * 148 | * @return {number} 149 | */ 150 | get minWidth(): number { 151 | return this.signaturePad.minWidth; 152 | } 153 | 154 | /** 155 | * Get the maximum width of a line. 156 | * 157 | * @param {number} maxWidth 158 | * @return {void} 159 | */ 160 | set maxWidth(maxWidth: number) { 161 | this.signaturePad.maxWidth = maxWidth; 162 | } 163 | 164 | /** 165 | * Get the maximum width of a line. 166 | * 167 | * @return {number} 168 | */ 169 | get maxWidth(): number { 170 | return this.signaturePad.maxWidth; 171 | } 172 | 173 | /** 174 | * Set the throttle for drawing the next point at most once every x ms. 175 | * 176 | * @param {number} throttle 177 | * @return {void} 178 | */ 179 | set throttle(throttle: number) { 180 | this.signaturePad.throttle = throttle; 181 | } 182 | 183 | /** 184 | * Get the throttle for drawing the next point at most once every x ms. 185 | * 186 | * @return {number} 187 | */ 188 | get throttle(): number { 189 | return this.signaturePad.throttle; 190 | } 191 | 192 | /** 193 | * Set the color used to clear the background. 194 | * 195 | * @param {string} color 196 | * @return {void} 197 | */ 198 | set backgroundColor(color: string) { 199 | this.signaturePad.backgroundColor = color; 200 | } 201 | 202 | /** 203 | * Get the color used to clear the background. 204 | * 205 | * @return {string} 206 | */ 207 | get backgroundColor(): string { 208 | return this.signaturePad.backgroundColor; 209 | } 210 | 211 | /** 212 | * Set the color used to draw the lines. 213 | * 214 | * @param {string} color 215 | * @return {void} 216 | */ 217 | set penColor(color: string) { 218 | this.signaturePad.penColor = color; 219 | } 220 | 221 | /** 222 | * Get the color used to draw the lines. 223 | * 224 | * @return {string} 225 | */ 226 | get penColor(): string { 227 | return this.signaturePad.penColor; 228 | } 229 | 230 | /** 231 | * Set weight used to modify new velocity based on the previous velocity. 232 | * 233 | * @param {number} weight 234 | * @return {void} 235 | */ 236 | set velocityFilterWeight(weight: number) { 237 | this.signaturePad.velocityFilterWeight = weight; 238 | } 239 | 240 | /** 241 | * Get weight used to modify new velocity based on the previous velocity. 242 | * 243 | * @return {number} 244 | */ 245 | get velocityFilterWeight(): number { 246 | return this.signaturePad.velocityFilterWeight; 247 | } 248 | 249 | /** 250 | * Determine if the canvas is empty. 251 | * 252 | * @return {boolean} 253 | */ 254 | public isEmpty(): boolean { 255 | return this.signaturePad.isEmpty(); 256 | } 257 | 258 | /** 259 | * Clear the canvas. 260 | * 261 | * @return {void} 262 | */ 263 | public clear(): void { 264 | this.signaturePad.clear(); 265 | } 266 | 267 | /** 268 | * Redraw the signature. 269 | * 270 | * @return {void} 271 | */ 272 | public redraw(): void { 273 | this.signaturePad.redraw(); 274 | } 275 | 276 | /** 277 | * Draw a signature from a data URL. 278 | * 279 | * @param {string} dataUrl 280 | * @param {object} options 281 | * @return {void} 282 | */ 283 | public fromDataURL(dataUrl: string, options: FromDataUrlOptions = {}): void { 284 | this.signaturePad.fromDataURL(dataUrl, options); 285 | } 286 | 287 | /** 288 | * Get the signature data as a data URL. 289 | * 290 | * @param {?string} mime 291 | * @param {?number} encoderOptions 292 | * @return {string} 293 | */ 294 | public toDataURL(type?: string, encoderOptions?: number): string { 295 | return this.signaturePad.toDataURL(type, encoderOptions); 296 | } 297 | 298 | /** 299 | * Get the signature data as an SVG data URL. 300 | * 301 | * @param {string} mime 302 | * @param {?ToSVGOptions} encoderOptions 303 | * @return {string} 304 | */ 305 | public toSvgDataUrl(encoderOptions?: ToSVGOptions): string { 306 | return this.signaturePad.toDataURL('image/svg+xml', encoderOptions); 307 | } 308 | 309 | /** 310 | * Get the signature data as an SVG string without converting to base64. 311 | * 312 | * @param {?ToSVGOptions} svgOptions 313 | * @return {string} 314 | */ 315 | toSVG(svgOptions?: ToSVGOptions): string { 316 | return this.signaturePad.toSVG(svgOptions); 317 | } 318 | 319 | /** 320 | * Draw a signature from an array of point groups. 321 | * 322 | * @param {PointGroup[]} data 323 | * @return {void} 324 | */ 325 | fromData(data: PointGroup[]): void { 326 | this.signaturePad.fromData(data); 327 | } 328 | 329 | /** 330 | * Get the signature pad data an array of point groups. 331 | * 332 | * @return {PointGroup[]} 333 | */ 334 | toData(): PointGroup[] { 335 | return this.signaturePad.toData(); 336 | } 337 | 338 | /** 339 | * Turn the signature pad off. 340 | * 341 | * @return {void} 342 | */ 343 | off(): void { 344 | this.signaturePad.off(); 345 | } 346 | 347 | /** 348 | * Turn the signature pad on. 349 | * 350 | * @return {void} 351 | */ 352 | on(): void { 353 | this.signaturePad.on(); 354 | } 355 | 356 | /** 357 | * Handle a resize event. 358 | * 359 | * @return {void} 360 | */ 361 | handleResize(): void { 362 | const canvas = this.canvasRef.current; 363 | 364 | if (canvas) { 365 | this.scaleCanvas(canvas); 366 | } 367 | } 368 | 369 | /** 370 | * Scale the canvas. 371 | * 372 | * @param {HTMLCanvasElement} canvas 373 | * @return {void} 374 | */ 375 | scaleCanvas(canvas: HTMLCanvasElement): void { 376 | const ratio = Math.max(window.devicePixelRatio || 1, 1); 377 | const width = (this.props.width || canvas.offsetWidth) * ratio; 378 | const height = (this.props.height || canvas.offsetHeight) * ratio; 379 | 380 | // Avoid needlessly setting height / width if dimensions haven't changed 381 | const { canvasWidth, canvasHeight } = this.state; 382 | 383 | if (width === canvasWidth && height === canvasHeight) return; 384 | 385 | canvas.width = width; 386 | canvas.height = height; 387 | 388 | this.setState({ canvasWidth: width, canvasHeight: height }); 389 | 390 | const ctx = canvas.getContext('2d'); 391 | 392 | if (ctx) { 393 | ctx.scale(ratio, ratio); 394 | } 395 | 396 | if (this.props.redrawOnResize && this.signaturePad) { 397 | this.redraw(); 398 | } 399 | } 400 | 401 | /** 402 | * Render the signature pad component. 403 | * 404 | * @return {ReactNode} 405 | */ 406 | render(): React.ReactNode { 407 | const { canvasProps } = this.props; 408 | 409 | return ; 410 | } 411 | } 412 | 413 | export default SignaturePad; 414 | -------------------------------------------------------------------------------- /__tests__/helpers/signature.ts: -------------------------------------------------------------------------------- 1 | export default 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAApgAAAF4CAYAAADwnYnWAAAgAElEQVR4Xu3dT2heWfkH8FMRxEFoRxCchabBnQppEHdCGwUXijRdOaKQdOUyjcOAwkDbGRCEwU4XbgSJ2TgDLtLCICIynSpujdmIO1twGHQxY3HEPwj58Vzm5ve2TZr3ffMk7z33fi6EWvPe857zec7At+fee+6p3d3d3eIgQIAAAQIECBAgkCRwSsBMktQMAQIECBAgQIBAIyBgmggECBAgQIAAAQKpAgJmKqfGCBAgQIAAAQIEBExzgAABAgQIECBAIFVAwEzl1BgBAgQIECBAgICAaQ4QIECAAAECBAikCgiYqZwaI0CAAAECBAgQEDDNAQIECBAgQIAAgVQBATOVU2MECBAgQIAAAQICpjlAgAABAgQIECCQKiBgpnJqjAABAgQIECBAQMA0BwgQIECAAAECBFIFBMxUTo0RIECAAAECBAgImOYAAQIECBAgQIBAqoCAmcqpMQIECBAgQIAAAQHTHCBAgAABAgQIEEgVEDBTOTVGgAABAgQIECAgYJoDBAgQIECAAAECqQICZiqnxggQIECAAAECBARMc4AAAQIECBAgQCBVQMBM5dQYAQIECBAgQICAgGkOECBAgAABAgQIpAoImKmcGiNAgAABAgQIEBAwzQECBAgQIECAAIFUAQEzlVNjBAgQIECAAAECAqY5QIAAAQIECBAgkCogYKZyaowAAQIECBAgQEDANAcIECBAgAABAgRSBQTMVE6NESBAgAABAgQICJjmAAECBAgQIECAQKqAgJnKqTECBAgQIECAAAEB0xwgQIAAAQIECBBIFRAwUzk1RoAAAQIECBAgIGCaAwQIECBAgAABAqkCAmYqp8YIECBAgAABAgQETHOAAAECBAgQIEAgVUDATOXUGAECBAgQIECAgIBpDhAgQIAAAQIECKQKCJipnBojQIAAAQIECBAQMM0BAgQIECBAgACBVAEBM5VTYwQIECBAgAABAgKmOUCAAAECBAgQIJAqIGCmcmqMAAECBAgQIEBAwDQHCBAgQIAAAQIEUgUEzFROjREgQIAAAQIECAiY5gABAgQIECBAgECqgICZyqkxAgQIECBAgAABAdMcIECAAAECBAgQSBUQMFM5NUaAAAECBAgQICBgmgMECBAgQIAAAQKpAgJmKqfGCBAg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PwEB07wgQIAAAQIECBBIFRAwUzk1RoAAAQIECBAgIGCaAwQIECBAgAABAqkCAmYqp8YIECBAgAABAgQETHOAAAECBAgQIEAgVUDATOXUGAECBAgQIECAgIBpDhAgQIAAAQIECKQKCJipnBojQIAAAQIECBAQMM0BAgQIECBAgACBVAEBM5VTYwQIECBAgAABAgKmOUCAAAECBAgQIJAqIGCmcmqMAAECBAgQIEBAwDQHCBAgQIAAAQIEUgUEzFROjREgQIAAAQIECAiY5gABAgQIECBAgECqgICZyqkxAgQIECBAgAABAdMcIECAAAECBAgQSBUQMFM5NUaAAAECBAgQICBgmgMECBAgQIAAAQKpAgJmKqfGCBAgQIAAAQIEBExzgAABAgQIECBAIFVAwEzl1BgBAgQIECBAgICAaQ4QIECAAAECBAikCgiYqZwaI0CAAAECBAgQEDDNAQIECBAgQIAAgVQBATOVU2MECBAgQIAAAQICpjlAgAABAgQIECCQKiBgpnJqjAABAgQIECBAQMA0BwgQIECAAAECBFIFBMxUTo0RIECAAAECBAgImOYAAQIECBAgQIBAqoCAmcqpMQIECBAgQIAAAQHTHCBAgAABAgQIEEgVEDBTOTVGgAABAgQIECAgYJoDBAgQIECAAAECqQICZiqnxggQIECAAAECBARMc4AAAQIECBAgQCBVQMBM5dQYAQIECBAgQICAgGkOECBAgAABAgQIpAoImKmcGiNAgAABAgQIEBAwzQECBAgQIECAAIFUAQEzlVNjBAgQIECAAAECAqY5QIAAAQIECBAgkCogYKZyaowAAQIECBAgQEDANAcIECBAgAABAgRSBQTMVE6NESBAgAABAgQICJjmAAECBAgQIECAQKqAgJnKqTECBAgQIECAAAEB0xwgQIAAAQIECBBIFRAwUzk1RoAAAQIECBAgIGCaAwQIECBAgAABAqkCAmYqp8YIECBAgAABAgQETHOAAAECBAgQIEAgVUDATOXUGAECBAgQIECAgIBpDhAgQIAAAQIECKQKCJipnBojQIAAAQIECBAQMM0BAgQIECBAgACBVAEBM5VTYwQIECBAgAABAgKmOUCAAAECBAgQIJAqIGCmcmqMAAECBAgQIEBAwDQHCBAgQIAAAQIEUgUEzFROjREgQIAAAQIECAiY5gABAgQIECBAgECqgICZyqkxAgQIECBAgAABAdMcIECAAAECBAgQSBUQMFM5NUaAAAECBAgQICBgmgMECBAgQIAAAQKpAgJmKqfGCBAgQIAAAQIEBExzgAABAgQIECBAIFVAwEzl1BgBAgQIECBAgICAaQ4QIECAAAECBAikCgiYqZwaI0CAAAECBAgQEH4MVtcAAASpSURBVDDNAQIECBAgQIAAgVQBATOVU2MECBAgQIAAAQICpjlAgAABAgQIECCQKiBgpnJqjAABAgQIECBAQMA0BwgQIECAAAECBFIFBMxUTo0RIECAAAECBAgImOYAAQIECBAgQIBAqoCAmcqpMQIECBAgQIAAAQHTHCBAgAABAgQIEEgVEDBTOTVGgAABAgQIECAgYJoDBAgQIECAAAECqQICZiqnxggQIECAAAECBARMc4AAAQIECBAgQCBVQMBM5dQYAQIECBAgQICAgGkOECBAgAABAgQIpAoImKmcGiNAgAABAgQIEBAwzQECBAgQIECAAIFUAQEzlVNjBAgQIECAAAECAqY5QIAAAQIECBAgkCogYKZyaowAAQIECBAgQEDANAcIECBAgAABAgRSBQTMVE6NESBAgAABAgQICJjmAAECBAgQIECAQKqAgJnKqTECBAgQIECAAAEB0xwgQIAAAQIECBBIFRAwUzk1RoAAAQIECBAgIGCaAwQIECBAgAABAqkCAmYqp8YIECBAgAABAgQETHOAAAECBAgQIEAgVUDATOXUGAECBAgQIECAgIBpDhAgQIAAAQIECKQKCJipnBojQIAAAQIECBAQMM0BAgQIECBAgACBVAEBM5VTYwQIECBAgAABAgKmOUCAAAECBAgQIJAqIGCmcmqMAAECBAgQIEBAwDQHCBAgQIAAAQIEUgUEzFROjREgQIAAAQIECAiY5gABAgQIECBAgECqgICZyqkxAgQIECBAgAABAdMcIECAAAECBAgQSBUQMFM5NUaAAAECBAgQICBgmgMECBAgQIAAAQKpAgJmKqfGCBAgQIAAAQIEBExzgAABAgQIECBAIFVAwEzl1BgBAgQIECBAgICAaQ4QIECAAAECBAikCgiYqZwaI0CAAAECBAgQEDDNAQIECBAgQIAAgVQBATOVU2MECBAgQIAAAQICpjlAgAABAgQIECCQKiBgpnJqjAABAgQIECBAQMA0BwgQIECAAAECBFIFBMxUTo0RIECAAAECBAgImOYAAQIECBAgQIBAqoCAmcqpMQIECBAgQIAAAQHTHCBAgAABAgQIEEgVEDBTOTVGgAABAgQIECAgYJoDBAgQIECAAAECqQICZiqnxggQIECAAAECBARMc4AAAQIECBAgQCBVQMBM5dQYAQIECBAgQICAgGkOECBAgAABAgQIpAoImKmcGiNAgAABAgQIEBAwzQECBAgQIECAAIFUAQEzlVNjBAgQIECAAAECAqY5QIAAAQIECBAgkCogYKZyaowAAQIECBAgQEDANAcIECBAgAABAgRSBQTMVE6NESBAgAABAgQICJjmAAECBAgQIECAQKqAgJnKqTECBAgQIECAAAEB0xwgQIAAAQIECBBIFRAwUzk1RoAAAQIECBAgIGCaAwQIECBAgAABAqkCAmYqp8YIECBAgAABAgQETHOAAAECBAgQIEAgVUDATOXUGAECBAgQIECAgIBpDhAgQIAAAQIECKQKCJipnBojQIAAAQIECBAQMM0BAgQIECBAgACBVAEBM5VTYwQIECBAgAABAgKmOUCAAAECBAgQIJAq8H+ZbYZcipJsNAAAAABJRU5ErkJggg=='; 2 | -------------------------------------------------------------------------------- /__tests__/SignaturePad.test.tsx: -------------------------------------------------------------------------------- 1 | import * as React from 'react'; 2 | import { render } from '@testing-library/react'; 3 | import '@testing-library/jest-dom'; 4 | import SigPad from 'signature_pad'; 5 | 6 | import signature from './helpers/signature'; 7 | import SignaturePad from '../src/SignaturePad'; 8 | 9 | /** 10 | * Set the dimension of the HTML canvas element to the given width and height. 11 | * 12 | * @param {number} width 13 | * @param {number} height 14 | * @return {void} 15 | */ 16 | const scaleCanvas = (width: number, height: number): void => { 17 | Object.defineProperty(HTMLCanvasElement.prototype, 'offsetWidth', { 18 | configurable: true, 19 | value: width, 20 | }); 21 | Object.defineProperty(HTMLCanvasElement.prototype, 'offsetHeight', { 22 | configurable: true, 23 | value: height, 24 | }); 25 | }; 26 | 27 | describe('Component', () => { 28 | describe('SignaturePad', () => { 29 | beforeEach(() => { 30 | jest.restoreAllMocks(); 31 | scaleCanvas(1024, 768); 32 | }); 33 | 34 | it('renders the component', () => { 35 | const { queryByTestId } = render(); 36 | 37 | expect(queryByTestId('canvas-element')).toBeInTheDocument(); 38 | }); 39 | 40 | it('loads a signature', () => { 41 | const instance = React.createRef(); 42 | 43 | render(); 44 | 45 | const signaturePad = instance.current as SignaturePad; 46 | 47 | signaturePad.clear(); 48 | 49 | expect(signaturePad.isEmpty()).toBeTruthy(); 50 | 51 | signaturePad.fromDataURL(signature); 52 | 53 | expect(signaturePad.isEmpty()).toBeFalsy(); 54 | }); 55 | 56 | it('clears the signature pad', () => { 57 | const instance = React.createRef(); 58 | 59 | render(); 60 | 61 | const signaturePad = instance.current as SignaturePad; 62 | 63 | signaturePad.fromDataURL(signature); 64 | 65 | expect(signaturePad.isEmpty()).toBeFalsy(); 66 | 67 | signaturePad.clear(); 68 | 69 | expect(signaturePad.isEmpty()).toBeTruthy(); 70 | }); 71 | 72 | it('returns the underlying signature pad instance', () => { 73 | const instance = React.createRef(); 74 | 75 | render(); 76 | 77 | const signaturePad = instance.current as SignaturePad; 78 | 79 | expect(signaturePad.instance).toBeInstanceOf(SigPad); 80 | }); 81 | 82 | it('returns a ref to the underlying HTML canvas element', () => { 83 | const instance = React.createRef(); 84 | 85 | render(); 86 | 87 | const signaturePad = instance.current as SignaturePad; 88 | 89 | expect(signaturePad.canvas).toBeInstanceOf(Object); 90 | }); 91 | 92 | [ 93 | { name: 'dot size', option: 'dotSize', expected: 3 }, 94 | { name: 'min width', option: 'minWidth', expected: 1 }, 95 | { name: 'max width', option: 'maxWidth', expected: 3 }, 96 | { name: 'throttle', option: 'throttle', expected: 20 }, 97 | { name: 'background color', option: 'backgroundColor', expected: 'rgba(255,255,255)' }, 98 | { name: 'pen color', option: 'penColor', expected: 'red' }, 99 | { name: 'velocity filter weight', option: 'velocityFilterWeight', expected: 1 }, 100 | ].forEach(({ name, option, expected }) => { 101 | it(`sets and gets the ${name}`, () => { 102 | const instance = React.createRef(); 103 | 104 | render(); 105 | 106 | Reflect.set(instance.current as SignaturePad, option, expected); 107 | 108 | expect(Reflect.get(instance.current as SignaturePad, option)).toBe(expected); 109 | }); 110 | }); 111 | 112 | it('returns a signature as a data URL', () => { 113 | const instance = React.createRef(); 114 | 115 | render(); 116 | 117 | const signaturePad = instance.current as SignaturePad; 118 | 119 | expect(signaturePad.toDataURL()).toContain('data:image/png;base64'); 120 | }); 121 | 122 | it('returns a signature as an SVG data URL', () => { 123 | const instance = React.createRef(); 124 | 125 | render(); 126 | 127 | const signaturePad = instance.current as SignaturePad; 128 | 129 | expect(signaturePad.toSvgDataUrl({ includeDataUrl: true })).toContain('data:image/svg+xml;base64'); 130 | }); 131 | 132 | it('returns a signature as an SVG string', () => { 133 | const instance = React.createRef(); 134 | 135 | render(); 136 | 137 | const signaturePad = instance.current as SignaturePad; 138 | 139 | expect(signaturePad.toSVG()).toContain(' { 143 | const instance = React.createRef(); 144 | 145 | render(); 146 | 147 | const signaturePad = instance.current as SignaturePad; 148 | 149 | expect(signaturePad.toData()).toHaveLength(0); 150 | }); 151 | 152 | it('draws a signature from an array of data points', () => { 153 | const instance = React.createRef(); 154 | 155 | render(); 156 | 157 | const signaturePad = instance.current as SignaturePad; 158 | const canvas = signaturePad.canvas.current as HTMLCanvasElement; 159 | const context = canvas.getContext('2d') as CanvasRenderingContext2D; 160 | const data = [ 161 | { 162 | dotSize: 0, 163 | maxWidth: 2.5, 164 | minWidth: 0.5, 165 | penColor: 'black', 166 | velocityFilterWeight: 0.7, 167 | compositeOperation: context.globalCompositeOperation, 168 | points: [{ pressure: 0.5, time: 1641476147709, x: 100, y: 100 }], 169 | }, 170 | ]; 171 | 172 | signaturePad.fromData(data); 173 | 174 | expect(signaturePad.toData()).toStrictEqual(data); 175 | }); 176 | 177 | it('unbinds all event handlers', () => { 178 | const instance = React.createRef(); 179 | 180 | render(); 181 | 182 | const signaturePad = instance.current as SignaturePad; 183 | const spy = jest.spyOn(signaturePad.instance, 'off'); 184 | 185 | signaturePad.off(); 186 | 187 | expect(spy).toHaveBeenCalled(); 188 | 189 | spy.mockRestore(); 190 | }); 191 | 192 | it('rebinds all event handlers', () => { 193 | const instance = React.createRef(); 194 | 195 | render(); 196 | 197 | const signaturePad = instance.current as SignaturePad; 198 | const spy = jest.spyOn(signaturePad.instance, 'on'); 199 | 200 | signaturePad.on(); 201 | 202 | expect(spy).toHaveBeenCalled(); 203 | 204 | spy.mockRestore(); 205 | }); 206 | 207 | it('does not redraw a signature by default when the viewport dimensions change', () => { 208 | const instance = React.createRef(); 209 | 210 | render(); 211 | 212 | const signaturePad = instance.current as SignaturePad; 213 | const spy = jest.spyOn(signaturePad.instance, 'redraw'); 214 | 215 | scaleCanvas(768, 768); 216 | React.act(() => { 217 | signaturePad.handleResize(); 218 | }); 219 | 220 | expect(spy).not.toHaveBeenCalled(); 221 | }); 222 | 223 | it('redraws a signature when the viewport dimensions change', () => { 224 | const instance = React.createRef(); 225 | 226 | render(); 227 | 228 | const signaturePad = instance.current as SignaturePad; 229 | 230 | signaturePad.fromDataURL(signature); 231 | 232 | const spy = jest.spyOn(signaturePad.instance, 'redraw'); 233 | 234 | scaleCanvas(768, 768); 235 | React.act(() => { 236 | signaturePad.handleResize(); 237 | }); 238 | 239 | expect(spy).toHaveBeenCalled(); 240 | }); 241 | 242 | it('does not redraw a signature when the viewport dimensions have not changed', () => { 243 | const instance = React.createRef(); 244 | 245 | render(); 246 | 247 | const signaturePad = instance.current as SignaturePad; 248 | 249 | signaturePad.fromDataURL(signature); 250 | 251 | const spy = jest.spyOn(signaturePad.instance, 'redraw'); 252 | 253 | signaturePad.handleResize(); 254 | 255 | expect(spy).not.toHaveBeenCalled(); 256 | }); 257 | 258 | it('does not redraw a signature when the canvas is empty', () => { 259 | const instance = React.createRef(); 260 | 261 | render(); 262 | 263 | const signaturePad = instance.current as SignaturePad; 264 | 265 | expect(signaturePad.isEmpty()).toBeTruthy(); 266 | 267 | scaleCanvas(768, 768); 268 | 269 | expect(signaturePad.isEmpty()).toBeTruthy(); 270 | }); 271 | 272 | it('does not add the resize event listener on mount', () => { 273 | const spy = jest.spyOn(window, 'addEventListener'); 274 | const instance = React.createRef(); 275 | 276 | render(); 277 | 278 | expect(spy).not.toHaveBeenCalledWith( 279 | 'resize', 280 | Reflect.get(instance.current as SignaturePad, 'callResizeHandler'), 281 | ); 282 | }); 283 | 284 | it('adds the resize event listener on mount', () => { 285 | const spy = jest.spyOn(window, 'addEventListener'); 286 | const instance = React.createRef(); 287 | 288 | render(); 289 | 290 | expect(spy).toHaveBeenCalledWith( 291 | 'resize', 292 | Reflect.get(instance.current as SignaturePad, 'callResizeHandler'), 293 | ); 294 | }); 295 | 296 | it('removes the resize event listener on unmount', () => { 297 | const spy = jest.spyOn(window, 'removeEventListener'); 298 | const instance = React.createRef(); 299 | 300 | const { unmount } = render(); 301 | const handler = Reflect.get(instance.current as SignaturePad, 'callResizeHandler'); 302 | 303 | unmount(); 304 | 305 | expect(spy).toHaveBeenCalledWith('resize', handler); 306 | }); 307 | 308 | it('does not remove the resize event listener on unmount', () => { 309 | const spy = jest.spyOn(window, 'removeEventListener'); 310 | const instance = React.createRef(); 311 | 312 | const { unmount } = render(); 313 | const handler = Reflect.get(instance.current as SignaturePad, 'callResizeHandler'); 314 | 315 | unmount(); 316 | 317 | expect(spy).not.toHaveBeenCalledWith('resize', handler); 318 | }); 319 | }); 320 | }); 321 | -------------------------------------------------------------------------------- /example/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "example", 3 | "version": "1.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "example", 9 | "version": "1.0.0", 10 | "license": "MIT", 11 | "dependencies": { 12 | "bulma": "^1.0.0", 13 | "react": "^19.0.0", 14 | "react-dom": "^19.0.0" 15 | }, 16 | "devDependencies": { 17 | "@types/react": "^19.0.0", 18 | "@types/react-dom": "^19.0.0", 19 | "ts-loader": "^9.2.6", 20 | "typescript": "^5.0.2", 21 | "webpack": "^5.53.0", 22 | "webpack-cli": "^6.0.1" 23 | } 24 | }, 25 | "node_modules/@discoveryjs/json-ext": { 26 | "version": "0.6.3", 27 | "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.6.3.tgz", 28 | "integrity": "sha512-4B4OijXeVNOPZlYA2oEwWOTkzyltLao+xbotHQeqN++Rv27Y6s818+n2Qkp8q+Fxhn0t/5lA5X1Mxktud8eayQ==", 29 | "dev": true, 30 | "license": "MIT", 31 | "engines": { 32 | "node": ">=14.17.0" 33 | } 34 | }, 35 | "node_modules/@jridgewell/gen-mapping": { 36 | "version": "0.3.13", 37 | "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", 38 | "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", 39 | "dev": true, 40 | "license": "MIT", 41 | "dependencies": { 42 | "@jridgewell/sourcemap-codec": "^1.5.0", 43 | "@jridgewell/trace-mapping": "^0.3.24" 44 | } 45 | }, 46 | "node_modules/@jridgewell/resolve-uri": { 47 | "version": "3.1.2", 48 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", 49 | "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", 50 | "dev": true, 51 | "license": "MIT", 52 | "engines": { 53 | "node": ">=6.0.0" 54 | } 55 | }, 56 | "node_modules/@jridgewell/source-map": { 57 | "version": "0.3.11", 58 | "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.11.tgz", 59 | "integrity": "sha512-ZMp1V8ZFcPG5dIWnQLr3NSI1MiCU7UETdS/A0G8V/XWHvJv3ZsFqutJn1Y5RPmAPX6F3BiE397OqveU/9NCuIA==", 60 | "dev": true, 61 | "license": "MIT", 62 | "dependencies": { 63 | "@jridgewell/gen-mapping": "^0.3.5", 64 | "@jridgewell/trace-mapping": "^0.3.25" 65 | } 66 | }, 67 | "node_modules/@jridgewell/sourcemap-codec": { 68 | "version": "1.5.5", 69 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", 70 | "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", 71 | "dev": true, 72 | "license": "MIT" 73 | }, 74 | "node_modules/@jridgewell/trace-mapping": { 75 | "version": "0.3.31", 76 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", 77 | "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", 78 | "dev": true, 79 | "license": "MIT", 80 | "dependencies": { 81 | "@jridgewell/resolve-uri": "^3.1.0", 82 | "@jridgewell/sourcemap-codec": "^1.4.14" 83 | } 84 | }, 85 | "node_modules/@types/eslint": { 86 | "version": "9.6.1", 87 | "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-9.6.1.tgz", 88 | "integrity": "sha512-FXx2pKgId/WyYo2jXw63kk7/+TY7u7AziEJxJAnSFzHlqTAS3Ync6SvgYAN/k4/PQpnnVuzoMuVnByKK2qp0ag==", 89 | "dev": true, 90 | "license": "MIT", 91 | "dependencies": { 92 | "@types/estree": "*", 93 | "@types/json-schema": "*" 94 | } 95 | }, 96 | "node_modules/@types/eslint-scope": { 97 | "version": "3.7.7", 98 | "resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.7.tgz", 99 | "integrity": "sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg==", 100 | "dev": true, 101 | "license": "MIT", 102 | "dependencies": { 103 | "@types/eslint": "*", 104 | "@types/estree": "*" 105 | } 106 | }, 107 | "node_modules/@types/estree": { 108 | "version": "1.0.8", 109 | "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", 110 | "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", 111 | "dev": true, 112 | "license": "MIT" 113 | }, 114 | "node_modules/@types/json-schema": { 115 | "version": "7.0.15", 116 | "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", 117 | "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", 118 | "dev": true, 119 | "license": "MIT" 120 | }, 121 | "node_modules/@types/node": { 122 | "version": "24.10.1", 123 | "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.1.tgz", 124 | "integrity": "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ==", 125 | "dev": true, 126 | "license": "MIT", 127 | "dependencies": { 128 | "undici-types": "~7.16.0" 129 | } 130 | }, 131 | "node_modules/@types/react": { 132 | "version": "19.2.7", 133 | "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.7.tgz", 134 | "integrity": "sha512-MWtvHrGZLFttgeEj28VXHxpmwYbor/ATPYbBfSFZEIRK0ecCFLl2Qo55z52Hss+UV9CRN7trSeq1zbgx7YDWWg==", 135 | "dev": true, 136 | "license": "MIT", 137 | "peer": true, 138 | "dependencies": { 139 | "csstype": "^3.2.2" 140 | } 141 | }, 142 | "node_modules/@types/react-dom": { 143 | "version": "19.2.3", 144 | "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz", 145 | "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==", 146 | "dev": true, 147 | "license": "MIT", 148 | "peerDependencies": { 149 | "@types/react": "^19.2.0" 150 | } 151 | }, 152 | "node_modules/@webassemblyjs/ast": { 153 | "version": "1.14.1", 154 | "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.14.1.tgz", 155 | "integrity": "sha512-nuBEDgQfm1ccRp/8bCQrx1frohyufl4JlbMMZ4P1wpeOfDhF6FQkxZJ1b/e+PLwr6X1Nhw6OLme5usuBWYBvuQ==", 156 | "dev": true, 157 | "license": "MIT", 158 | "dependencies": { 159 | "@webassemblyjs/helper-numbers": "1.13.2", 160 | "@webassemblyjs/helper-wasm-bytecode": "1.13.2" 161 | } 162 | }, 163 | "node_modules/@webassemblyjs/floating-point-hex-parser": { 164 | "version": "1.13.2", 165 | "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.13.2.tgz", 166 | "integrity": "sha512-6oXyTOzbKxGH4steLbLNOu71Oj+C8Lg34n6CqRvqfS2O71BxY6ByfMDRhBytzknj9yGUPVJ1qIKhRlAwO1AovA==", 167 | "dev": true, 168 | "license": "MIT" 169 | }, 170 | "node_modules/@webassemblyjs/helper-api-error": { 171 | "version": "1.13.2", 172 | "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.13.2.tgz", 173 | "integrity": "sha512-U56GMYxy4ZQCbDZd6JuvvNV/WFildOjsaWD3Tzzvmw/mas3cXzRJPMjP83JqEsgSbyrmaGjBfDtV7KDXV9UzFQ==", 174 | "dev": true, 175 | "license": "MIT" 176 | }, 177 | "node_modules/@webassemblyjs/helper-buffer": { 178 | "version": "1.14.1", 179 | "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.14.1.tgz", 180 | "integrity": "sha512-jyH7wtcHiKssDtFPRB+iQdxlDf96m0E39yb0k5uJVhFGleZFoNw1c4aeIcVUPPbXUVJ94wwnMOAqUHyzoEPVMA==", 181 | "dev": true, 182 | "license": "MIT" 183 | }, 184 | "node_modules/@webassemblyjs/helper-numbers": { 185 | "version": "1.13.2", 186 | "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.13.2.tgz", 187 | "integrity": "sha512-FE8aCmS5Q6eQYcV3gI35O4J789wlQA+7JrqTTpJqn5emA4U2hvwJmvFRC0HODS+3Ye6WioDklgd6scJ3+PLnEA==", 188 | "dev": true, 189 | "license": "MIT", 190 | "dependencies": { 191 | "@webassemblyjs/floating-point-hex-parser": "1.13.2", 192 | "@webassemblyjs/helper-api-error": "1.13.2", 193 | "@xtuc/long": "4.2.2" 194 | } 195 | }, 196 | "node_modules/@webassemblyjs/helper-wasm-bytecode": { 197 | "version": "1.13.2", 198 | "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.13.2.tgz", 199 | "integrity": "sha512-3QbLKy93F0EAIXLh0ogEVR6rOubA9AoZ+WRYhNbFyuB70j3dRdwH9g+qXhLAO0kiYGlg3TxDV+I4rQTr/YNXkA==", 200 | "dev": true, 201 | "license": "MIT" 202 | }, 203 | "node_modules/@webassemblyjs/helper-wasm-section": { 204 | "version": "1.14.1", 205 | "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.14.1.tgz", 206 | "integrity": "sha512-ds5mXEqTJ6oxRoqjhWDU83OgzAYjwsCV8Lo/N+oRsNDmx/ZDpqalmrtgOMkHwxsG0iI//3BwWAErYRHtgn0dZw==", 207 | "dev": true, 208 | "license": "MIT", 209 | "dependencies": { 210 | "@webassemblyjs/ast": "1.14.1", 211 | "@webassemblyjs/helper-buffer": "1.14.1", 212 | "@webassemblyjs/helper-wasm-bytecode": "1.13.2", 213 | "@webassemblyjs/wasm-gen": "1.14.1" 214 | } 215 | }, 216 | "node_modules/@webassemblyjs/ieee754": { 217 | "version": "1.13.2", 218 | "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.13.2.tgz", 219 | "integrity": "sha512-4LtOzh58S/5lX4ITKxnAK2USuNEvpdVV9AlgGQb8rJDHaLeHciwG4zlGr0j/SNWlr7x3vO1lDEsuePvtcDNCkw==", 220 | "dev": true, 221 | "license": "MIT", 222 | "dependencies": { 223 | "@xtuc/ieee754": "^1.2.0" 224 | } 225 | }, 226 | "node_modules/@webassemblyjs/leb128": { 227 | "version": "1.13.2", 228 | "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.13.2.tgz", 229 | "integrity": "sha512-Lde1oNoIdzVzdkNEAWZ1dZ5orIbff80YPdHx20mrHwHrVNNTjNr8E3xz9BdpcGqRQbAEa+fkrCb+fRFTl/6sQw==", 230 | "dev": true, 231 | "license": "Apache-2.0", 232 | "dependencies": { 233 | "@xtuc/long": "4.2.2" 234 | } 235 | }, 236 | "node_modules/@webassemblyjs/utf8": { 237 | "version": "1.13.2", 238 | "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.13.2.tgz", 239 | "integrity": "sha512-3NQWGjKTASY1xV5m7Hr0iPeXD9+RDobLll3T9d2AO+g3my8xy5peVyjSag4I50mR1bBSN/Ct12lo+R9tJk0NZQ==", 240 | "dev": true, 241 | "license": "MIT" 242 | }, 243 | "node_modules/@webassemblyjs/wasm-edit": { 244 | "version": "1.14.1", 245 | "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.14.1.tgz", 246 | "integrity": "sha512-RNJUIQH/J8iA/1NzlE4N7KtyZNHi3w7at7hDjvRNm5rcUXa00z1vRz3glZoULfJ5mpvYhLybmVcwcjGrC1pRrQ==", 247 | "dev": true, 248 | "license": "MIT", 249 | "dependencies": { 250 | "@webassemblyjs/ast": "1.14.1", 251 | "@webassemblyjs/helper-buffer": "1.14.1", 252 | "@webassemblyjs/helper-wasm-bytecode": "1.13.2", 253 | "@webassemblyjs/helper-wasm-section": "1.14.1", 254 | "@webassemblyjs/wasm-gen": "1.14.1", 255 | "@webassemblyjs/wasm-opt": "1.14.1", 256 | "@webassemblyjs/wasm-parser": "1.14.1", 257 | "@webassemblyjs/wast-printer": "1.14.1" 258 | } 259 | }, 260 | "node_modules/@webassemblyjs/wasm-gen": { 261 | "version": "1.14.1", 262 | "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.14.1.tgz", 263 | "integrity": "sha512-AmomSIjP8ZbfGQhumkNvgC33AY7qtMCXnN6bL2u2Js4gVCg8fp735aEiMSBbDR7UQIj90n4wKAFUSEd0QN2Ukg==", 264 | "dev": true, 265 | "license": "MIT", 266 | "dependencies": { 267 | "@webassemblyjs/ast": "1.14.1", 268 | "@webassemblyjs/helper-wasm-bytecode": "1.13.2", 269 | "@webassemblyjs/ieee754": "1.13.2", 270 | "@webassemblyjs/leb128": "1.13.2", 271 | "@webassemblyjs/utf8": "1.13.2" 272 | } 273 | }, 274 | "node_modules/@webassemblyjs/wasm-opt": { 275 | "version": "1.14.1", 276 | "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.14.1.tgz", 277 | "integrity": "sha512-PTcKLUNvBqnY2U6E5bdOQcSM+oVP/PmrDY9NzowJjislEjwP/C4an2303MCVS2Mg9d3AJpIGdUFIQQWbPds0Sw==", 278 | "dev": true, 279 | "license": "MIT", 280 | "dependencies": { 281 | "@webassemblyjs/ast": "1.14.1", 282 | "@webassemblyjs/helper-buffer": "1.14.1", 283 | "@webassemblyjs/wasm-gen": "1.14.1", 284 | "@webassemblyjs/wasm-parser": "1.14.1" 285 | } 286 | }, 287 | "node_modules/@webassemblyjs/wasm-parser": { 288 | "version": "1.14.1", 289 | "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.14.1.tgz", 290 | "integrity": "sha512-JLBl+KZ0R5qB7mCnud/yyX08jWFw5MsoalJ1pQ4EdFlgj9VdXKGuENGsiCIjegI1W7p91rUlcB/LB5yRJKNTcQ==", 291 | "dev": true, 292 | "license": "MIT", 293 | "dependencies": { 294 | "@webassemblyjs/ast": "1.14.1", 295 | "@webassemblyjs/helper-api-error": "1.13.2", 296 | "@webassemblyjs/helper-wasm-bytecode": "1.13.2", 297 | "@webassemblyjs/ieee754": "1.13.2", 298 | "@webassemblyjs/leb128": "1.13.2", 299 | "@webassemblyjs/utf8": "1.13.2" 300 | } 301 | }, 302 | "node_modules/@webassemblyjs/wast-printer": { 303 | "version": "1.14.1", 304 | "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.14.1.tgz", 305 | "integrity": "sha512-kPSSXE6De1XOR820C90RIo2ogvZG+c3KiHzqUoO/F34Y2shGzesfqv7o57xrxovZJH/MetF5UjroJ/R/3isoiw==", 306 | "dev": true, 307 | "license": "MIT", 308 | "dependencies": { 309 | "@webassemblyjs/ast": "1.14.1", 310 | "@xtuc/long": "4.2.2" 311 | } 312 | }, 313 | "node_modules/@webpack-cli/configtest": { 314 | "version": "3.0.1", 315 | "resolved": "https://registry.npmjs.org/@webpack-cli/configtest/-/configtest-3.0.1.tgz", 316 | "integrity": "sha512-u8d0pJ5YFgneF/GuvEiDA61Tf1VDomHHYMjv/wc9XzYj7nopltpG96nXN5dJRstxZhcNpV1g+nT6CydO7pHbjA==", 317 | "dev": true, 318 | "license": "MIT", 319 | "engines": { 320 | "node": ">=18.12.0" 321 | }, 322 | "peerDependencies": { 323 | "webpack": "^5.82.0", 324 | "webpack-cli": "6.x.x" 325 | } 326 | }, 327 | "node_modules/@webpack-cli/info": { 328 | "version": "3.0.1", 329 | "resolved": "https://registry.npmjs.org/@webpack-cli/info/-/info-3.0.1.tgz", 330 | "integrity": "sha512-coEmDzc2u/ffMvuW9aCjoRzNSPDl/XLuhPdlFRpT9tZHmJ/039az33CE7uH+8s0uL1j5ZNtfdv0HkfaKRBGJsQ==", 331 | "dev": true, 332 | "license": "MIT", 333 | "engines": { 334 | "node": ">=18.12.0" 335 | }, 336 | "peerDependencies": { 337 | "webpack": "^5.82.0", 338 | "webpack-cli": "6.x.x" 339 | } 340 | }, 341 | "node_modules/@webpack-cli/serve": { 342 | "version": "3.0.1", 343 | "resolved": "https://registry.npmjs.org/@webpack-cli/serve/-/serve-3.0.1.tgz", 344 | "integrity": "sha512-sbgw03xQaCLiT6gcY/6u3qBDn01CWw/nbaXl3gTdTFuJJ75Gffv3E3DBpgvY2fkkrdS1fpjaXNOmJlnbtKauKg==", 345 | "dev": true, 346 | "license": "MIT", 347 | "engines": { 348 | "node": ">=18.12.0" 349 | }, 350 | "peerDependencies": { 351 | "webpack": "^5.82.0", 352 | "webpack-cli": "6.x.x" 353 | }, 354 | "peerDependenciesMeta": { 355 | "webpack-dev-server": { 356 | "optional": true 357 | } 358 | } 359 | }, 360 | "node_modules/@xtuc/ieee754": { 361 | "version": "1.2.0", 362 | "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", 363 | "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", 364 | "dev": true, 365 | "license": "BSD-3-Clause" 366 | }, 367 | "node_modules/@xtuc/long": { 368 | "version": "4.2.2", 369 | "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", 370 | "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", 371 | "dev": true, 372 | "license": "Apache-2.0" 373 | }, 374 | "node_modules/acorn": { 375 | "version": "8.15.0", 376 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", 377 | "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", 378 | "dev": true, 379 | "license": "MIT", 380 | "peer": true, 381 | "bin": { 382 | "acorn": "bin/acorn" 383 | }, 384 | "engines": { 385 | "node": ">=0.4.0" 386 | } 387 | }, 388 | "node_modules/acorn-import-phases": { 389 | "version": "1.0.4", 390 | "resolved": "https://registry.npmjs.org/acorn-import-phases/-/acorn-import-phases-1.0.4.tgz", 391 | "integrity": "sha512-wKmbr/DDiIXzEOiWrTTUcDm24kQ2vGfZQvM2fwg2vXqR5uW6aapr7ObPtj1th32b9u90/Pf4AItvdTh42fBmVQ==", 392 | "dev": true, 393 | "license": "MIT", 394 | "engines": { 395 | "node": ">=10.13.0" 396 | }, 397 | "peerDependencies": { 398 | "acorn": "^8.14.0" 399 | } 400 | }, 401 | "node_modules/ajv": { 402 | "version": "8.17.1", 403 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.17.1.tgz", 404 | "integrity": "sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==", 405 | "dev": true, 406 | "license": "MIT", 407 | "peer": true, 408 | "dependencies": { 409 | "fast-deep-equal": "^3.1.3", 410 | "fast-uri": "^3.0.1", 411 | "json-schema-traverse": "^1.0.0", 412 | "require-from-string": "^2.0.2" 413 | }, 414 | "funding": { 415 | "type": "github", 416 | "url": "https://github.com/sponsors/epoberezkin" 417 | } 418 | }, 419 | "node_modules/ajv-formats": { 420 | "version": "2.1.1", 421 | "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-2.1.1.tgz", 422 | "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", 423 | "dev": true, 424 | "license": "MIT", 425 | "dependencies": { 426 | "ajv": "^8.0.0" 427 | }, 428 | "peerDependencies": { 429 | "ajv": "^8.0.0" 430 | }, 431 | "peerDependenciesMeta": { 432 | "ajv": { 433 | "optional": true 434 | } 435 | } 436 | }, 437 | "node_modules/ajv-keywords": { 438 | "version": "5.1.0", 439 | "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", 440 | "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", 441 | "dev": true, 442 | "license": "MIT", 443 | "dependencies": { 444 | "fast-deep-equal": "^3.1.3" 445 | }, 446 | "peerDependencies": { 447 | "ajv": "^8.8.2" 448 | } 449 | }, 450 | "node_modules/ansi-styles": { 451 | "version": "4.3.0", 452 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 453 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 454 | "dev": true, 455 | "license": "MIT", 456 | "dependencies": { 457 | "color-convert": "^2.0.1" 458 | }, 459 | "engines": { 460 | "node": ">=8" 461 | }, 462 | "funding": { 463 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 464 | } 465 | }, 466 | "node_modules/baseline-browser-mapping": { 467 | "version": "2.9.2", 468 | "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.2.tgz", 469 | "integrity": "sha512-PxSsosKQjI38iXkmb3d0Y32efqyA0uW4s41u4IVBsLlWLhCiYNpH/AfNOVWRqCQBlD8TFJTz6OUWNd4DFJCnmw==", 470 | "dev": true, 471 | "license": "Apache-2.0", 472 | "bin": { 473 | "baseline-browser-mapping": "dist/cli.js" 474 | } 475 | }, 476 | "node_modules/braces": { 477 | "version": "3.0.3", 478 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 479 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 480 | "dev": true, 481 | "license": "MIT", 482 | "dependencies": { 483 | "fill-range": "^7.1.1" 484 | }, 485 | "engines": { 486 | "node": ">=8" 487 | } 488 | }, 489 | "node_modules/browserslist": { 490 | "version": "4.28.1", 491 | "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz", 492 | "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==", 493 | "dev": true, 494 | "funding": [ 495 | { 496 | "type": "opencollective", 497 | "url": "https://opencollective.com/browserslist" 498 | }, 499 | { 500 | "type": "tidelift", 501 | "url": "https://tidelift.com/funding/github/npm/browserslist" 502 | }, 503 | { 504 | "type": "github", 505 | "url": "https://github.com/sponsors/ai" 506 | } 507 | ], 508 | "license": "MIT", 509 | "peer": true, 510 | "dependencies": { 511 | "baseline-browser-mapping": "^2.9.0", 512 | "caniuse-lite": "^1.0.30001759", 513 | "electron-to-chromium": "^1.5.263", 514 | "node-releases": "^2.0.27", 515 | "update-browserslist-db": "^1.2.0" 516 | }, 517 | "bin": { 518 | "browserslist": "cli.js" 519 | }, 520 | "engines": { 521 | "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" 522 | } 523 | }, 524 | "node_modules/buffer-from": { 525 | "version": "1.1.2", 526 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", 527 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", 528 | "dev": true, 529 | "license": "MIT" 530 | }, 531 | "node_modules/bulma": { 532 | "version": "1.0.4", 533 | "resolved": "https://registry.npmjs.org/bulma/-/bulma-1.0.4.tgz", 534 | "integrity": "sha512-Ffb6YGXDiZYX3cqvSbHWqQ8+LkX6tVoTcZuVB3lm93sbAVXlO0D6QlOTMnV6g18gILpAXqkG2z9hf9z4hCjz2g==", 535 | "license": "MIT" 536 | }, 537 | "node_modules/caniuse-lite": { 538 | "version": "1.0.30001759", 539 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001759.tgz", 540 | "integrity": "sha512-Pzfx9fOKoKvevQf8oCXoyNRQ5QyxJj+3O0Rqx2V5oxT61KGx8+n6hV/IUyJeifUci2clnmmKVpvtiqRzgiWjSw==", 541 | "dev": true, 542 | "funding": [ 543 | { 544 | "type": "opencollective", 545 | "url": "https://opencollective.com/browserslist" 546 | }, 547 | { 548 | "type": "tidelift", 549 | "url": "https://tidelift.com/funding/github/npm/caniuse-lite" 550 | }, 551 | { 552 | "type": "github", 553 | "url": "https://github.com/sponsors/ai" 554 | } 555 | ], 556 | "license": "CC-BY-4.0" 557 | }, 558 | "node_modules/chalk": { 559 | "version": "4.1.2", 560 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 561 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 562 | "dev": true, 563 | "license": "MIT", 564 | "dependencies": { 565 | "ansi-styles": "^4.1.0", 566 | "supports-color": "^7.1.0" 567 | }, 568 | "engines": { 569 | "node": ">=10" 570 | }, 571 | "funding": { 572 | "url": "https://github.com/chalk/chalk?sponsor=1" 573 | } 574 | }, 575 | "node_modules/chrome-trace-event": { 576 | "version": "1.0.4", 577 | "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.4.tgz", 578 | "integrity": "sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==", 579 | "dev": true, 580 | "license": "MIT", 581 | "engines": { 582 | "node": ">=6.0" 583 | } 584 | }, 585 | "node_modules/clone-deep": { 586 | "version": "4.0.1", 587 | "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", 588 | "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", 589 | "dev": true, 590 | "license": "MIT", 591 | "dependencies": { 592 | "is-plain-object": "^2.0.4", 593 | "kind-of": "^6.0.2", 594 | "shallow-clone": "^3.0.0" 595 | }, 596 | "engines": { 597 | "node": ">=6" 598 | } 599 | }, 600 | "node_modules/color-convert": { 601 | "version": "2.0.1", 602 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 603 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 604 | "dev": true, 605 | "license": "MIT", 606 | "dependencies": { 607 | "color-name": "~1.1.4" 608 | }, 609 | "engines": { 610 | "node": ">=7.0.0" 611 | } 612 | }, 613 | "node_modules/color-name": { 614 | "version": "1.1.4", 615 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 616 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 617 | "dev": true, 618 | "license": "MIT" 619 | }, 620 | "node_modules/colorette": { 621 | "version": "2.0.20", 622 | "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz", 623 | "integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==", 624 | "dev": true, 625 | "license": "MIT" 626 | }, 627 | "node_modules/commander": { 628 | "version": "2.20.3", 629 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 630 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 631 | "dev": true, 632 | "license": "MIT" 633 | }, 634 | "node_modules/cross-spawn": { 635 | "version": "7.0.6", 636 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", 637 | "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", 638 | "dev": true, 639 | "license": "MIT", 640 | "dependencies": { 641 | "path-key": "^3.1.0", 642 | "shebang-command": "^2.0.0", 643 | "which": "^2.0.1" 644 | }, 645 | "engines": { 646 | "node": ">= 8" 647 | } 648 | }, 649 | "node_modules/csstype": { 650 | "version": "3.2.3", 651 | "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", 652 | "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", 653 | "dev": true, 654 | "license": "MIT" 655 | }, 656 | "node_modules/electron-to-chromium": { 657 | "version": "1.5.266", 658 | "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.266.tgz", 659 | "integrity": "sha512-kgWEglXvkEfMH7rxP5OSZZwnaDWT7J9EoZCujhnpLbfi0bbNtRkgdX2E3gt0Uer11c61qCYktB3hwkAS325sJg==", 660 | "dev": true, 661 | "license": "ISC" 662 | }, 663 | "node_modules/enhanced-resolve": { 664 | "version": "5.18.3", 665 | "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.3.tgz", 666 | "integrity": "sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==", 667 | "dev": true, 668 | "license": "MIT", 669 | "dependencies": { 670 | "graceful-fs": "^4.2.4", 671 | "tapable": "^2.2.0" 672 | }, 673 | "engines": { 674 | "node": ">=10.13.0" 675 | } 676 | }, 677 | "node_modules/envinfo": { 678 | "version": "7.21.0", 679 | "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.21.0.tgz", 680 | "integrity": "sha512-Lw7I8Zp5YKHFCXL7+Dz95g4CcbMEpgvqZNNq3AmlT5XAV6CgAAk6gyAMqn2zjw08K9BHfcNuKrMiCPLByGafow==", 681 | "dev": true, 682 | "license": "MIT", 683 | "bin": { 684 | "envinfo": "dist/cli.js" 685 | }, 686 | "engines": { 687 | "node": ">=4" 688 | } 689 | }, 690 | "node_modules/es-module-lexer": { 691 | "version": "1.7.0", 692 | "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", 693 | "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", 694 | "dev": true, 695 | "license": "MIT" 696 | }, 697 | "node_modules/escalade": { 698 | "version": "3.2.0", 699 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", 700 | "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", 701 | "dev": true, 702 | "license": "MIT", 703 | "engines": { 704 | "node": ">=6" 705 | } 706 | }, 707 | "node_modules/eslint-scope": { 708 | "version": "5.1.1", 709 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", 710 | "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", 711 | "dev": true, 712 | "license": "BSD-2-Clause", 713 | "dependencies": { 714 | "esrecurse": "^4.3.0", 715 | "estraverse": "^4.1.1" 716 | }, 717 | "engines": { 718 | "node": ">=8.0.0" 719 | } 720 | }, 721 | "node_modules/esrecurse": { 722 | "version": "4.3.0", 723 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 724 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 725 | "dev": true, 726 | "license": "BSD-2-Clause", 727 | "dependencies": { 728 | "estraverse": "^5.2.0" 729 | }, 730 | "engines": { 731 | "node": ">=4.0" 732 | } 733 | }, 734 | "node_modules/esrecurse/node_modules/estraverse": { 735 | "version": "5.3.0", 736 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", 737 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", 738 | "dev": true, 739 | "license": "BSD-2-Clause", 740 | "engines": { 741 | "node": ">=4.0" 742 | } 743 | }, 744 | "node_modules/estraverse": { 745 | "version": "4.3.0", 746 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", 747 | "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", 748 | "dev": true, 749 | "license": "BSD-2-Clause", 750 | "engines": { 751 | "node": ">=4.0" 752 | } 753 | }, 754 | "node_modules/events": { 755 | "version": "3.3.0", 756 | "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", 757 | "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", 758 | "dev": true, 759 | "license": "MIT", 760 | "engines": { 761 | "node": ">=0.8.x" 762 | } 763 | }, 764 | "node_modules/fast-deep-equal": { 765 | "version": "3.1.3", 766 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 767 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", 768 | "dev": true, 769 | "license": "MIT" 770 | }, 771 | "node_modules/fast-uri": { 772 | "version": "3.1.0", 773 | "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.1.0.tgz", 774 | "integrity": "sha512-iPeeDKJSWf4IEOasVVrknXpaBV0IApz/gp7S2bb7Z4Lljbl2MGJRqInZiUrQwV16cpzw/D3S5j5Julj/gT52AA==", 775 | "dev": true, 776 | "funding": [ 777 | { 778 | "type": "github", 779 | "url": "https://github.com/sponsors/fastify" 780 | }, 781 | { 782 | "type": "opencollective", 783 | "url": "https://opencollective.com/fastify" 784 | } 785 | ], 786 | "license": "BSD-3-Clause" 787 | }, 788 | "node_modules/fastest-levenshtein": { 789 | "version": "1.0.16", 790 | "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz", 791 | "integrity": "sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==", 792 | "dev": true, 793 | "license": "MIT", 794 | "engines": { 795 | "node": ">= 4.9.1" 796 | } 797 | }, 798 | "node_modules/fill-range": { 799 | "version": "7.1.1", 800 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 801 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 802 | "dev": true, 803 | "license": "MIT", 804 | "dependencies": { 805 | "to-regex-range": "^5.0.1" 806 | }, 807 | "engines": { 808 | "node": ">=8" 809 | } 810 | }, 811 | "node_modules/find-up": { 812 | "version": "4.1.0", 813 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 814 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 815 | "dev": true, 816 | "license": "MIT", 817 | "dependencies": { 818 | "locate-path": "^5.0.0", 819 | "path-exists": "^4.0.0" 820 | }, 821 | "engines": { 822 | "node": ">=8" 823 | } 824 | }, 825 | "node_modules/flat": { 826 | "version": "5.0.2", 827 | "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", 828 | "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", 829 | "dev": true, 830 | "license": "BSD-3-Clause", 831 | "bin": { 832 | "flat": "cli.js" 833 | } 834 | }, 835 | "node_modules/function-bind": { 836 | "version": "1.1.2", 837 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 838 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 839 | "dev": true, 840 | "license": "MIT", 841 | "funding": { 842 | "url": "https://github.com/sponsors/ljharb" 843 | } 844 | }, 845 | "node_modules/glob-to-regexp": { 846 | "version": "0.4.1", 847 | "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", 848 | "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", 849 | "dev": true, 850 | "license": "BSD-2-Clause" 851 | }, 852 | "node_modules/graceful-fs": { 853 | "version": "4.2.11", 854 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", 855 | "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", 856 | "dev": true, 857 | "license": "ISC" 858 | }, 859 | "node_modules/has-flag": { 860 | "version": "4.0.0", 861 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 862 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 863 | "dev": true, 864 | "license": "MIT", 865 | "engines": { 866 | "node": ">=8" 867 | } 868 | }, 869 | "node_modules/hasown": { 870 | "version": "2.0.2", 871 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 872 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 873 | "dev": true, 874 | "license": "MIT", 875 | "dependencies": { 876 | "function-bind": "^1.1.2" 877 | }, 878 | "engines": { 879 | "node": ">= 0.4" 880 | } 881 | }, 882 | "node_modules/import-local": { 883 | "version": "3.2.0", 884 | "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", 885 | "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", 886 | "dev": true, 887 | "license": "MIT", 888 | "dependencies": { 889 | "pkg-dir": "^4.2.0", 890 | "resolve-cwd": "^3.0.0" 891 | }, 892 | "bin": { 893 | "import-local-fixture": "fixtures/cli.js" 894 | }, 895 | "engines": { 896 | "node": ">=8" 897 | }, 898 | "funding": { 899 | "url": "https://github.com/sponsors/sindresorhus" 900 | } 901 | }, 902 | "node_modules/interpret": { 903 | "version": "3.1.1", 904 | "resolved": "https://registry.npmjs.org/interpret/-/interpret-3.1.1.tgz", 905 | "integrity": "sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==", 906 | "dev": true, 907 | "license": "MIT", 908 | "engines": { 909 | "node": ">=10.13.0" 910 | } 911 | }, 912 | "node_modules/is-core-module": { 913 | "version": "2.16.1", 914 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", 915 | "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", 916 | "dev": true, 917 | "license": "MIT", 918 | "dependencies": { 919 | "hasown": "^2.0.2" 920 | }, 921 | "engines": { 922 | "node": ">= 0.4" 923 | }, 924 | "funding": { 925 | "url": "https://github.com/sponsors/ljharb" 926 | } 927 | }, 928 | "node_modules/is-number": { 929 | "version": "7.0.0", 930 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 931 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 932 | "dev": true, 933 | "license": "MIT", 934 | "engines": { 935 | "node": ">=0.12.0" 936 | } 937 | }, 938 | "node_modules/is-plain-object": { 939 | "version": "2.0.4", 940 | "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", 941 | "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", 942 | "dev": true, 943 | "license": "MIT", 944 | "dependencies": { 945 | "isobject": "^3.0.1" 946 | }, 947 | "engines": { 948 | "node": ">=0.10.0" 949 | } 950 | }, 951 | "node_modules/isexe": { 952 | "version": "2.0.0", 953 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 954 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 955 | "dev": true, 956 | "license": "ISC" 957 | }, 958 | "node_modules/isobject": { 959 | "version": "3.0.1", 960 | "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", 961 | "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", 962 | "dev": true, 963 | "license": "MIT", 964 | "engines": { 965 | "node": ">=0.10.0" 966 | } 967 | }, 968 | "node_modules/jest-worker": { 969 | "version": "27.5.1", 970 | "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", 971 | "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", 972 | "dev": true, 973 | "license": "MIT", 974 | "dependencies": { 975 | "@types/node": "*", 976 | "merge-stream": "^2.0.0", 977 | "supports-color": "^8.0.0" 978 | }, 979 | "engines": { 980 | "node": ">= 10.13.0" 981 | } 982 | }, 983 | "node_modules/jest-worker/node_modules/supports-color": { 984 | "version": "8.1.1", 985 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", 986 | "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", 987 | "dev": true, 988 | "license": "MIT", 989 | "dependencies": { 990 | "has-flag": "^4.0.0" 991 | }, 992 | "engines": { 993 | "node": ">=10" 994 | }, 995 | "funding": { 996 | "url": "https://github.com/chalk/supports-color?sponsor=1" 997 | } 998 | }, 999 | "node_modules/json-parse-even-better-errors": { 1000 | "version": "2.3.1", 1001 | "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", 1002 | "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", 1003 | "dev": true, 1004 | "license": "MIT" 1005 | }, 1006 | "node_modules/json-schema-traverse": { 1007 | "version": "1.0.0", 1008 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", 1009 | "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", 1010 | "dev": true, 1011 | "license": "MIT" 1012 | }, 1013 | "node_modules/kind-of": { 1014 | "version": "6.0.3", 1015 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", 1016 | "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", 1017 | "dev": true, 1018 | "license": "MIT", 1019 | "engines": { 1020 | "node": ">=0.10.0" 1021 | } 1022 | }, 1023 | "node_modules/loader-runner": { 1024 | "version": "4.3.1", 1025 | "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.1.tgz", 1026 | "integrity": "sha512-IWqP2SCPhyVFTBtRcgMHdzlf9ul25NwaFx4wCEH/KjAXuuHY4yNjvPXsBokp8jCB936PyWRaPKUNh8NvylLp2Q==", 1027 | "dev": true, 1028 | "license": "MIT", 1029 | "engines": { 1030 | "node": ">=6.11.5" 1031 | }, 1032 | "funding": { 1033 | "type": "opencollective", 1034 | "url": "https://opencollective.com/webpack" 1035 | } 1036 | }, 1037 | "node_modules/locate-path": { 1038 | "version": "5.0.0", 1039 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 1040 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 1041 | "dev": true, 1042 | "license": "MIT", 1043 | "dependencies": { 1044 | "p-locate": "^4.1.0" 1045 | }, 1046 | "engines": { 1047 | "node": ">=8" 1048 | } 1049 | }, 1050 | "node_modules/merge-stream": { 1051 | "version": "2.0.0", 1052 | "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", 1053 | "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", 1054 | "dev": true, 1055 | "license": "MIT" 1056 | }, 1057 | "node_modules/micromatch": { 1058 | "version": "4.0.8", 1059 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", 1060 | "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", 1061 | "dev": true, 1062 | "license": "MIT", 1063 | "dependencies": { 1064 | "braces": "^3.0.3", 1065 | "picomatch": "^2.3.1" 1066 | }, 1067 | "engines": { 1068 | "node": ">=8.6" 1069 | } 1070 | }, 1071 | "node_modules/mime-db": { 1072 | "version": "1.52.0", 1073 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 1074 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 1075 | "dev": true, 1076 | "license": "MIT", 1077 | "engines": { 1078 | "node": ">= 0.6" 1079 | } 1080 | }, 1081 | "node_modules/mime-types": { 1082 | "version": "2.1.35", 1083 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 1084 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 1085 | "dev": true, 1086 | "license": "MIT", 1087 | "dependencies": { 1088 | "mime-db": "1.52.0" 1089 | }, 1090 | "engines": { 1091 | "node": ">= 0.6" 1092 | } 1093 | }, 1094 | "node_modules/neo-async": { 1095 | "version": "2.6.2", 1096 | "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", 1097 | "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", 1098 | "dev": true, 1099 | "license": "MIT" 1100 | }, 1101 | "node_modules/node-releases": { 1102 | "version": "2.0.27", 1103 | "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", 1104 | "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", 1105 | "dev": true, 1106 | "license": "MIT" 1107 | }, 1108 | "node_modules/p-limit": { 1109 | "version": "2.3.0", 1110 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 1111 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 1112 | "dev": true, 1113 | "license": "MIT", 1114 | "dependencies": { 1115 | "p-try": "^2.0.0" 1116 | }, 1117 | "engines": { 1118 | "node": ">=6" 1119 | }, 1120 | "funding": { 1121 | "url": "https://github.com/sponsors/sindresorhus" 1122 | } 1123 | }, 1124 | "node_modules/p-locate": { 1125 | "version": "4.1.0", 1126 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 1127 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 1128 | "dev": true, 1129 | "license": "MIT", 1130 | "dependencies": { 1131 | "p-limit": "^2.2.0" 1132 | }, 1133 | "engines": { 1134 | "node": ">=8" 1135 | } 1136 | }, 1137 | "node_modules/p-try": { 1138 | "version": "2.2.0", 1139 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1140 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1141 | "dev": true, 1142 | "license": "MIT", 1143 | "engines": { 1144 | "node": ">=6" 1145 | } 1146 | }, 1147 | "node_modules/path-exists": { 1148 | "version": "4.0.0", 1149 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 1150 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 1151 | "dev": true, 1152 | "license": "MIT", 1153 | "engines": { 1154 | "node": ">=8" 1155 | } 1156 | }, 1157 | "node_modules/path-key": { 1158 | "version": "3.1.1", 1159 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 1160 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 1161 | "dev": true, 1162 | "license": "MIT", 1163 | "engines": { 1164 | "node": ">=8" 1165 | } 1166 | }, 1167 | "node_modules/path-parse": { 1168 | "version": "1.0.7", 1169 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1170 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 1171 | "dev": true, 1172 | "license": "MIT" 1173 | }, 1174 | "node_modules/picocolors": { 1175 | "version": "1.1.1", 1176 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", 1177 | "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", 1178 | "dev": true, 1179 | "license": "ISC" 1180 | }, 1181 | "node_modules/picomatch": { 1182 | "version": "2.3.1", 1183 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 1184 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 1185 | "dev": true, 1186 | "license": "MIT", 1187 | "engines": { 1188 | "node": ">=8.6" 1189 | }, 1190 | "funding": { 1191 | "url": "https://github.com/sponsors/jonschlinkert" 1192 | } 1193 | }, 1194 | "node_modules/pkg-dir": { 1195 | "version": "4.2.0", 1196 | "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", 1197 | "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", 1198 | "dev": true, 1199 | "license": "MIT", 1200 | "dependencies": { 1201 | "find-up": "^4.0.0" 1202 | }, 1203 | "engines": { 1204 | "node": ">=8" 1205 | } 1206 | }, 1207 | "node_modules/randombytes": { 1208 | "version": "2.1.0", 1209 | "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", 1210 | "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", 1211 | "dev": true, 1212 | "license": "MIT", 1213 | "dependencies": { 1214 | "safe-buffer": "^5.1.0" 1215 | } 1216 | }, 1217 | "node_modules/react": { 1218 | "version": "19.2.1", 1219 | "resolved": "https://registry.npmjs.org/react/-/react-19.2.1.tgz", 1220 | "integrity": "sha512-DGrYcCWK7tvYMnWh79yrPHt+vdx9tY+1gPZa7nJQtO/p8bLTDaHp4dzwEhQB7pZ4Xe3ok4XKuEPrVuc+wlpkmw==", 1221 | "license": "MIT", 1222 | "peer": true, 1223 | "engines": { 1224 | "node": ">=0.10.0" 1225 | } 1226 | }, 1227 | "node_modules/react-dom": { 1228 | "version": "19.2.1", 1229 | "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.1.tgz", 1230 | "integrity": "sha512-ibrK8llX2a4eOskq1mXKu/TGZj9qzomO+sNfO98M6d9zIPOEhlBkMkBUBLd1vgS0gQsLDBzA+8jJBVXDnfHmJg==", 1231 | "license": "MIT", 1232 | "dependencies": { 1233 | "scheduler": "^0.27.0" 1234 | }, 1235 | "peerDependencies": { 1236 | "react": "^19.2.1" 1237 | } 1238 | }, 1239 | "node_modules/rechoir": { 1240 | "version": "0.8.0", 1241 | "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.8.0.tgz", 1242 | "integrity": "sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ==", 1243 | "dev": true, 1244 | "license": "MIT", 1245 | "dependencies": { 1246 | "resolve": "^1.20.0" 1247 | }, 1248 | "engines": { 1249 | "node": ">= 10.13.0" 1250 | } 1251 | }, 1252 | "node_modules/require-from-string": { 1253 | "version": "2.0.2", 1254 | "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", 1255 | "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", 1256 | "dev": true, 1257 | "license": "MIT", 1258 | "engines": { 1259 | "node": ">=0.10.0" 1260 | } 1261 | }, 1262 | "node_modules/resolve": { 1263 | "version": "1.22.11", 1264 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", 1265 | "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", 1266 | "dev": true, 1267 | "license": "MIT", 1268 | "dependencies": { 1269 | "is-core-module": "^2.16.1", 1270 | "path-parse": "^1.0.7", 1271 | "supports-preserve-symlinks-flag": "^1.0.0" 1272 | }, 1273 | "bin": { 1274 | "resolve": "bin/resolve" 1275 | }, 1276 | "engines": { 1277 | "node": ">= 0.4" 1278 | }, 1279 | "funding": { 1280 | "url": "https://github.com/sponsors/ljharb" 1281 | } 1282 | }, 1283 | "node_modules/resolve-cwd": { 1284 | "version": "3.0.0", 1285 | "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", 1286 | "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", 1287 | "dev": true, 1288 | "license": "MIT", 1289 | "dependencies": { 1290 | "resolve-from": "^5.0.0" 1291 | }, 1292 | "engines": { 1293 | "node": ">=8" 1294 | } 1295 | }, 1296 | "node_modules/resolve-from": { 1297 | "version": "5.0.0", 1298 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", 1299 | "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", 1300 | "dev": true, 1301 | "license": "MIT", 1302 | "engines": { 1303 | "node": ">=8" 1304 | } 1305 | }, 1306 | "node_modules/safe-buffer": { 1307 | "version": "5.2.1", 1308 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1309 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1310 | "dev": true, 1311 | "funding": [ 1312 | { 1313 | "type": "github", 1314 | "url": "https://github.com/sponsors/feross" 1315 | }, 1316 | { 1317 | "type": "patreon", 1318 | "url": "https://www.patreon.com/feross" 1319 | }, 1320 | { 1321 | "type": "consulting", 1322 | "url": "https://feross.org/support" 1323 | } 1324 | ], 1325 | "license": "MIT" 1326 | }, 1327 | "node_modules/scheduler": { 1328 | "version": "0.27.0", 1329 | "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz", 1330 | "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==", 1331 | "license": "MIT" 1332 | }, 1333 | "node_modules/schema-utils": { 1334 | "version": "4.3.3", 1335 | "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.3.tgz", 1336 | "integrity": "sha512-eflK8wEtyOE6+hsaRVPxvUKYCpRgzLqDTb8krvAsRIwOGlHoSgYLgBXoubGgLd2fT41/OUYdb48v4k4WWHQurA==", 1337 | "dev": true, 1338 | "license": "MIT", 1339 | "dependencies": { 1340 | "@types/json-schema": "^7.0.9", 1341 | "ajv": "^8.9.0", 1342 | "ajv-formats": "^2.1.1", 1343 | "ajv-keywords": "^5.1.0" 1344 | }, 1345 | "engines": { 1346 | "node": ">= 10.13.0" 1347 | }, 1348 | "funding": { 1349 | "type": "opencollective", 1350 | "url": "https://opencollective.com/webpack" 1351 | } 1352 | }, 1353 | "node_modules/semver": { 1354 | "version": "7.7.3", 1355 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", 1356 | "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", 1357 | "dev": true, 1358 | "license": "ISC", 1359 | "bin": { 1360 | "semver": "bin/semver.js" 1361 | }, 1362 | "engines": { 1363 | "node": ">=10" 1364 | } 1365 | }, 1366 | "node_modules/serialize-javascript": { 1367 | "version": "6.0.2", 1368 | "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.2.tgz", 1369 | "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", 1370 | "dev": true, 1371 | "license": "BSD-3-Clause", 1372 | "dependencies": { 1373 | "randombytes": "^2.1.0" 1374 | } 1375 | }, 1376 | "node_modules/shallow-clone": { 1377 | "version": "3.0.1", 1378 | "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", 1379 | "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", 1380 | "dev": true, 1381 | "license": "MIT", 1382 | "dependencies": { 1383 | "kind-of": "^6.0.2" 1384 | }, 1385 | "engines": { 1386 | "node": ">=8" 1387 | } 1388 | }, 1389 | "node_modules/shebang-command": { 1390 | "version": "2.0.0", 1391 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 1392 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 1393 | "dev": true, 1394 | "license": "MIT", 1395 | "dependencies": { 1396 | "shebang-regex": "^3.0.0" 1397 | }, 1398 | "engines": { 1399 | "node": ">=8" 1400 | } 1401 | }, 1402 | "node_modules/shebang-regex": { 1403 | "version": "3.0.0", 1404 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 1405 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 1406 | "dev": true, 1407 | "license": "MIT", 1408 | "engines": { 1409 | "node": ">=8" 1410 | } 1411 | }, 1412 | "node_modules/source-map": { 1413 | "version": "0.7.6", 1414 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.6.tgz", 1415 | "integrity": "sha512-i5uvt8C3ikiWeNZSVZNWcfZPItFQOsYTUAOkcUPGd8DqDy1uOUikjt5dG+uRlwyvR108Fb9DOd4GvXfT0N2/uQ==", 1416 | "dev": true, 1417 | "license": "BSD-3-Clause", 1418 | "engines": { 1419 | "node": ">= 12" 1420 | } 1421 | }, 1422 | "node_modules/source-map-support": { 1423 | "version": "0.5.21", 1424 | "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", 1425 | "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", 1426 | "dev": true, 1427 | "license": "MIT", 1428 | "dependencies": { 1429 | "buffer-from": "^1.0.0", 1430 | "source-map": "^0.6.0" 1431 | } 1432 | }, 1433 | "node_modules/source-map-support/node_modules/source-map": { 1434 | "version": "0.6.1", 1435 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 1436 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 1437 | "dev": true, 1438 | "license": "BSD-3-Clause", 1439 | "engines": { 1440 | "node": ">=0.10.0" 1441 | } 1442 | }, 1443 | "node_modules/supports-color": { 1444 | "version": "7.2.0", 1445 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 1446 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 1447 | "dev": true, 1448 | "license": "MIT", 1449 | "dependencies": { 1450 | "has-flag": "^4.0.0" 1451 | }, 1452 | "engines": { 1453 | "node": ">=8" 1454 | } 1455 | }, 1456 | "node_modules/supports-preserve-symlinks-flag": { 1457 | "version": "1.0.0", 1458 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 1459 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 1460 | "dev": true, 1461 | "license": "MIT", 1462 | "engines": { 1463 | "node": ">= 0.4" 1464 | }, 1465 | "funding": { 1466 | "url": "https://github.com/sponsors/ljharb" 1467 | } 1468 | }, 1469 | "node_modules/tapable": { 1470 | "version": "2.3.0", 1471 | "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz", 1472 | "integrity": "sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg==", 1473 | "dev": true, 1474 | "license": "MIT", 1475 | "engines": { 1476 | "node": ">=6" 1477 | }, 1478 | "funding": { 1479 | "type": "opencollective", 1480 | "url": "https://opencollective.com/webpack" 1481 | } 1482 | }, 1483 | "node_modules/terser": { 1484 | "version": "5.44.1", 1485 | "resolved": "https://registry.npmjs.org/terser/-/terser-5.44.1.tgz", 1486 | "integrity": "sha512-t/R3R/n0MSwnnazuPpPNVO60LX0SKL45pyl9YlvxIdkH0Of7D5qM2EVe+yASRIlY5pZ73nclYJfNANGWPwFDZw==", 1487 | "dev": true, 1488 | "license": "BSD-2-Clause", 1489 | "dependencies": { 1490 | "@jridgewell/source-map": "^0.3.3", 1491 | "acorn": "^8.15.0", 1492 | "commander": "^2.20.0", 1493 | "source-map-support": "~0.5.20" 1494 | }, 1495 | "bin": { 1496 | "terser": "bin/terser" 1497 | }, 1498 | "engines": { 1499 | "node": ">=10" 1500 | } 1501 | }, 1502 | "node_modules/terser-webpack-plugin": { 1503 | "version": "5.3.15", 1504 | "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.15.tgz", 1505 | "integrity": "sha512-PGkOdpRFK+rb1TzVz+msVhw4YMRT9txLF4kRqvJhGhCM324xuR3REBSHALN+l+sAhKUmz0aotnjp5D+P83mLhQ==", 1506 | "dev": true, 1507 | "license": "MIT", 1508 | "dependencies": { 1509 | "@jridgewell/trace-mapping": "^0.3.25", 1510 | "jest-worker": "^27.4.5", 1511 | "schema-utils": "^4.3.0", 1512 | "serialize-javascript": "^6.0.2", 1513 | "terser": "^5.31.1" 1514 | }, 1515 | "engines": { 1516 | "node": ">= 10.13.0" 1517 | }, 1518 | "funding": { 1519 | "type": "opencollective", 1520 | "url": "https://opencollective.com/webpack" 1521 | }, 1522 | "peerDependencies": { 1523 | "webpack": "^5.1.0" 1524 | }, 1525 | "peerDependenciesMeta": { 1526 | "@swc/core": { 1527 | "optional": true 1528 | }, 1529 | "esbuild": { 1530 | "optional": true 1531 | }, 1532 | "uglify-js": { 1533 | "optional": true 1534 | } 1535 | } 1536 | }, 1537 | "node_modules/to-regex-range": { 1538 | "version": "5.0.1", 1539 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 1540 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 1541 | "dev": true, 1542 | "license": "MIT", 1543 | "dependencies": { 1544 | "is-number": "^7.0.0" 1545 | }, 1546 | "engines": { 1547 | "node": ">=8.0" 1548 | } 1549 | }, 1550 | "node_modules/ts-loader": { 1551 | "version": "9.5.4", 1552 | "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.5.4.tgz", 1553 | "integrity": "sha512-nCz0rEwunlTZiy6rXFByQU1kVVpCIgUpc/psFiKVrUwrizdnIbRFu8w7bxhUF0X613DYwT4XzrZHpVyMe758hQ==", 1554 | "dev": true, 1555 | "license": "MIT", 1556 | "dependencies": { 1557 | "chalk": "^4.1.0", 1558 | "enhanced-resolve": "^5.0.0", 1559 | "micromatch": "^4.0.0", 1560 | "semver": "^7.3.4", 1561 | "source-map": "^0.7.4" 1562 | }, 1563 | "engines": { 1564 | "node": ">=12.0.0" 1565 | }, 1566 | "peerDependencies": { 1567 | "typescript": "*", 1568 | "webpack": "^5.0.0" 1569 | } 1570 | }, 1571 | "node_modules/typescript": { 1572 | "version": "5.9.3", 1573 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", 1574 | "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", 1575 | "dev": true, 1576 | "license": "Apache-2.0", 1577 | "peer": true, 1578 | "bin": { 1579 | "tsc": "bin/tsc", 1580 | "tsserver": "bin/tsserver" 1581 | }, 1582 | "engines": { 1583 | "node": ">=14.17" 1584 | } 1585 | }, 1586 | "node_modules/undici-types": { 1587 | "version": "7.16.0", 1588 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", 1589 | "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", 1590 | "dev": true, 1591 | "license": "MIT" 1592 | }, 1593 | "node_modules/update-browserslist-db": { 1594 | "version": "1.2.2", 1595 | "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.2.tgz", 1596 | "integrity": "sha512-E85pfNzMQ9jpKkA7+TJAi4TJN+tBCuWh5rUcS/sv6cFi+1q9LYDwDI5dpUL0u/73EElyQ8d3TEaeW4sPedBqYA==", 1597 | "dev": true, 1598 | "funding": [ 1599 | { 1600 | "type": "opencollective", 1601 | "url": "https://opencollective.com/browserslist" 1602 | }, 1603 | { 1604 | "type": "tidelift", 1605 | "url": "https://tidelift.com/funding/github/npm/browserslist" 1606 | }, 1607 | { 1608 | "type": "github", 1609 | "url": "https://github.com/sponsors/ai" 1610 | } 1611 | ], 1612 | "license": "MIT", 1613 | "dependencies": { 1614 | "escalade": "^3.2.0", 1615 | "picocolors": "^1.1.1" 1616 | }, 1617 | "bin": { 1618 | "update-browserslist-db": "cli.js" 1619 | }, 1620 | "peerDependencies": { 1621 | "browserslist": ">= 4.21.0" 1622 | } 1623 | }, 1624 | "node_modules/watchpack": { 1625 | "version": "2.4.4", 1626 | "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.4.tgz", 1627 | "integrity": "sha512-c5EGNOiyxxV5qmTtAB7rbiXxi1ooX1pQKMLX/MIabJjRA0SJBQOjKF+KSVfHkr9U1cADPon0mRiVe/riyaiDUA==", 1628 | "dev": true, 1629 | "license": "MIT", 1630 | "dependencies": { 1631 | "glob-to-regexp": "^0.4.1", 1632 | "graceful-fs": "^4.1.2" 1633 | }, 1634 | "engines": { 1635 | "node": ">=10.13.0" 1636 | } 1637 | }, 1638 | "node_modules/webpack": { 1639 | "version": "5.103.0", 1640 | "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.103.0.tgz", 1641 | "integrity": "sha512-HU1JOuV1OavsZ+mfigY0j8d1TgQgbZ6M+J75zDkpEAwYeXjWSqrGJtgnPblJjd/mAyTNQ7ygw0MiKOn6etz8yw==", 1642 | "dev": true, 1643 | "license": "MIT", 1644 | "peer": true, 1645 | "dependencies": { 1646 | "@types/eslint-scope": "^3.7.7", 1647 | "@types/estree": "^1.0.8", 1648 | "@types/json-schema": "^7.0.15", 1649 | "@webassemblyjs/ast": "^1.14.1", 1650 | "@webassemblyjs/wasm-edit": "^1.14.1", 1651 | "@webassemblyjs/wasm-parser": "^1.14.1", 1652 | "acorn": "^8.15.0", 1653 | "acorn-import-phases": "^1.0.3", 1654 | "browserslist": "^4.26.3", 1655 | "chrome-trace-event": "^1.0.2", 1656 | "enhanced-resolve": "^5.17.3", 1657 | "es-module-lexer": "^1.2.1", 1658 | "eslint-scope": "5.1.1", 1659 | "events": "^3.2.0", 1660 | "glob-to-regexp": "^0.4.1", 1661 | "graceful-fs": "^4.2.11", 1662 | "json-parse-even-better-errors": "^2.3.1", 1663 | "loader-runner": "^4.3.1", 1664 | "mime-types": "^2.1.27", 1665 | "neo-async": "^2.6.2", 1666 | "schema-utils": "^4.3.3", 1667 | "tapable": "^2.3.0", 1668 | "terser-webpack-plugin": "^5.3.11", 1669 | "watchpack": "^2.4.4", 1670 | "webpack-sources": "^3.3.3" 1671 | }, 1672 | "bin": { 1673 | "webpack": "bin/webpack.js" 1674 | }, 1675 | "engines": { 1676 | "node": ">=10.13.0" 1677 | }, 1678 | "funding": { 1679 | "type": "opencollective", 1680 | "url": "https://opencollective.com/webpack" 1681 | }, 1682 | "peerDependenciesMeta": { 1683 | "webpack-cli": { 1684 | "optional": true 1685 | } 1686 | } 1687 | }, 1688 | "node_modules/webpack-cli": { 1689 | "version": "6.0.1", 1690 | "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-6.0.1.tgz", 1691 | "integrity": "sha512-MfwFQ6SfwinsUVi0rNJm7rHZ31GyTcpVE5pgVA3hwFRb7COD4TzjUUwhGWKfO50+xdc2MQPuEBBJoqIMGt3JDw==", 1692 | "dev": true, 1693 | "license": "MIT", 1694 | "peer": true, 1695 | "dependencies": { 1696 | "@discoveryjs/json-ext": "^0.6.1", 1697 | "@webpack-cli/configtest": "^3.0.1", 1698 | "@webpack-cli/info": "^3.0.1", 1699 | "@webpack-cli/serve": "^3.0.1", 1700 | "colorette": "^2.0.14", 1701 | "commander": "^12.1.0", 1702 | "cross-spawn": "^7.0.3", 1703 | "envinfo": "^7.14.0", 1704 | "fastest-levenshtein": "^1.0.12", 1705 | "import-local": "^3.0.2", 1706 | "interpret": "^3.1.1", 1707 | "rechoir": "^0.8.0", 1708 | "webpack-merge": "^6.0.1" 1709 | }, 1710 | "bin": { 1711 | "webpack-cli": "bin/cli.js" 1712 | }, 1713 | "engines": { 1714 | "node": ">=18.12.0" 1715 | }, 1716 | "funding": { 1717 | "type": "opencollective", 1718 | "url": "https://opencollective.com/webpack" 1719 | }, 1720 | "peerDependencies": { 1721 | "webpack": "^5.82.0" 1722 | }, 1723 | "peerDependenciesMeta": { 1724 | "webpack-bundle-analyzer": { 1725 | "optional": true 1726 | }, 1727 | "webpack-dev-server": { 1728 | "optional": true 1729 | } 1730 | } 1731 | }, 1732 | "node_modules/webpack-cli/node_modules/commander": { 1733 | "version": "12.1.0", 1734 | "resolved": "https://registry.npmjs.org/commander/-/commander-12.1.0.tgz", 1735 | "integrity": "sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==", 1736 | "dev": true, 1737 | "license": "MIT", 1738 | "engines": { 1739 | "node": ">=18" 1740 | } 1741 | }, 1742 | "node_modules/webpack-merge": { 1743 | "version": "6.0.1", 1744 | "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-6.0.1.tgz", 1745 | "integrity": "sha512-hXXvrjtx2PLYx4qruKl+kyRSLc52V+cCvMxRjmKwoA+CBbbF5GfIBtR6kCvl0fYGqTUPKB+1ktVmTHqMOzgCBg==", 1746 | "dev": true, 1747 | "license": "MIT", 1748 | "dependencies": { 1749 | "clone-deep": "^4.0.1", 1750 | "flat": "^5.0.2", 1751 | "wildcard": "^2.0.1" 1752 | }, 1753 | "engines": { 1754 | "node": ">=18.0.0" 1755 | } 1756 | }, 1757 | "node_modules/webpack-sources": { 1758 | "version": "3.3.3", 1759 | "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.3.3.tgz", 1760 | "integrity": "sha512-yd1RBzSGanHkitROoPFd6qsrxt+oFhg/129YzheDGqeustzX0vTZJZsSsQjVQC4yzBQ56K55XU8gaNCtIzOnTg==", 1761 | "dev": true, 1762 | "license": "MIT", 1763 | "engines": { 1764 | "node": ">=10.13.0" 1765 | } 1766 | }, 1767 | "node_modules/which": { 1768 | "version": "2.0.2", 1769 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 1770 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 1771 | "dev": true, 1772 | "license": "ISC", 1773 | "dependencies": { 1774 | "isexe": "^2.0.0" 1775 | }, 1776 | "bin": { 1777 | "node-which": "bin/node-which" 1778 | }, 1779 | "engines": { 1780 | "node": ">= 8" 1781 | } 1782 | }, 1783 | "node_modules/wildcard": { 1784 | "version": "2.0.1", 1785 | "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.1.tgz", 1786 | "integrity": "sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ==", 1787 | "dev": true, 1788 | "license": "MIT" 1789 | } 1790 | } 1791 | } 1792 | --------------------------------------------------------------------------------