From 4a7eb9be31c8b1112306378a877922aca2d4b450 Mon Sep 17 00:00:00 2001 From: Jayprakash <0freerunning@gmail.com> Date: Tue, 10 Dec 2024 01:16:38 +0530 Subject: [PATCH] v2 version (#239) * Draft v2 version * Updated the code * Add Rating/Ranking/Edit vote with i18n * Added documentation * Move all message to i18n * Fixed i18n minor error * Add keyDown * Improved edit vote and other * Fixed documentation * Add comment to docker-compose.yml --------- Co-authored-by: Mahmoud Hashemi --- .dockerignore | 5 + Makefile | 14 + deployment_v2.md | 126 + dev_v2.md | 200 + docker-compose.yml | 13 + dockerfile | 12 + frontend/.eslintrc.cjs | 14 + frontend/.gitignore | 30 + frontend/.prettierrc.json | 8 + frontend/.vscode/extensions.json | 7 + frontend/README.md | 3 + frontend/index.html | 17 + frontend/jsconfig.json | 8 + frontend/package-lock.json | 3525 +++++++++++++++++ frontend/package.json | 43 + frontend/public/favicon.ico | Bin 0 -> 176289 bytes frontend/src/App.vue | 47 + frontend/src/assets/logo_white.svg | 71 + frontend/src/assets/main.css | 63 + frontend/src/components/AddOrganizer.vue | 78 + frontend/src/components/AppFooter.vue | 52 + frontend/src/components/AppHeader.vue | 101 + .../components/Campaign/ActiveCampaign.vue | 187 + .../src/components/Campaign/AllCampaign.vue | 153 + .../Campaign/CoordinatorCampaignCard.vue | 106 + .../components/Campaign/JurorCampaignCard.vue | 142 + .../src/components/Campaign/NewCampaign.vue | 221 ++ .../src/components/Campaign/ViewCampaign.vue | 321 ++ frontend/src/components/LoginBox.vue | 74 + frontend/src/components/Round/RoundEdit.vue | 182 + frontend/src/components/Round/RoundInfo.vue | 269 ++ frontend/src/components/Round/RoundNew.vue | 482 +++ frontend/src/components/Round/RoundView.vue | 103 + .../src/components/UserAvatarWithName.vue | 57 + frontend/src/components/UserList.vue | 52 + frontend/src/components/Vote/Vote.vue | 30 + frontend/src/components/Vote/VoteEdit.vue | 534 +++ frontend/src/components/Vote/VoteRanking.vue | 304 ++ frontend/src/components/Vote/VoteRating.vue | 591 +++ frontend/src/components/Vote/VoteYesNo.vue | 592 +++ frontend/src/i18n.js | 40 + frontend/src/i18n/en.json | 186 + frontend/src/i18n/hi.json | 186 + frontend/src/i18n/qqq.json | 186 + frontend/src/main.js | 28 + frontend/src/router/index.js | 68 + frontend/src/services/adminService.js | 56 + frontend/src/services/alertService.js | 28 + frontend/src/services/api.js | 60 + frontend/src/services/dataService.js | 75 + frontend/src/services/dialogService.js | 67 + frontend/src/services/jurorService.js | 58 + frontend/src/stores/loading.js | 12 + frontend/src/stores/user.js | 33 + frontend/src/utils.js | 63 + frontend/src/views/AllCampaignView.vue | 7 + frontend/src/views/CampaignView.vue | 9 + frontend/src/views/HomeView.vue | 14 + frontend/src/views/NewCampaignView.vue | 7 + frontend/src/views/VoteEditView.vue | 15 + frontend/src/views/VoteView.vue | 16 + frontend/vite.config.js | 16 + montage/rdb.py | 2 +- requirements.txt | 2 +- 64 files changed, 10069 insertions(+), 2 deletions(-) create mode 100644 .dockerignore create mode 100644 Makefile create mode 100644 deployment_v2.md create mode 100644 dev_v2.md create mode 100644 docker-compose.yml create mode 100644 dockerfile create mode 100644 frontend/.eslintrc.cjs create mode 100644 frontend/.gitignore create mode 100644 frontend/.prettierrc.json create mode 100644 frontend/.vscode/extensions.json create mode 100644 frontend/README.md create mode 100644 frontend/index.html create mode 100644 frontend/jsconfig.json create mode 100644 frontend/package-lock.json create mode 100644 frontend/package.json create mode 100644 frontend/public/favicon.ico create mode 100644 frontend/src/App.vue create mode 100644 frontend/src/assets/logo_white.svg create mode 100644 frontend/src/assets/main.css create mode 100644 frontend/src/components/AddOrganizer.vue create mode 100644 frontend/src/components/AppFooter.vue create mode 100644 frontend/src/components/AppHeader.vue create mode 100644 frontend/src/components/Campaign/ActiveCampaign.vue create mode 100644 frontend/src/components/Campaign/AllCampaign.vue create mode 100644 frontend/src/components/Campaign/CoordinatorCampaignCard.vue create mode 100644 frontend/src/components/Campaign/JurorCampaignCard.vue create mode 100644 frontend/src/components/Campaign/NewCampaign.vue create mode 100644 frontend/src/components/Campaign/ViewCampaign.vue create mode 100644 frontend/src/components/LoginBox.vue create mode 100644 frontend/src/components/Round/RoundEdit.vue create mode 100644 frontend/src/components/Round/RoundInfo.vue create mode 100644 frontend/src/components/Round/RoundNew.vue create mode 100644 frontend/src/components/Round/RoundView.vue create mode 100644 frontend/src/components/UserAvatarWithName.vue create mode 100644 frontend/src/components/UserList.vue create mode 100644 frontend/src/components/Vote/Vote.vue create mode 100644 frontend/src/components/Vote/VoteEdit.vue create mode 100644 frontend/src/components/Vote/VoteRanking.vue create mode 100644 frontend/src/components/Vote/VoteRating.vue create mode 100644 frontend/src/components/Vote/VoteYesNo.vue create mode 100644 frontend/src/i18n.js create mode 100644 frontend/src/i18n/en.json create mode 100644 frontend/src/i18n/hi.json create mode 100644 frontend/src/i18n/qqq.json create mode 100644 frontend/src/main.js create mode 100644 frontend/src/router/index.js create mode 100644 frontend/src/services/adminService.js create mode 100644 frontend/src/services/alertService.js create mode 100644 frontend/src/services/api.js create mode 100644 frontend/src/services/dataService.js create mode 100644 frontend/src/services/dialogService.js create mode 100644 frontend/src/services/jurorService.js create mode 100644 frontend/src/stores/loading.js create mode 100644 frontend/src/stores/user.js create mode 100644 frontend/src/utils.js create mode 100644 frontend/src/views/AllCampaignView.vue create mode 100644 frontend/src/views/CampaignView.vue create mode 100644 frontend/src/views/HomeView.vue create mode 100644 frontend/src/views/NewCampaignView.vue create mode 100644 frontend/src/views/VoteEditView.vue create mode 100644 frontend/src/views/VoteView.vue create mode 100644 frontend/vite.config.js diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 00000000..a1c09b22 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,5 @@ +frontend +client +test_data +fe +docs \ No newline at end of file diff --git a/Makefile b/Makefile new file mode 100644 index 00000000..6ebdd606 --- /dev/null +++ b/Makefile @@ -0,0 +1,14 @@ +start: + docker compose build && docker compose up + +start-detached: + docker compose build && docker compose up -d + +stop: + docker compose down + +logs: + docker compose logs -f + +restart: + docker compose down && docker compose up --build \ No newline at end of file diff --git a/deployment_v2.md b/deployment_v2.md new file mode 100644 index 00000000..d83c4713 --- /dev/null +++ b/deployment_v2.md @@ -0,0 +1,126 @@ +# Montage Deployment + +These are instructions for deploying Montage on Toolforge. + +## Deploying on Toolforge from scratch +These instructions is only first time when setuping project on Toolforge + +##### 1. Get the OAuth credentials. +[Register your app](https://meta.wikimedia.org/wiki/Special:OAuthConsumerRegistration/propose) and save your consumer token and secret token for later. + +##### 2. SSH to Toolforge and then inside tool +```bash +ssh @login.toolforge.org +become montage-beta +``` +Here, we are using `montage-beta` instance but it can be `montage` or `montage-dev` as well. + +##### 3. Clone the repo as src directory +```bash +mkdir -p $HOME/www/python +cd $HOME/www/python +git clone https://github.com/hatnote/montage.git src +``` + +##### 4. Make the frontend build +```bash +toolforge webservice node18 shell -m 2G +cd $HOME/www/python/src/frontend +npm install +npm run toolforge:build +exit +``` +This will build the vue prod bundle and put in backend's `template` and `static` directory. + +##### 5. Create your database +* Get the user name of database (`cat ~/replica.my.cnf`) +* Open up MariaDB with `sql local` +* Create a [Toolforge user database](https://wikitech.wikimedia.org/wiki/Help:Toolforge/Database#User_databases) (`create database __;`), and remember the name for the config + +##### 6. Set up the montage config +* Make a copy of `config.default.yaml` for your environment + * You may need to update `USER_ENV_MAP` in `montage/utils.py` if you need to add a new environment +* Add the `oauth_consumer_token` and `oauth_secret_token` +* Add a `cookie_secret: ` +* Add the `db_url` with your user database name, and the password from `~/.replica.my.cnf` + * The format is: `mysql://:@tools.labsdb/?charset=utf8` +* Add `api_log_path: /data/project//logs/montage_api.log` +* Add `replay_log_path: /data/project//logs/montage_replay.log` +* Add `labs_db: True` +* Add `db_echo: False` +* Add `root_path: '/'` + + +##### 7. Creating a virtual environment +```bash +toolforge webservice python3.9 shell +python3 -m venv $HOME/www/python/venv +source $HOME/www/python/venv/bin/activate +pip install --upgrade pip wheel +pip install -r $HOME/www/python/src/requirements.txt +exit +``` + +##### 8. Start the backend service +```bash +toolforge webservice python3.9 start +``` + +##### 9. Testing of deployment +* Visit /meta to see the API. Example: https://montage-beta.toolforge.org/meta/ +* In the top section, you should see that the service was restarted in the last few seconds/minutes. + + +--- + + +## Deploying new changes + +If montage is already deployed then you just need following to deploy new changes. + +##### 1. Check the instance usage +Login to the tool webapp. Make sure, you are maintainer on the webapp instance. Use the audit log endpoint to check that the instance isn't in active use. Example: https://montage-beta.toolforge.org/v1/logs/audit + +This will tell latest usage of instance by audit `create_date`. You can continue if instance is not being used. + +Sometimes, instance can in use, but there can be important bugfix and we can push anyways. + +##### 2. SSH to Toolforge and then inside tool +```bash +ssh @login.toolforge.org +become montage-beta +``` +Here, we are using `montage-beta` instance but it can be `montage` or `montage-dev` as well. + +##### 3. Get new changes from remote +```bash +cd $HOME/www/python/src +git pull +``` + +##### 4. Make the frontend build +```bash +toolforge webservice node18 shell -m 2G +cd $HOME/www/python/src/frontend +npm install +npm run toolforge:build +exit +``` + +##### 5. (Optional) Install python packages +If you added new python packages in changes then you have to install them in pod. +```bash +toolforge webservice python3.9 shell +source $HOME/www/python/venv/bin/activate +pip install -r $HOME/www/python/src/requirements.txt +exit +``` + +##### 8. Restart the backend service +```bash +toolforge webservice python3.9 restart +``` + +##### 9. Testing of deployment +* Visit /meta to see the API. Example: https://montage-beta.toolforge.org/meta/ +* In the top section, you should see that the service was restarted in the last few seconds/minutes. \ No newline at end of file diff --git a/dev_v2.md b/dev_v2.md new file mode 100644 index 00000000..22f2c749 --- /dev/null +++ b/dev_v2.md @@ -0,0 +1,200 @@ +# Developer Setup Guide + +Welcome to the **Montage Project**! This guide will help you set up the project for local development. + +--- + +## Overview + +The **Montage Project** is a web application with two main components: + +1. **Frontend**: + - Built with **Vue 3**. + - Includes **Wikimedia Codex** for UI components, **Axios** for API requests, and **Pinia** for state management. + - Compiled using **Vite** for fast builds. + +2. **Backend**: + - Built with **Clastic**, a Python framework. + - Uses various Python libraries such as: + - **SQLAlchemy**: Database interactions. + - **mwoauth**: Used for authentication with MediaWiki’s OAuth. + - **pymysql**: MySQL driver. + - Serves the frontend and exposes API endpoints for application functionality. + +--- + +## Prerequisites + +Ensure the following are installed: +- **Docker** and **Docker Compose**: [Install Docker](https://www.docker.com/products/docker-desktop). +- **Node.js** (v16 or above): [Install Node.js](https://nodejs.org). +- **Make**: Available on most Unix-based systems. + +--- + +## Setting Up the Project + +### 1. Clone the Repository +```bash +git clone git@github.com:hatnote/montage.git +cd montage +``` + +### 2. Start the Frontend Build Watcher +```bash +cd frontend +npm install +npm run watch:build +``` + +⚠️ **Frontend does not run in dev mode**: `npm run watch:build` above will watch for changes by using `chokidar` command and automatically copied to the backend's `static` and `template` directories. So there will not be any hot reloading. Developers must refresh the browser to see updates. + +Apart from `npm watch:build`, these are other commands: +* `npm run lint`: Lint the code +* `npm run format`: Format the code + +### 3. Use the Makefile to start the backend +* Open a new terminal tab and change directory to root of repo +* Copy and edit `config.dev.yaml` based on `config.default.yaml` +* (Optional) In `config.dev.yaml` there is a line for `dev_local_cookie_value`. To get it, +log in to the local app in your browser, and then copy the value from the +`clastic_cookie` in the apps' cookies. This is your login cookie. +* (Optional) Add your username as the `superuser` in the config. (This will allow you to +add `su_to=` to the backend, if you want to test submitting as another +juror.) +* Add your username to the list of maintainers in [rdb.py line 113](https://github.com/hatnote/montage/blob/master/montage/rdb.py#L113). +This will give your user top-level permissions in the full app, so you can view +some logs (audit logs, active users), add/remove organizers, and get a +coordinator view into all campaigns. +* Start the montage backend +```bash +make start +``` +This will build the docker image for the montage backend and start the container. Apart from `make start`, these are other commands: +* `make start-detached` : Start the backend container in detached mode +* `make stop` : Stop the backend container +* `make logs` : Stream the backend container logs in real-time. +* `make restart` : Restart the backend container + +### 4. Access the Application +* Open http://localhost:5001 in your browser. + +The application server runs on localhost port 5001, visit [http://localhost:5001/meta](http://localhost:5001/meta) to see a list +of valid URL patterns and other details. + +Almost all endpoints from backend (except for OAuth and `/static/`) return JSON as long as the proper Accept header is set (done by most libraries) or `format=json` is passed in the query string. + +## Project struture (v2 only) +```bash +├── DEV.md +├── Dockerfile +├── LICENSE +├── MANIFEST.in +├── Makefile +├── PROJECT_LOG.md +├── config +│   ├── beta-uwsgi.ini +│   ├── dev-uwsgi.ini +│   └── prod-uwsgi.ini +├── config.default.yaml +├── deployment.md +├── design.md +├── docker-compose.yml +├── frontend +│   ├── index.html +│   ├── jsconfig.json +│   ├── package-lock.json +│   ├── package.json +│   ├── public +│   ├── src +│   └── vite.config.js +├── montage +│   ├── __init__.py +│   ├── __main__.py +│   ├── __pycache__ +│   ├── admin_endpoints.py +│   ├── app.py +│   ├── check_rdb.py +│   ├── clastic_sentry.py +│   ├── docs +│   ├── imgutils.py +│   ├── juror_endpoints.py +│   ├── labs.py +│   ├── loaders.py +│   ├── log.py +│   ├── meta_endpoints.py +│   ├── mw +│   ├── public_endpoints.py +│   ├── rdb.py +│   ├── rendered_admin.py +│   ├── server.py +│   ├── simple_serdes.py +│   ├── static +│   ├── templates +│   ├── tests +│   └── utils.py +├── report.html +├── requirements-dev.txt +├── requirements.in +├── requirements.txt +├── setup.py +├── tools +│   ├── _admin.py +│   ├── admin.py +│   ├── check_schema.py +│   ├── create_schema.py +│   ├── drop_schema.py +│   └── trim_csv.py +└── tox.ini +``` +These provides a detailed explanation of the main components in the **Montage Project**. + +#### Directory: `montage` +##### **Core Files** +- **`app.py`**: Initializes the application and defines middleware, routes, or app-wide configurations. +- **`server.py`**: Starts the server, setting up the backend to listen on a specific port. +- **`__main__.py`**: Entry point when running the backend module. + +##### **API Endpoints** +- **`admin_endpoints.py`**: Handles admin-specific routes (e.g., user management, settings). +- **`juror_endpoints.py`**: Contains juror-related APIs (e.g., task assignments, voting). +- **`meta_endpoints.py`**: General application metadata or system information. +- **`public_endpoints.py`**: APIs accessible to public or unauthenticated users. + +##### **Utilities** +- **`imgutils.py`**: Handles image processing and manipulation. +- **`simple_serdes.py`**: Serializes and deserializes objects to JSON or other formats. +- **`log.py`**: Configures application logging. + +##### **Database** +- **`rdb.py`**: Manages database interactions (queries, migrations, etc.). +- **`check_rdb.py`**: Verifies database schema integrity. + +##### **Static and Templates** +- **`static/`**: Holds CSS, JavaScript, and other assets. +- **`templates/`**: Contains Jinja2 templates for dynamic HTML rendering. + +##### **Testing** +- **`tests/`**: Basic tests for backend components. + + +#### Directory: `frontend` + +##### **Core Files** +- **`src/`**: Source code, including components, routes, and utilities. +- **`public/`**: Static assets, such as images and global styles. +- **`vite.config.js`**: Configuration for the Vite build tool. + + +#### Directory: `tools` + +##### **Key Scripts** +- **`create_schema.py`**: Creates database tables. +- **`drop_schema.py`**: Drops all database tables. +- **`check_schema.py`**: Verifies schema correctness. +- **`trim_csv.py`**: Utility for cleaning CSV files + + +#### Docker Files +- **`Dockerfile`**: Builds the backend container. +- **`docker-compose.yml`**: Orchestrates service for backend. \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 00000000..fe620895 --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,13 @@ +services: + montage: + build: + context: . + dockerfile: Dockerfile + ports: + - "5001:5000" # Airplay runs on port 5000 on mac. See https://forums.developer.apple.com/forums/thread/682332 + environment: + - PYTHONPATH=/app + volumes: + - .:/app + command: > + bash -c "python tools/create_schema.py && python -m montage" diff --git a/dockerfile b/dockerfile new file mode 100644 index 00000000..b810809f --- /dev/null +++ b/dockerfile @@ -0,0 +1,12 @@ +FROM python:3.9-slim + +RUN apt-get update && apt-get install -y git + +WORKDIR /app + +COPY requirements.txt . + +RUN pip install --upgrade pip +RUN pip install -r requirements.txt + +EXPOSE 5000 diff --git a/frontend/.eslintrc.cjs b/frontend/.eslintrc.cjs new file mode 100644 index 00000000..b64731a0 --- /dev/null +++ b/frontend/.eslintrc.cjs @@ -0,0 +1,14 @@ +/* eslint-env node */ +require('@rushstack/eslint-patch/modern-module-resolution') + +module.exports = { + root: true, + 'extends': [ + 'plugin:vue/vue3-essential', + 'eslint:recommended', + '@vue/eslint-config-prettier/skip-formatting' + ], + parserOptions: { + ecmaVersion: 'latest' + } +} diff --git a/frontend/.gitignore b/frontend/.gitignore new file mode 100644 index 00000000..8ee54e8d --- /dev/null +++ b/frontend/.gitignore @@ -0,0 +1,30 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +node_modules +.DS_Store +dist +dist-ssr +coverage +*.local + +/cypress/videos/ +/cypress/screenshots/ + +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? + +*.tsbuildinfo diff --git a/frontend/.prettierrc.json b/frontend/.prettierrc.json new file mode 100644 index 00000000..66e23359 --- /dev/null +++ b/frontend/.prettierrc.json @@ -0,0 +1,8 @@ +{ + "$schema": "https://json.schemastore.org/prettierrc", + "semi": false, + "tabWidth": 2, + "singleQuote": true, + "printWidth": 100, + "trailingComma": "none" +} \ No newline at end of file diff --git a/frontend/.vscode/extensions.json b/frontend/.vscode/extensions.json new file mode 100644 index 00000000..93ea3e78 --- /dev/null +++ b/frontend/.vscode/extensions.json @@ -0,0 +1,7 @@ +{ + "recommendations": [ + "Vue.volar", + "dbaeumer.vscode-eslint", + "esbenp.prettier-vscode" + ] +} diff --git a/frontend/README.md b/frontend/README.md new file mode 100644 index 00000000..056dc7cf --- /dev/null +++ b/frontend/README.md @@ -0,0 +1,3 @@ +# Montage-frontend + +This is montage frontend \ No newline at end of file diff --git a/frontend/index.html b/frontend/index.html new file mode 100644 index 00000000..6cb0fa21 --- /dev/null +++ b/frontend/index.html @@ -0,0 +1,17 @@ + + + + + + + Montage + + + +
+ + + diff --git a/frontend/jsconfig.json b/frontend/jsconfig.json new file mode 100644 index 00000000..f2642f32 --- /dev/null +++ b/frontend/jsconfig.json @@ -0,0 +1,8 @@ +{ + "compilerOptions": { + "paths": { + "@/*": ["./src/*"] + } + }, + "exclude": ["node_modules", "dist"] +} \ No newline at end of file diff --git a/frontend/package-lock.json b/frontend/package-lock.json new file mode 100644 index 00000000..3a75c344 --- /dev/null +++ b/frontend/package-lock.json @@ -0,0 +1,3525 @@ +{ + "name": "montage-frontend", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "montage-frontend", + "version": "0.0.0", + "dependencies": { + "@wikimedia/codex": "^1.14.0", + "@wikimedia/codex-icons": "^1.14.0", + "axios": "^1.7.7", + "dayjs": "^1.11.13", + "iso-639-1": "^3.1.3", + "lodash": "^4.17.21", + "pinia": "^2.1.7", + "vue": "^3.4.27", + "vue-draggable-next": "^2.2.1", + "vue-i18n": "^10.0.4", + "vue-material-design-icons": "^5.3.0", + "vue-router": "^4.3.3", + "vue-spinner": "^1.0.4", + "vue-toastification": "^2.0.0-rc.5", + "vuedraggable": "^2.24.3", + "zod": "^3.23.8" + }, + "devDependencies": { + "@rushstack/eslint-patch": "^1.8.0", + "@vitejs/plugin-vue": "^5.0.5", + "@vue/eslint-config-prettier": "^9.0.0", + "chokidar-cli": "^3.0.0", + "eslint": "^8.57.0", + "eslint-plugin-vue": "^9.23.0", + "prettier": "^3.2.5", + "vite": "^5.3.1" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.8.tgz", + "integrity": "sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz", + "integrity": "sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.25.6", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.25.6.tgz", + "integrity": "sha512-trGdfBdbD0l1ZPmcJ83eNxB9rbEax4ALFTF7fN386TMYbeCQbyme5cOEXQhbGXKebwGaB/J52w1mrklMcbgy6Q==", + "license": "MIT", + "dependencies": { + "@babel/types": "^7.25.6" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/types": { + "version": "7.25.6", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.25.6.tgz", + "integrity": "sha512-/l42B1qxpG6RdfYf343Uw1vmDjeNhneUXtzhojE7pDgfpEypmRhI6j1kr17XCVv4Cgl9HdAiQY2x0GwKm7rWCw==", + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.24.8", + "@babel/helper-validator-identifier": "^7.24.7", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", + "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", + "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", + "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", + "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", + "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", + "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", + "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", + "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", + "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", + "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", + "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", + "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", + "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", + "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", + "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", + "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", + "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", + "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", + "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", + "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", + "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", + "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", + "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.11.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.11.1.tgz", + "integrity": "sha512-m4DVN9ZqskZoLU5GlWZadwDnYo3vAEydiUayB9widCl9ffWx2IvPnp6n3on5rJmziJSw9Bv+Z3ChDVdMwXCY8Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/js": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", + "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@floating-ui/core": { + "version": "1.6.8", + "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.6.8.tgz", + "integrity": "sha512-7XJ9cPU+yI2QeLS+FCSlqNFZJq8arvswefkZrYI1yQBbftw6FyrZOxYSh+9S7z7TpeWlRt9zJ5IhM1WIL334jA==", + "license": "MIT", + "dependencies": { + "@floating-ui/utils": "^0.2.8" + } + }, + "node_modules/@floating-ui/dom": { + "version": "1.6.11", + "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.6.11.tgz", + "integrity": "sha512-qkMCxSR24v2vGkhYDo/UzxfJN3D4syqSjyuTFz6C7XcpU1pASPRieNI0Kj5VP3/503mOfYiGY891ugBX1GlABQ==", + "license": "MIT", + "dependencies": { + "@floating-ui/core": "^1.6.0", + "@floating-ui/utils": "^0.2.8" + } + }, + "node_modules/@floating-ui/utils": { + "version": "0.2.8", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.8.tgz", + "integrity": "sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==", + "license": "MIT" + }, + "node_modules/@floating-ui/vue": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@floating-ui/vue/-/vue-1.0.6.tgz", + "integrity": "sha512-EdrOljjkpkkqZnrpqUcPoz9NvHxuTjUtSInh6GMv3+Mcy+giY2cE2pHh9rpacRcZ2eMSCxel9jWkWXTjLmY55w==", + "license": "MIT", + "dependencies": { + "@floating-ui/dom": "^1.6.1", + "@floating-ui/utils": "^0.2.1", + "vue-demi": ">=0.13.0" + } + }, + "node_modules/@floating-ui/vue/node_modules/vue-demi": { + "version": "0.14.10", + "resolved": "https://registry.npmjs.org/vue-demi/-/vue-demi-0.14.10.tgz", + "integrity": "sha512-nMZBOwuzabUO0nLgIcc6rycZEebF6eeUfaiQx9+WSk8e29IbLvPU9feI6tqW4kTo3hvoYAJkMh8n8D0fuISphg==", + "hasInstallScript": true, + "license": "MIT", + "bin": { + "vue-demi-fix": "bin/vue-demi-fix.js", + "vue-demi-switch": "bin/vue-demi-switch.js" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/antfu" + }, + "peerDependencies": { + "@vue/composition-api": "^1.0.0-rc.1", + "vue": "^3.0.0-0 || ^2.6.0" + }, + "peerDependenciesMeta": { + "@vue/composition-api": { + "optional": true + } + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", + "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", + "deprecated": "Use @eslint/config-array instead", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanwhocodes/object-schema": "^2.0.3", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "deprecated": "Use @eslint/object-schema instead", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/@intlify/core-base": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/@intlify/core-base/-/core-base-10.0.4.tgz", + "integrity": "sha512-GG428DkrrWCMhxRMRQZjuS7zmSUzarYcaHJqG9VB8dXAxw4iQDoKVQ7ChJRB6ZtsCsX3Jse1PEUlHrJiyQrOTg==", + "license": "MIT", + "dependencies": { + "@intlify/message-compiler": "10.0.4", + "@intlify/shared": "10.0.4" + }, + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://github.com/sponsors/kazupon" + } + }, + "node_modules/@intlify/message-compiler": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/@intlify/message-compiler/-/message-compiler-10.0.4.tgz", + "integrity": "sha512-AFbhEo10DP095/45EauinQJ5hJ3rJUmuuqltGguvc3WsvezZN+g8qNHLGWKu60FHQVizMrQY7VJ+zVlBXlQQkQ==", + "license": "MIT", + "dependencies": { + "@intlify/shared": "10.0.4", + "source-map-js": "^1.0.2" + }, + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://github.com/sponsors/kazupon" + } + }, + "node_modules/@intlify/shared": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/@intlify/shared/-/shared-10.0.4.tgz", + "integrity": "sha512-ukFn0I01HsSgr3VYhYcvkTCLS7rGa0gw4A4AMpcy/A9xx/zRJy7PS2BElMXLwUazVFMAr5zuiTk3MQeoeGXaJg==", + "license": "MIT", + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://github.com/sponsors/kazupon" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", + "license": "MIT" + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@pkgr/core": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.1.1.tgz", + "integrity": "sha512-cq8o4cWH0ibXh9VGi5P20Tu9XF/0fFXl9EUinr9QfTM7a7p0oTA4iJRCQWppXR1Pg8dSM0UCItCkPwsk9qWWYA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/unts" + } + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.21.3.tgz", + "integrity": "sha512-MmKSfaB9GX+zXl6E8z4koOr/xU63AMVleLEa64v7R0QF/ZloMs5vcD1sHgM64GXXS1csaJutG+ddtzcueI/BLg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.21.3.tgz", + "integrity": "sha512-zrt8ecH07PE3sB4jPOggweBjJMzI1JG5xI2DIsUbkA+7K+Gkjys6eV7i9pOenNSDJH3eOr/jLb/PzqtmdwDq5g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.21.3.tgz", + "integrity": "sha512-P0UxIOrKNBFTQaXTxOH4RxuEBVCgEA5UTNV6Yz7z9QHnUJ7eLX9reOd/NYMO3+XZO2cco19mXTxDMXxit4R/eQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.21.3.tgz", + "integrity": "sha512-L1M0vKGO5ASKntqtsFEjTq/fD91vAqnzeaF6sfNAy55aD+Hi2pBI5DKwCO+UNDQHWsDViJLqshxOahXyLSh3EA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.21.3.tgz", + "integrity": "sha512-btVgIsCjuYFKUjopPoWiDqmoUXQDiW2A4C3Mtmp5vACm7/GnyuprqIDPNczeyR5W8rTXEbkmrJux7cJmD99D2g==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.21.3.tgz", + "integrity": "sha512-zmjbSphplZlau6ZTkxd3+NMtE4UKVy7U4aVFMmHcgO5CUbw17ZP6QCgyxhzGaU/wFFdTfiojjbLG3/0p9HhAqA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.21.3.tgz", + "integrity": "sha512-nSZfcZtAnQPRZmUkUQwZq2OjQciR6tEoJaZVFvLHsj0MF6QhNMg0fQ6mUOsiCUpTqxTx0/O6gX0V/nYc7LrgPw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.21.3.tgz", + "integrity": "sha512-MnvSPGO8KJXIMGlQDYfvYS3IosFN2rKsvxRpPO2l2cum+Z3exiExLwVU+GExL96pn8IP+GdH8Tz70EpBhO0sIQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.21.3.tgz", + "integrity": "sha512-+W+p/9QNDr2vE2AXU0qIy0qQE75E8RTwTwgqS2G5CRQ11vzq0tbnfBd6brWhS9bCRjAjepJe2fvvkvS3dno+iw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.21.3.tgz", + "integrity": "sha512-yXH6K6KfqGXaxHrtr+Uoy+JpNlUlI46BKVyonGiaD74ravdnF9BUNC+vV+SIuB96hUMGShhKV693rF9QDfO6nQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.21.3.tgz", + "integrity": "sha512-R8cwY9wcnApN/KDYWTH4gV/ypvy9yZUHlbJvfaiXSB48JO3KpwSpjOGqO4jnGkLDSk1hgjYkTbTt6Q7uvPf8eg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.21.3.tgz", + "integrity": "sha512-kZPbX/NOPh0vhS5sI+dR8L1bU2cSO9FgxwM8r7wHzGydzfSjLRCFAT87GR5U9scj2rhzN3JPYVC7NoBbl4FZ0g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.21.3.tgz", + "integrity": "sha512-S0Yq+xA1VEH66uiMNhijsWAafffydd2X5b77eLHfRmfLsRSpbiAWiRHV6DEpz6aOToPsgid7TI9rGd6zB1rhbg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.21.3.tgz", + "integrity": "sha512-9isNzeL34yquCPyerog+IMCNxKR8XYmGd0tHSV+OVx0TmE0aJOo9uw4fZfUuk2qxobP5sug6vNdZR6u7Mw7Q+Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.21.3.tgz", + "integrity": "sha512-nMIdKnfZfzn1Vsk+RuOvl43ONTZXoAPUUxgcU0tXooqg4YrAqzfKzVenqqk2g5efWh46/D28cKFrOzDSW28gTA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.21.3.tgz", + "integrity": "sha512-fOvu7PCQjAj4eWDEuD8Xz5gpzFqXzGlxHZozHP4b9Jxv9APtdxL6STqztDzMLuRXEc4UpXGGhx029Xgm91QBeA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rushstack/eslint-patch": { + "version": "1.10.4", + "resolved": "https://registry.npmjs.org/@rushstack/eslint-patch/-/eslint-patch-1.10.4.tgz", + "integrity": "sha512-WJgX9nzTqknM393q1QJDJmoW28kUfEnybeTfVNcNAPnIx210RXm2DiXiHzfNPJNIUUb1tJnz/l4QGtJ30PgWmA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@ungap/structured-clone": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", + "integrity": "sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/@vitejs/plugin-vue": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-vue/-/plugin-vue-5.1.3.tgz", + "integrity": "sha512-3xbWsKEKXYlmX82aOHufFQVnkbMC/v8fLpWwh6hWOUrK5fbbtBh9Q/WWse27BFgSy2/e2c0fz5Scgya9h2GLhw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "peerDependencies": { + "vite": "^5.0.0", + "vue": "^3.2.25" + } + }, + "node_modules/@vue/compiler-core": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.4.27.tgz", + "integrity": "sha512-E+RyqY24KnyDXsCuQrI+mlcdW3ALND6U7Gqa/+bVwbcpcR3BRRIckFoz7Qyd4TTlnugtwuI7YgjbvsLmxb+yvg==", + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.24.4", + "@vue/shared": "3.4.27", + "entities": "^4.5.0", + "estree-walker": "^2.0.2", + "source-map-js": "^1.2.0" + } + }, + "node_modules/@vue/compiler-dom": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/compiler-dom/-/compiler-dom-3.4.27.tgz", + "integrity": "sha512-kUTvochG/oVgE1w5ViSr3KUBh9X7CWirebA3bezTbB5ZKBQZwR2Mwj9uoSKRMFcz4gSMzzLXBPD6KpCLb9nvWw==", + "license": "MIT", + "dependencies": { + "@vue/compiler-core": "3.4.27", + "@vue/shared": "3.4.27" + } + }, + "node_modules/@vue/compiler-sfc": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/compiler-sfc/-/compiler-sfc-3.4.27.tgz", + "integrity": "sha512-nDwntUEADssW8e0rrmE0+OrONwmRlegDA1pD6QhVeXxjIytV03yDqTey9SBDiALsvAd5U4ZrEKbMyVXhX6mCGA==", + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.24.4", + "@vue/compiler-core": "3.4.27", + "@vue/compiler-dom": "3.4.27", + "@vue/compiler-ssr": "3.4.27", + "@vue/shared": "3.4.27", + "estree-walker": "^2.0.2", + "magic-string": "^0.30.10", + "postcss": "^8.4.38", + "source-map-js": "^1.2.0" + } + }, + "node_modules/@vue/compiler-ssr": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/compiler-ssr/-/compiler-ssr-3.4.27.tgz", + "integrity": "sha512-CVRzSJIltzMG5FcidsW0jKNQnNRYC8bT21VegyMMtHmhW3UOI7knmUehzswXLrExDLE6lQCZdrhD4ogI7c+vuw==", + "license": "MIT", + "dependencies": { + "@vue/compiler-dom": "3.4.27", + "@vue/shared": "3.4.27" + } + }, + "node_modules/@vue/devtools-api": { + "version": "6.6.4", + "resolved": "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-6.6.4.tgz", + "integrity": "sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g==", + "license": "MIT" + }, + "node_modules/@vue/eslint-config-prettier": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/@vue/eslint-config-prettier/-/eslint-config-prettier-9.0.0.tgz", + "integrity": "sha512-z1ZIAAUS9pKzo/ANEfd2sO+v2IUalz7cM/cTLOZ7vRFOPk5/xuRKQteOu1DErFLAh/lYGXMVZ0IfYKlyInuDVg==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-config-prettier": "^9.0.0", + "eslint-plugin-prettier": "^5.0.0" + }, + "peerDependencies": { + "eslint": ">= 8.0.0", + "prettier": ">= 3.0.0" + } + }, + "node_modules/@vue/reactivity": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/reactivity/-/reactivity-3.4.27.tgz", + "integrity": "sha512-kK0g4NknW6JX2yySLpsm2jlunZJl2/RJGZ0H9ddHdfBVHcNzxmQ0sS0b09ipmBoQpY8JM2KmUw+a6sO8Zo+zIA==", + "license": "MIT", + "dependencies": { + "@vue/shared": "3.4.27" + } + }, + "node_modules/@vue/runtime-core": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/runtime-core/-/runtime-core-3.4.27.tgz", + "integrity": "sha512-7aYA9GEbOOdviqVvcuweTLe5Za4qBZkUY7SvET6vE8kyypxVgaT1ixHLg4urtOlrApdgcdgHoTZCUuTGap/5WA==", + "license": "MIT", + "dependencies": { + "@vue/reactivity": "3.4.27", + "@vue/shared": "3.4.27" + } + }, + "node_modules/@vue/runtime-dom": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/runtime-dom/-/runtime-dom-3.4.27.tgz", + "integrity": "sha512-ScOmP70/3NPM+TW9hvVAz6VWWtZJqkbdf7w6ySsws+EsqtHvkhxaWLecrTorFxsawelM5Ys9FnDEMt6BPBDS0Q==", + "license": "MIT", + "dependencies": { + "@vue/runtime-core": "3.4.27", + "@vue/shared": "3.4.27", + "csstype": "^3.1.3" + } + }, + "node_modules/@vue/server-renderer": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/server-renderer/-/server-renderer-3.4.27.tgz", + "integrity": "sha512-dlAMEuvmeA3rJsOMJ2J1kXU7o7pOxgsNHVr9K8hB3ImIkSuBrIdy0vF66h8gf8Tuinf1TK3mPAz2+2sqyf3KzA==", + "license": "MIT", + "dependencies": { + "@vue/compiler-ssr": "3.4.27", + "@vue/shared": "3.4.27" + }, + "peerDependencies": { + "vue": "3.4.27" + } + }, + "node_modules/@vue/shared": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.4.27.tgz", + "integrity": "sha512-DL3NmY2OFlqmYYrzp39yi3LDkKxa5vZVwxWdQ3rG0ekuWscHraeIbnI8t+aZK7qhYqEqWKTUdijadunb9pnrgA==", + "license": "MIT" + }, + "node_modules/@wikimedia/codex": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/@wikimedia/codex/-/codex-1.14.0.tgz", + "integrity": "sha512-BDoqpzcRN0tCWSIYtvb4ukFeSYStz09UZ2VkNPJ/YgtpmcSnFdpEcbL29k1W/BVntuL9XAkv88s0OHKoNX1ikA==", + "license": "GPL-2.0+", + "dependencies": { + "@floating-ui/vue": "1.0.6", + "@wikimedia/codex-icons": "1.14.0" + }, + "engines": { + "node": ">=18", + "npm": ">=7.21.0" + }, + "peerDependencies": { + "vue": "3.4.27" + } + }, + "node_modules/@wikimedia/codex-icons": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/@wikimedia/codex-icons/-/codex-icons-1.14.0.tgz", + "integrity": "sha512-KbevrFz4Z5Jw/zadfN3UxDs2owZ9inYP0MqARE4LA3NOWkJpD76e+58KABmAnSUtZ2swMi94CFWex9tODDFg0w==", + "license": "MIT", + "engines": { + "node": ">=18", + "npm": ">=7.21.0" + } + }, + "node_modules/acorn": { + "version": "8.12.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz", + "integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==", + "dev": true, + "license": "MIT", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true, + "license": "Python-2.0" + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "license": "MIT" + }, + "node_modules/axios": { + "version": "1.7.7", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.7.tgz", + "integrity": "sha512-S4kL7XrjgBmvdGut0sN3yJxqYzrDOnivkBiN0OFs6hLiUam3UPvswUo0kqGyhqUZGEOytHyumEdXsAkgCOUf3Q==", + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/binary-extensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==", + "dev": true, + "license": "ISC" + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chokidar-cli": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chokidar-cli/-/chokidar-cli-3.0.0.tgz", + "integrity": "sha512-xVW+Qeh7z15uZRxHOkP93Ux8A0xbPzwK4GaqD8dQOYc34TlkqUhVSS59fK36DOp5WdJlrRzlYSy02Ht99FjZqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "chokidar": "^3.5.2", + "lodash.debounce": "^4.0.8", + "lodash.throttle": "^4.1.1", + "yargs": "^13.3.0" + }, + "bin": { + "chokidar": "index.js" + }, + "engines": { + "node": ">= 8.10.0" + } + }, + "node_modules/chokidar/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" + } + }, + "node_modules/cliui/node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true, + "license": "MIT" + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "license": "MIT", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "dev": true, + "license": "MIT", + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/csstype": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "license": "MIT" + }, + "node_modules/dayjs": { + "version": "1.11.13", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.13.tgz", + "integrity": "sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==", + "license": "MIT" + }, + "node_modules/debug": { + "version": "4.3.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", + "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "license": "MIT", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true, + "license": "MIT" + }, + "node_modules/entities": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/esbuild": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", + "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.21.5", + "@esbuild/android-arm": "0.21.5", + "@esbuild/android-arm64": "0.21.5", + "@esbuild/android-x64": "0.21.5", + "@esbuild/darwin-arm64": "0.21.5", + "@esbuild/darwin-x64": "0.21.5", + "@esbuild/freebsd-arm64": "0.21.5", + "@esbuild/freebsd-x64": "0.21.5", + "@esbuild/linux-arm": "0.21.5", + "@esbuild/linux-arm64": "0.21.5", + "@esbuild/linux-ia32": "0.21.5", + "@esbuild/linux-loong64": "0.21.5", + "@esbuild/linux-mips64el": "0.21.5", + "@esbuild/linux-ppc64": "0.21.5", + "@esbuild/linux-riscv64": "0.21.5", + "@esbuild/linux-s390x": "0.21.5", + "@esbuild/linux-x64": "0.21.5", + "@esbuild/netbsd-x64": "0.21.5", + "@esbuild/openbsd-x64": "0.21.5", + "@esbuild/sunos-x64": "0.21.5", + "@esbuild/win32-arm64": "0.21.5", + "@esbuild/win32-ia32": "0.21.5", + "@esbuild/win32-x64": "0.21.5" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", + "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.4", + "@eslint/js": "8.57.1", + "@humanwhocodes/config-array": "^0.13.0", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-config-prettier": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.0.tgz", + "integrity": "sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==", + "dev": true, + "license": "MIT", + "bin": { + "eslint-config-prettier": "bin/cli.js" + }, + "peerDependencies": { + "eslint": ">=7.0.0" + } + }, + "node_modules/eslint-plugin-prettier": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.2.1.tgz", + "integrity": "sha512-gH3iR3g4JfF+yYPaJYkN7jEl9QbweL/YfkoRlNnuIEHEz1vHVlCmWOS+eGGiRuzHQXdJFCOTxRgvju9b8VUmrw==", + "dev": true, + "license": "MIT", + "dependencies": { + "prettier-linter-helpers": "^1.0.0", + "synckit": "^0.9.1" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint-plugin-prettier" + }, + "peerDependencies": { + "@types/eslint": ">=8.0.0", + "eslint": ">=8.0.0", + "eslint-config-prettier": "*", + "prettier": ">=3.0.0" + }, + "peerDependenciesMeta": { + "@types/eslint": { + "optional": true + }, + "eslint-config-prettier": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-vue": { + "version": "9.28.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-vue/-/eslint-plugin-vue-9.28.0.tgz", + "integrity": "sha512-ShrihdjIhOTxs+MfWun6oJWuk+g/LAhN+CiuOl/jjkG3l0F2AuK5NMTaWqyvBgkFtpYmyks6P4603mLmhNJW8g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "globals": "^13.24.0", + "natural-compare": "^1.4.0", + "nth-check": "^2.1.1", + "postcss-selector-parser": "^6.0.15", + "semver": "^7.6.3", + "vue-eslint-parser": "^9.4.3", + "xml-name-validator": "^4.0.0" + }, + "engines": { + "node": "^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "eslint": "^6.2.0 || ^7.0.0 || ^8.0.0 || ^9.0.0" + } + }, + "node_modules/eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "license": "MIT" + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-diff": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz", + "integrity": "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==", + "dev": true, + "license": "Apache-2.0" + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fastq": { + "version": "1.17.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dev": true, + "license": "ISC", + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true, + "license": "ISC" + }, + "node_modules/follow-redirects": { + "version": "1.15.9", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", + "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/form-data": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", + "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true, + "license": "MIT" + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "license": "MIT", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/iso-639-1": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/iso-639-1/-/iso-639-1-3.1.3.tgz", + "integrity": "sha512-1jz0Wh9hyLMRwqEPchb/KZCiTqfFWtc9R3nm7GHPygBAKS8wdKJ3FH4lvLsri6UtAE5Kz5SnowtXZa//6bqMyw==", + "license": "MIT", + "engines": { + "node": ">=6.0" + } + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "license": "MIT" + }, + "node_modules/lodash.debounce": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", + "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.throttle": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz", + "integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/magic-string": { + "version": "0.30.11", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.11.tgz", + "integrity": "sha512-+Wri9p0QHMy+545hKww7YAu5NyzF8iomPL/RQazugQ9+Ez4Ic3mERMd8ZTX5rfK944j+560ZJi8iAwgak1Ac7A==", + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/nth-check": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", + "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "boolbase": "^1.0.0" + }, + "funding": { + "url": "https://github.com/fb55/nth-check?sponsor=1" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.0.tgz", + "integrity": "sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==", + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pinia": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/pinia/-/pinia-2.2.2.tgz", + "integrity": "sha512-ja2XqFWZC36mupU4z1ZzxeTApV7DOw44cV4dhQ9sGwun+N89v/XP7+j7q6TanS1u1tdbK4r+1BUx7heMaIdagA==", + "license": "MIT", + "dependencies": { + "@vue/devtools-api": "^6.6.3", + "vue-demi": "^0.14.10" + }, + "funding": { + "url": "https://github.com/sponsors/posva" + }, + "peerDependencies": { + "@vue/composition-api": "^1.4.0", + "typescript": ">=4.4.4", + "vue": "^2.6.14 || ^3.3.0" + }, + "peerDependenciesMeta": { + "@vue/composition-api": { + "optional": true + }, + "typescript": { + "optional": true + } + } + }, + "node_modules/pinia/node_modules/vue-demi": { + "version": "0.14.10", + "resolved": "https://registry.npmjs.org/vue-demi/-/vue-demi-0.14.10.tgz", + "integrity": "sha512-nMZBOwuzabUO0nLgIcc6rycZEebF6eeUfaiQx9+WSk8e29IbLvPU9feI6tqW4kTo3hvoYAJkMh8n8D0fuISphg==", + "hasInstallScript": true, + "license": "MIT", + "bin": { + "vue-demi-fix": "bin/vue-demi-fix.js", + "vue-demi-switch": "bin/vue-demi-switch.js" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/antfu" + }, + "peerDependencies": { + "@vue/composition-api": "^1.0.0-rc.1", + "vue": "^3.0.0-0 || ^2.6.0" + }, + "peerDependenciesMeta": { + "@vue/composition-api": { + "optional": true + } + } + }, + "node_modules/postcss": { + "version": "8.4.47", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.47.tgz", + "integrity": "sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==", + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "nanoid": "^3.3.7", + "picocolors": "^1.1.0", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/postcss-selector-parser": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz", + "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz", + "integrity": "sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==", + "dev": true, + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/prettier-linter-helpers": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", + "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-diff": "^1.1.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "license": "MIT" + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-main-filename": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", + "dev": true, + "license": "ISC" + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rollup": { + "version": "4.21.3", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.21.3.tgz", + "integrity": "sha512-7sqRtBNnEbcBtMeRVc6VRsJMmpI+JU1z9VTvW8D4gXIYQFz0aLcsE6rRkyghZkLfEgUZgVvOG7A5CVz/VW5GIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.5" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.21.3", + "@rollup/rollup-android-arm64": "4.21.3", + "@rollup/rollup-darwin-arm64": "4.21.3", + "@rollup/rollup-darwin-x64": "4.21.3", + "@rollup/rollup-linux-arm-gnueabihf": "4.21.3", + "@rollup/rollup-linux-arm-musleabihf": "4.21.3", + "@rollup/rollup-linux-arm64-gnu": "4.21.3", + "@rollup/rollup-linux-arm64-musl": "4.21.3", + "@rollup/rollup-linux-powerpc64le-gnu": "4.21.3", + "@rollup/rollup-linux-riscv64-gnu": "4.21.3", + "@rollup/rollup-linux-s390x-gnu": "4.21.3", + "@rollup/rollup-linux-x64-gnu": "4.21.3", + "@rollup/rollup-linux-x64-musl": "4.21.3", + "@rollup/rollup-win32-arm64-msvc": "4.21.3", + "@rollup/rollup-win32-ia32-msvc": "4.21.3", + "@rollup/rollup-win32-x64-msvc": "4.21.3", + "fsevents": "~2.3.2" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/set-blocking": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==", + "dev": true, + "license": "ISC" + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/sortablejs": { + "version": "1.15.3", + "resolved": "https://registry.npmjs.org/sortablejs/-/sortablejs-1.15.3.tgz", + "integrity": "sha512-zdK3/kwwAK1cJgy1rwl1YtNTbRmc8qW/+vgXf75A7NHag5of4pyI6uK86ktmQETyWRH7IGaE73uZOOBcGxgqZg==", + "license": "MIT", + "peer": true + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/string-width/node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/string-width/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/synckit": { + "version": "0.9.1", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.9.1.tgz", + "integrity": "sha512-7gr8p9TQP6RAHusBOSLs46F4564ZrjV8xFmw5zCmgmhGUcw2hxsShhJ6CEiHQMgPDwAQ1fWHPM0ypc4RMAig4A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@pkgr/core": "^0.1.0", + "tslib": "^2.6.2" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/unts" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true, + "license": "MIT" + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/tslib": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.7.0.tgz", + "integrity": "sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==", + "dev": true, + "license": "0BSD" + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "dev": true, + "license": "MIT" + }, + "node_modules/vite": { + "version": "5.4.6", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.6.tgz", + "integrity": "sha512-IeL5f8OO5nylsgzd9tq4qD2QqI0k2CQLGrWD0rCN0EQJZpBK5vJAx0I+GDkMOXxQX/OfFHMuLIx6ddAxGX/k+Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.21.3", + "postcss": "^8.4.43", + "rollup": "^4.20.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || >=20.0.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + } + } + }, + "node_modules/vue": { + "version": "3.4.27", + "resolved": "https://registry.npmjs.org/vue/-/vue-3.4.27.tgz", + "integrity": "sha512-8s/56uK6r01r1icG/aEOHqyMVxd1bkYcSe9j8HcKtr/xTOFWvnzIVTehNW+5Yt89f+DLBe4A569pnZLS5HzAMA==", + "license": "MIT", + "dependencies": { + "@vue/compiler-dom": "3.4.27", + "@vue/compiler-sfc": "3.4.27", + "@vue/runtime-dom": "3.4.27", + "@vue/server-renderer": "3.4.27", + "@vue/shared": "3.4.27" + }, + "peerDependencies": { + "typescript": "*" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/vue-draggable-next": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/vue-draggable-next/-/vue-draggable-next-2.2.1.tgz", + "integrity": "sha512-EAMS1IRHF0kZO0o5PMOinsQsXIqsrKT1hKmbICxG3UEtn7zLFkLxlAtajcCcUTisNvQ6TtCB5COjD9a1raNADw==", + "license": "MIT", + "peerDependencies": { + "sortablejs": "^1.14.0", + "vue": "^3.2.2" + } + }, + "node_modules/vue-eslint-parser": { + "version": "9.4.3", + "resolved": "https://registry.npmjs.org/vue-eslint-parser/-/vue-eslint-parser-9.4.3.tgz", + "integrity": "sha512-2rYRLWlIpaiN8xbPiDyXZXRgLGOtWxERV7ND5fFAv5qo1D2N9Fu9MNajBNc6o13lZ+24DAWCkQCvj4klgmcITg==", + "dev": true, + "license": "MIT", + "dependencies": { + "debug": "^4.3.4", + "eslint-scope": "^7.1.1", + "eslint-visitor-keys": "^3.3.0", + "espree": "^9.3.1", + "esquery": "^1.4.0", + "lodash": "^4.17.21", + "semver": "^7.3.6" + }, + "engines": { + "node": "^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/mysticatea" + }, + "peerDependencies": { + "eslint": ">=6.0.0" + } + }, + "node_modules/vue-i18n": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/vue-i18n/-/vue-i18n-10.0.4.tgz", + "integrity": "sha512-1xkzVxqBLk2ZFOmeI+B5r1J7aD/WtNJ4j9k2mcFcQo5BnOmHBmD7z4/oZohh96AAaRZ4Q7mNQvxc9h+aT+Md3w==", + "license": "MIT", + "dependencies": { + "@intlify/core-base": "10.0.4", + "@intlify/shared": "10.0.4", + "@vue/devtools-api": "^6.5.0" + }, + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://github.com/sponsors/kazupon" + }, + "peerDependencies": { + "vue": "^3.0.0" + } + }, + "node_modules/vue-material-design-icons": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/vue-material-design-icons/-/vue-material-design-icons-5.3.0.tgz", + "integrity": "sha512-wnbRh+48RwX/Gt+iqwCSdWpm0hPBwwv9F7MSouUzZ2PsphYVMJB9KkG9iGs+tgBiT57ZiurFEK07Y/rFKx+Ekg==", + "license": "MIT" + }, + "node_modules/vue-router": { + "version": "4.4.5", + "resolved": "https://registry.npmjs.org/vue-router/-/vue-router-4.4.5.tgz", + "integrity": "sha512-4fKZygS8cH1yCyuabAXGUAsyi1b2/o/OKgu/RUb+znIYOxPRxdkytJEx+0wGcpBE1pX6vUgh5jwWOKRGvuA/7Q==", + "license": "MIT", + "dependencies": { + "@vue/devtools-api": "^6.6.4" + }, + "funding": { + "url": "https://github.com/sponsors/posva" + }, + "peerDependencies": { + "vue": "^3.2.0" + } + }, + "node_modules/vue-spinner": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/vue-spinner/-/vue-spinner-1.0.4.tgz", + "integrity": "sha512-GNG2F+8BLX201JT/jUX+84Gsi3ZteVQwt9K7mues3ts9FcQ95dGn7uu6a5ndSxdYYUEzfh1KngZiOE0u+l4itA==", + "license": "MIT" + }, + "node_modules/vue-toastification": { + "version": "2.0.0-rc.5", + "resolved": "https://registry.npmjs.org/vue-toastification/-/vue-toastification-2.0.0-rc.5.tgz", + "integrity": "sha512-q73e5jy6gucEO/U+P48hqX+/qyXDozAGmaGgLFm5tXX4wJBcVsnGp4e/iJqlm9xzHETYOilUuwOUje2Qg1JdwA==", + "license": "MIT", + "peerDependencies": { + "vue": "^3.0.2" + } + }, + "node_modules/vuedraggable": { + "version": "2.24.3", + "resolved": "https://registry.npmjs.org/vuedraggable/-/vuedraggable-2.24.3.tgz", + "integrity": "sha512-6/HDXi92GzB+Hcs9fC6PAAozK1RLt1ewPTLjK0anTYguXLAeySDmcnqE8IC0xa7shvSzRjQXq3/+dsZ7ETGF3g==", + "license": "MIT", + "dependencies": { + "sortablejs": "1.10.2" + } + }, + "node_modules/vuedraggable/node_modules/sortablejs": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/sortablejs/-/sortablejs-1.10.2.tgz", + "integrity": "sha512-YkPGufevysvfwn5rfdlGyrGjt7/CRHwvRPogD/lC+TnvcN29jDpCifKP+rBqf+LRldfXSTh+0CGLcSg0VIxq3A==", + "license": "MIT" + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/which-module": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.1.tgz", + "integrity": "sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrap-ansi": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", + "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/wrap-ansi/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/wrap-ansi/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/xml-name-validator": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-4.0.0.tgz", + "integrity": "sha512-ICP2e+jsHvAj2E2lIHxa5tjXRlKDJo4IdvPvCXbXQGdzSfmSpNVyIKMvoZHjDY9DP0zV17iI85o90vRFXNccRw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12" + } + }, + "node_modules/y18n": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", + "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.2" + } + }, + "node_modules/yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "license": "ISC", + "dependencies": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + }, + "node_modules/yargs/node_modules/find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/yargs/node_modules/p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/zod": { + "version": "3.23.8", + "resolved": "https://registry.npmjs.org/zod/-/zod-3.23.8.tgz", + "integrity": "sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + } + } +} diff --git a/frontend/package.json b/frontend/package.json new file mode 100644 index 00000000..33880b93 --- /dev/null +++ b/frontend/package.json @@ -0,0 +1,43 @@ +{ + "name": "montage-frontend", + "version": "0.0.0", + "private": true, + "type": "module", + "scripts": { + "dev": "vite", + "build": "vite build", + "preview": "vite preview", + "watch:build": "chokidar 'src/**/*' -c 'npm run build && cp -r dist/* ../montage/static' --initial", + "toolforge:build": "export SHELL=/bin/sh && npm run build && cp -r dist/* ../montage/static", + "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs --fix --ignore-path .gitignore", + "format": "prettier --write src/" + }, + "dependencies": { + "@wikimedia/codex": "^1.14.0", + "@wikimedia/codex-icons": "^1.14.0", + "axios": "^1.7.7", + "dayjs": "^1.11.13", + "iso-639-1": "^3.1.3", + "lodash": "^4.17.21", + "pinia": "^2.1.7", + "vue": "^3.4.27", + "vue-draggable-next": "^2.2.1", + "vue-i18n": "^10.0.4", + "vue-material-design-icons": "^5.3.0", + "vue-router": "^4.3.3", + "vue-spinner": "^1.0.4", + "vue-toastification": "^2.0.0-rc.5", + "vuedraggable": "^2.24.3", + "zod": "^3.23.8" + }, + "devDependencies": { + "@rushstack/eslint-patch": "^1.8.0", + "@vitejs/plugin-vue": "^5.0.5", + "@vue/eslint-config-prettier": "^9.0.0", + "chokidar-cli": "^3.0.0", + "eslint": "^8.57.0", + "eslint-plugin-vue": "^9.23.0", + "prettier": "^3.2.5", + "vite": "^5.3.1" + } +} diff --git a/frontend/public/favicon.ico b/frontend/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..d0979bb63eb13580f28f7f51dc842a066466271f GIT binary patch literal 176289 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 z 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 literal 0 HcmV?d00001 diff --git a/frontend/src/App.vue b/frontend/src/App.vue new file mode 100644 index 00000000..fc35ed60 --- /dev/null +++ b/frontend/src/App.vue @@ -0,0 +1,47 @@ + + + + + diff --git a/frontend/src/assets/logo_white.svg b/frontend/src/assets/logo_white.svg new file mode 100644 index 00000000..2f486607 --- /dev/null +++ b/frontend/src/assets/logo_white.svg @@ -0,0 +1,71 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + diff --git a/frontend/src/assets/main.css b/frontend/src/assets/main.css new file mode 100644 index 00000000..5cb3464c --- /dev/null +++ b/frontend/src/assets/main.css @@ -0,0 +1,63 @@ +* { + margin: 0; + padding: 0; + box-sizing: border-box; + font-family: 'Lato', sans-serif; +} + +.greyed { + color: #8c8c8c; +} + +.icon-small { + font-size: 6px !important; +} + +.key { + display: inline-block; + margin: 0 0.1em; + width: 18px; + line-height: 18px; + height: 18px; + + text-align: center; + color: darkgray; + background: white; + font-size: 11px; + border-radius: 3px; + text-shadow: 0 1px 0 white; + white-space: nowrap; + border: 1px solid grey; + + -moz-box-shadow: + 0 1px 0px rgba(0, 0, 0, 0.2), + 0 0 0 2px #fff inset; + -webkit-box-shadow: + 0 1px 0px rgba(0, 0, 0, 0.2), + 0 0 0 2px #fff inset; + box-shadow: + 0 1px 0px rgba(0, 0, 0, 0.2), + 0 0 0 2px #fff inset; +} + +.juror-campaign-accordion summary:focus { + border-color: white !important; + box-shadow: none !important; +} + +.information-card .cdx-card__text { + width: 100%; +} + +.info-accordion summary { + padding-left: 0 !important; +} + +.date-time-inputs .cdx-label { + font-size: 14px !important; + color: gray !important; +} + +.cdx-select-vue__handle { + min-width: 120px; +} \ No newline at end of file diff --git a/frontend/src/components/AddOrganizer.vue b/frontend/src/components/AddOrganizer.vue new file mode 100644 index 00000000..ebf4f7da --- /dev/null +++ b/frontend/src/components/AddOrganizer.vue @@ -0,0 +1,78 @@ + + + + diff --git a/frontend/src/components/AppFooter.vue b/frontend/src/components/AppFooter.vue new file mode 100644 index 00000000..2963371f --- /dev/null +++ b/frontend/src/components/AppFooter.vue @@ -0,0 +1,52 @@ + + + diff --git a/frontend/src/components/AppHeader.vue b/frontend/src/components/AppHeader.vue new file mode 100644 index 00000000..bf94177a --- /dev/null +++ b/frontend/src/components/AppHeader.vue @@ -0,0 +1,101 @@ + + + + + diff --git a/frontend/src/components/Campaign/ActiveCampaign.vue b/frontend/src/components/Campaign/ActiveCampaign.vue new file mode 100644 index 00000000..ee3e799d --- /dev/null +++ b/frontend/src/components/Campaign/ActiveCampaign.vue @@ -0,0 +1,187 @@ + + + + + diff --git a/frontend/src/components/Campaign/AllCampaign.vue b/frontend/src/components/Campaign/AllCampaign.vue new file mode 100644 index 00000000..40b817f4 --- /dev/null +++ b/frontend/src/components/Campaign/AllCampaign.vue @@ -0,0 +1,153 @@ + + + + + diff --git a/frontend/src/components/Campaign/CoordinatorCampaignCard.vue b/frontend/src/components/Campaign/CoordinatorCampaignCard.vue new file mode 100644 index 00000000..9b689c3f --- /dev/null +++ b/frontend/src/components/Campaign/CoordinatorCampaignCard.vue @@ -0,0 +1,106 @@ + + + + + diff --git a/frontend/src/components/Campaign/JurorCampaignCard.vue b/frontend/src/components/Campaign/JurorCampaignCard.vue new file mode 100644 index 00000000..a15806dc --- /dev/null +++ b/frontend/src/components/Campaign/JurorCampaignCard.vue @@ -0,0 +1,142 @@ + + + + + diff --git a/frontend/src/components/Campaign/NewCampaign.vue b/frontend/src/components/Campaign/NewCampaign.vue new file mode 100644 index 00000000..739df180 --- /dev/null +++ b/frontend/src/components/Campaign/NewCampaign.vue @@ -0,0 +1,221 @@ + + + + + diff --git a/frontend/src/components/Campaign/ViewCampaign.vue b/frontend/src/components/Campaign/ViewCampaign.vue new file mode 100644 index 00000000..008657b4 --- /dev/null +++ b/frontend/src/components/Campaign/ViewCampaign.vue @@ -0,0 +1,321 @@ + + + + + diff --git a/frontend/src/components/LoginBox.vue b/frontend/src/components/LoginBox.vue new file mode 100644 index 00000000..5b284ff3 --- /dev/null +++ b/frontend/src/components/LoginBox.vue @@ -0,0 +1,74 @@ + + + + + diff --git a/frontend/src/components/Round/RoundEdit.vue b/frontend/src/components/Round/RoundEdit.vue new file mode 100644 index 00000000..67144552 --- /dev/null +++ b/frontend/src/components/Round/RoundEdit.vue @@ -0,0 +1,182 @@ + + + diff --git a/frontend/src/components/Round/RoundInfo.vue b/frontend/src/components/Round/RoundInfo.vue new file mode 100644 index 00000000..0d432730 --- /dev/null +++ b/frontend/src/components/Round/RoundInfo.vue @@ -0,0 +1,269 @@ + + + + + diff --git a/frontend/src/components/Round/RoundNew.vue b/frontend/src/components/Round/RoundNew.vue new file mode 100644 index 00000000..002c2d35 --- /dev/null +++ b/frontend/src/components/Round/RoundNew.vue @@ -0,0 +1,482 @@ + + + + + diff --git a/frontend/src/components/Round/RoundView.vue b/frontend/src/components/Round/RoundView.vue new file mode 100644 index 00000000..df1ede6f --- /dev/null +++ b/frontend/src/components/Round/RoundView.vue @@ -0,0 +1,103 @@ + + + + + diff --git a/frontend/src/components/UserAvatarWithName.vue b/frontend/src/components/UserAvatarWithName.vue new file mode 100644 index 00000000..6ed2a4aa --- /dev/null +++ b/frontend/src/components/UserAvatarWithName.vue @@ -0,0 +1,57 @@ + + + + + diff --git a/frontend/src/components/UserList.vue b/frontend/src/components/UserList.vue new file mode 100644 index 00000000..fdc60d32 --- /dev/null +++ b/frontend/src/components/UserList.vue @@ -0,0 +1,52 @@ + + + diff --git a/frontend/src/components/Vote/Vote.vue b/frontend/src/components/Vote/Vote.vue new file mode 100644 index 00000000..f72133c0 --- /dev/null +++ b/frontend/src/components/Vote/Vote.vue @@ -0,0 +1,30 @@ + + + \ No newline at end of file diff --git a/frontend/src/components/Vote/VoteEdit.vue b/frontend/src/components/Vote/VoteEdit.vue new file mode 100644 index 00000000..630a5cca --- /dev/null +++ b/frontend/src/components/Vote/VoteEdit.vue @@ -0,0 +1,534 @@ + + + + + diff --git a/frontend/src/components/Vote/VoteRanking.vue b/frontend/src/components/Vote/VoteRanking.vue new file mode 100644 index 00000000..ed1b981f --- /dev/null +++ b/frontend/src/components/Vote/VoteRanking.vue @@ -0,0 +1,304 @@ + + + + + diff --git a/frontend/src/components/Vote/VoteRating.vue b/frontend/src/components/Vote/VoteRating.vue new file mode 100644 index 00000000..b81a8596 --- /dev/null +++ b/frontend/src/components/Vote/VoteRating.vue @@ -0,0 +1,591 @@ + + + + + diff --git a/frontend/src/components/Vote/VoteYesNo.vue b/frontend/src/components/Vote/VoteYesNo.vue new file mode 100644 index 00000000..bc819f26 --- /dev/null +++ b/frontend/src/components/Vote/VoteYesNo.vue @@ -0,0 +1,592 @@ + + + + + diff --git a/frontend/src/i18n.js b/frontend/src/i18n.js new file mode 100644 index 00000000..75fea532 --- /dev/null +++ b/frontend/src/i18n.js @@ -0,0 +1,40 @@ +import { createI18n } from 'vue-i18n'; +import en from '@/i18n/en.json'; + +// Initialize with default English messages in case of error +const messages = { en }; + +// Dynamically load all other messages +async function loadMessages() { + try { + const modules = import.meta.glob('./i18n/*.json'); + await Promise.all( + Object.entries(modules).map(async ([path, importFn]) => { + const lang = path.replace('./i18n/', '').replace('.json', ''); + if (lang !== 'en' && lang !== 'qqq') { + const module = await importFn(); + messages[lang] = module.default; + } + }) + ); + } catch (error) { + console.error('Error loading i18n messages:', error); + } +} + +// Initialize i18n instance +async function initializeI18n() { + try { + await loadMessages(); + } catch (error) { + console.error('Error initializing i18n:', error); + } + + return createI18n({ + locale: 'en', + fallbackLocale: 'en', + messages + }); +} + +export default initializeI18n; \ No newline at end of file diff --git a/frontend/src/i18n/en.json b/frontend/src/i18n/en.json new file mode 100644 index 00000000..f20db0d3 --- /dev/null +++ b/frontend/src/i18n/en.json @@ -0,0 +1,186 @@ +{ + "@metadata": { + "authors": [ + "Jayprakash12345" + ] + }, + "montage-title": "Montage", + "montage-about": "About", + "montage-credit": "by Montage team", + "montage-source-code": "Source code", + "montage-login-heading": "Please log in", + "montage-login-description": "You will be redirected to Meta-Wiki in order to confirm your identity. Montage will not publish anything on Wikimedia projects using your account.", + "montage-login-account-instructions": "If you don't have a Wikimedia account, you can create one on {0}.", + "montage-login-metawiki": "Meta-Wiki", + "montage-login-button": "Log in using Wikimedia account", + "montage-login-logout": "Log out", + "montage-active-campaigns": "Active campaigns", + "montage-all-campaigns": "All campaigns", + "montage-new-campaign": "Create new campaign", + "montage-manage-current": "Manage current campaigns below", + "montage-manage-all": "View all campaigns, active and archived below", + "montage-or": "or", + "montage-view-all": "view all campaigns and rounds.", + "montage-view-active": "view only active campaigns and rounds.", + "montage-coordinator-campaigns": "Coordinator campaigns", + "montage-juror-campaigns": "Juror campaigns", + "montage-active-voting-round": "Active voting rounds", + "montage-latest-round": "Latest round", + "montage-coordinators": "Coordinators", + "montage-voting-deadline": "Voting deadline", + "montage-directions": "Directions", + "montage-your-progress": "Your progress", + "montage-vote": "Vote", + "montage-edit-previous-vote": "Edit previous votes", + "montage-progress-status": "{0} out of {1}", + "montage-new-campaig-heading": "New campaign", + "montage-placeholder-campaign-name": "Campaign name", + "montage-description-campaign-name": "Enter the campaign name", + "montage-required-campaign-name": "Campaign name is required", + "montage-placeholder-campaign-url": "Campaign URL", + "montage-description-campaign-url": "Enter URL of campaign landing page, e.g., on Commons or local Wiki Loves.", + "montage-required-campaign-url": "Campaign URL is required", + "montage-invalid-campaign-url": "Invalid URL", + "montage-label-date-range": "Date range (UTC)", + "montage-description-date-range": "Once the images are imported to the first round, changing date and time won't affect the images your jury can view. However, changing these values before the import to round 1 will ensure that only photos uploaded before the end date and time are visible to your jury.", + "montage-label-open-date": "Open date", + "montage-required-open-date": "Open date is required", + "montage-label-open-time": "Open time", + "montage-required-open-time": "Open time is required", + "montage-label-close-date": "Close date", + "montage-required-close-date": "Close date is required", + "montage-label-close-time": "Close time", + "montage-required-close-time": "Close time is required", + "montage-label-campaign-coordinators": "Campaign Coordinators", + "montage-description-campaign-coordinators": "Coordinators are people who have access to edit the campaign, rounds, and round statistics.", + "montage-required-campaign-coordinators": "At least one coordinator is required", + "montage-campaign-added-success": "Campaign added successfully", + "montage-something-went-wrong": "Something went wrong", + "montage-btn-create-campaign": "Create campaign", + "montage-btn-save": "Save", + "montage-btn-cancel": "Cancel", + "montage-close-campaign": "Close campaign", + "montage-archive": "Archive", + "montage-unarchive": "Unarchive", + "montage-edit-campaign": "Edit campaign", + "montage-round-add": "Add Round", + "montage-round-open-date": "Open date (UTC)", + "montage-round-open-time": "Open time (UTC)", + "montage-round-close-date": "Close date (UTC)", + "montage-round-close-time": "Close time (UTC)", + "montage-round-name": "Round name", + "montage-round-deadline": "Voting deadline", + "montage-label-round-stats": "Show own statistics (Beta)", + "montage-description-round-stats": "Whether to show own voting statistics (e.g. number of accepted or declined images) of juror for the round.", + "montage-label-round-quorum": "Quorum", + "montage-description-round-quorum": "The number of jurors that must vote on each image", + "montage-label-round-jurors": "Jurors", + "montage-description-round-jurors": "Enter the username of the juror you want to add to this round.", + "montage-round-stats": "Show own statistics (Beta)", + "montage-round-stats-description": "Whether to show own voting statistics (e.g., number of accepted or declined images) of juror for the round.", + "montage-round-file-setting": "Round file settings", + "montage-round-delete": "Delete round", + "montage-round-edit": "Edit round", + "montage-round-save": "Save Round", + "montage-round-delete-confirm": "Are you sure you want to delete this round?", + "montage-round-allowed-filetypes": "Allowed filetypes", + "montage-round-dq-by-filetype": "Disqualify by filetype", + "montage-round-dq-by-resolution": "Disqualify by resolution", + "montage-round-dq-by-upload-date": "Disqualify by upload date", + "montage-round-dq-by-uploader": "Disqualify jurors", + "montage-round-dq-coords": "Disqualify coordinators", + "montage-round-dq-maintainers": "Disqualify maintainers", + "montage-round-dq-organizers": "Disqualify organizers", + "montage-round-final-threshold": "Final threshold", + "montage-round-min-resolution": "Min. resolution", + "montage-round-show-filename": "Show filename", + "montage-round-show-link": "Show link", + "montage-round-show-resolution": "Show resolution", + "montage-round-show-stats": "Show stats", + "montage-round-vote-ending": "in {0} days", + "montage-round-no-direction-given": "No directions provided", + "montage-round-ranking": "Ranking", + "montage-round-rating": "Rating", + "montage-round-yesno": "Yes/No", + "montage-round-file-info": "Round file information", + "montage-round-file-type": "File type", + "montage-round-voting-details": "Voting details", + "montage-round-activate": "Activate", + "montage-round-activated": "Round activated successfully.", + "montage-round-pause": "Pause", + "montage-round-paused": "Round paused successfully.", + "montage-round-added": "Round added successfully.", + "montage-round-finalize": "Finalize", + "montage-round-finalized": "Round finalized successfully.", + "montage-round-download-results": "Download results", + "montage-round-download-entries": "Download entries", + "montage-round-source": "Source", + "montage-round-source-category": "Category on Wikimedia Commons", + "montage-round-source-csv": "File List URL", + "montage-round-source-filelist": "File List", + "montage-round-source-category-help": "Category on Wikimedia Commons that gathers all contest images. Example: Images from Wiki Loves Monuments 2017 in Ghana.", + "montage-round-source-csv-help": "List of files saved as CSV and uploaded as a Google Sheet or Gist.", + "montage-round-source-selected-help": "List of files, one each line without File: prefix.", + "montage-round-category-placeholder": "Enter category", + "montage-round-category-label": "Enter category", + "montage-round-no-category": "No categories found.", + "montage-round-file-url": "Enter File URL", + "montage-round-file-list": "List (One file per line)", + "montage-round-threshold": "Threshold", + "montage-round-vote-method": "Vote method", + "montage-round-threshold-description": "Minimal average rating for photo", + "montage-round-threshold-default": "Choose the threshold", + "montage-round-jurors-description": "Enter the username of the juror you want to add to this round.", + "montage-round-quorum-description": "The number of jurors that must vote on each image", + "montage-no-results": "No results found", + "montage-vote-round-inactive": "Round is not active", + "montage-vote-contact-organizer": "This round is not active. Please contact the organizer.", + "montage-vote-show-full-size": "Show full-size", + "montage-vote-commons-page": "Commons page", + "montage-vote-accept": "Accept", + "montage-vote-decline": "Decline", + "montage-vote-keyboard-instructions": "You can also use the keyboard to vote.", + "montage-vote-actions": "Actions", + "montage-vote-add-favorites": "Add to favorites", + "montage-vote-added-favorites": "Image added to favorites", + "montage-vote-remove-favorites": "Remove from favorites", + "montage-vote-removed-favorites": "Image removed from favorites", + "montage-vote-skip": "Skip (vote later)", + "montage-vote-description": "Description", + "montage-vote-version": "Version", + "montage-vote-last-version": "Last version at {0}", + "montage-vote-all-done": "All done!", + "montage-vote-no-images": "You voted on all images in this round. You can still edit your previous votes using the button below.", + "montage-vote-hide-panel": "Hide panel", + "montage-vote-show-panel": "Show panel", + "montage-vote-image": "Image", + "montage-vote-image-remains": "{0} image remaining", + "montage-vote-rating-instructions": "One to five stars", + "montage-vote-round-part-of-campaign": "Part of {0}", + "montage-vote-grid-size-large": "Large", + "montage-vote-grid-size-medium": "Medium", + "montage-vote-grid-size-small": "Small", + "montage-vote-image-review": "Image review", + "montage-vote-ordinal-place": "{0} place", + "montage-vote-order-by": "Order by:", + "montage-vote-gallery-size": "Gallery size", + "montage-voted-time": "voted {0}", + "montage-no-votes-yet": "No votes yet!", + "montage-no-votes-this-round": "You haven't voted on any image this round.", + "montage-option-yes": "Yes", + "montage-option-no": "No", + "montage-vote-edit-for": "Edit votes for {0}", + "montage-add-organizer": "Add campaign organizer", + "montage-btn-add": "Add", + "montage-added-organizer": "Organizer added successfully", + "montage-at-least-one-user": "Please add at least one user", + "montage-only-one-user": "Please add only one user", + "montage-round-quorum-per-photo": "{0} jurors per photo", + "montage-round-open-task-percentage": "Percentage of opened tasks", + "montage-round-cancelled-tasks": "Cancelled tasks", + "montage-round-disqualified-files": "Disqualified files", + "montage-round-open-tasks": "Open tasks", + "montage-round-files": "Files", + "montage-round-tasks": "Tasks", + "montage-round-uploaders": "Uploaders" +} \ No newline at end of file diff --git a/frontend/src/i18n/hi.json b/frontend/src/i18n/hi.json new file mode 100644 index 00000000..82fe40a5 --- /dev/null +++ b/frontend/src/i18n/hi.json @@ -0,0 +1,186 @@ +{ + "@metadata": { + "authors": [ + "Jayprakash12345" + ] + }, + "montage-title": "Montage", + "montage-about": "बारे में", + "montage-credit": "Montage टीम द्वारा", + "montage-source-code": "स्रोत कोड", + "montage-login-heading": "कृपया लॉग इन करें", + "montage-login-description": "आपकी पहचान की पुष्टि के लिए आपको मेटा-विकि पर पुनर्निर्देशित किया जाएगा। मोंटाज आपके खाते का उपयोग करके विकिमीडिया परियोजनाओं पर कुछ भी प्रकाशित नहीं करेगा।", + "montage-login-account-instructions": "यदि आपके पास विकिमीडिया खाता नहीं है, तो आप {0} पर खाता बना सकते हैं।", + "montage-login-metawiki": "मेटा-विकि", + "montage-login-button": "विकिमीडिया खाता उपयोग करके लॉग इन करें", + "montage-login-logout": "लॉग आउट", + "montage-active-campaigns": "सक्रिय अभियान", + "montage-all-campaigns": "सभी अभियान", + "montage-new-campaign": "नया अभियान बनाएं", + "montage-manage-current": "नीचे वर्तमान अभियान प्रबंधित करें", + "montage-manage-all": "नीचे सभी सक्रिय और संग्रहित अभियान देखें", + "montage-or": "या", + "montage-view-all": "सभी अभियान और राउंड देखें।", + "montage-view-active": "केवल सक्रिय अभियान और राउंड देखें।", + "montage-coordinator-campaigns": "समन्वयक अभियान", + "montage-juror-campaigns": "जूरी अभियान", + "montage-active-voting-round": "सक्रिय मतदान राउंड", + "montage-latest-round": "नवीनतम राउंड", + "montage-coordinators": "समन्वयक", + "montage-voting-deadline": "मतदान की समय सीमा", + "montage-directions": "निर्देश", + "montage-your-progress": "आपकी प्रगति", + "montage-vote": "मत", + "montage-edit-previous-vote": "पिछले मत संपादित करें", + "montage-progress-status": "{0} में से {1}", + "montage-new-campaig-heading": "नया अभियान", + "montage-placeholder-campaign-name": "अभियान का नाम", + "montage-description-campaign-name": "अभियान का नाम दर्ज करें", + "montage-required-campaign-name": "अभियान का नाम आवश्यक है", + "montage-placeholder-campaign-url": "अभियान URL", + "montage-description-campaign-url": "अभियान लैंडिंग पृष्ठ का URL दर्ज करें, जैसे कॉमन्स या स्थानीय विकि लव्स।", + "montage-required-campaign-url": "अभियान URL आवश्यक है", + "montage-invalid-campaign-url": "अमान्य URL", + "montage-label-date-range": "तिथि सीमा (UTC)", + "montage-description-date-range": "एक बार जब छवियां पहले राउंड में आयात हो जाती हैं, तो तिथि और समय बदलने से आपकी जूरी द्वारा देखी जाने वाली छवियों पर प्रभाव नहीं पड़ेगा। हालाँकि, राउंड 1 में आयात से पहले इन मानों को बदलने से सुनिश्चित होगा कि केवल अंतिम तिथि और समय से पहले अपलोड की गई तस्वीरें आपकी जूरी को दिखाई दें।", + "montage-label-open-date": "खुलने की तिथि", + "montage-required-open-date": "खुलने की तिथि आवश्यक है", + "montage-label-open-time": "खुलने का समय", + "montage-required-open-time": "खुलने का समय आवश्यक है", + "montage-label-close-date": "बंद होने की तिथि", + "montage-required-close-date": "बंद होने की तिथि आवश्यक है", + "montage-label-close-time": "बंद होने का समय", + "montage-required-close-time": "बंद होने का समय आवश्यक है", + "montage-label-campaign-coordinators": "अभियान समन्वयक", + "montage-description-campaign-coordinators": "समन्वयक वे लोग हैं जिनके पास अभियान, राउंड और राउंड सांख्यिकी को संपादित करने की अनुमति है।", + "montage-required-campaign-coordinators": "कम से कम एक समन्वयक आवश्यक है", + "montage-campaign-added-success": "अभियान सफलतापूर्वक जोड़ा गया", + "montage-something-went-wrong": "कुछ गलत हो गया", + "montage-btn-create-campaign": "अभियान बनाएं", + "montage-btn-save": "सहेजें", + "montage-btn-cancel": "रद्द करें", + "montage-close-campaign": "अभियान बंद करें", + "montage-archive": "संग्रह", + "montage-unarchive": "संग्रह से हटाएं", + "montage-edit-campaign": "अभियान संपादित करें", + "montage-round-add": "राउंड जोड़ें", + "montage-round-open-date": "खुलने की तिथि (UTC)", + "montage-round-open-time": "खुलने का समय (UTC)", + "montage-round-close-date": "बंद होने की तिथि (UTC)", + "montage-round-close-time": "बंद होने का समय (UTC)", + "montage-round-name": "राउंड का नाम", + "montage-round-deadline": "मतदान की समय सीमा", + "montage-label-round-stats": "स्वयं की सांख्यिकी दिखाएं (बीटा)", + "montage-description-round-stats": "चाहे जूरी की स्वयं की सांख्यिकी (जैसे स्वीकृत या अस्वीकृत छवियों की संख्या) दिखानी हो।", + "montage-label-round-quorum": "क्वोरम", + "montage-description-round-quorum": "प्रत्येक छवि पर मतदान करने वाले जूरी सदस्यों की संख्या", + "montage-label-round-jurors": "जूरी सदस्य", + "montage-description-round-jurors": "जूरी सदस्य का उपयोगकर्ता नाम दर्ज करें जिसे आप इस राउंड में जोड़ना चाहते हैं।", + "montage-round-stats": "स्वयं की सांख्यिकी दिखाएं (बीटा)", + "montage-round-stats-description": "चाहे जूरी की स्वयं की सांख्यिकी (जैसे स्वीकृत या अस्वीकृत छवियों की संख्या) दिखानी हो।", + "montage-round-file-setting": "राउंड फाइल सेटिंग्स", + "montage-round-delete": "राउंड हटाएं", + "montage-round-edit": "राउंड संपादित करें", + "montage-round-save": "राउंड सहेजें", + "montage-round-delete-confirm": "क्या आप वाकई इस राउंड को हटाना चाहते हैं?", + "montage-round-allowed-filetypes": "अनुमत फाइल प्रकार", + "montage-round-dq-by-filetype": "फाइल प्रकार के अनुसार अयोग्य घोषित करें", + "montage-round-dq-by-resolution": "रेज़ोल्यूशन के अनुसार अयोग्य घोषित करें", + "montage-round-dq-by-upload-date": "अपलोड तिथि के अनुसार अयोग्य घोषित करें", + "montage-round-dq-by-uploader": "जूरी सदस्यों को अयोग्य घोषित करें", + "montage-round-dq-coords": "समन्वयकों को अयोग्य घोषित करें", + "montage-round-dq-maintainers": "रखरखाव करने वालों को अयोग्य घोषित करें", + "montage-round-dq-organizers": "आयोजकों को अयोग्य घोषित करें", + "montage-round-final-threshold": "अंतिम सीमा", + "montage-round-min-resolution": "न्यूनतम रेज़ोल्यूशन", + "montage-round-show-filename": "फाइल नाम दिखाएं", + "montage-round-show-link": "लिंक दिखाएं", + "montage-round-show-resolution": "रेज़ोल्यूशन दिखाएं", + "montage-round-show-stats": "सांख्यिकी दिखाएं", + "montage-round-vote-ending": "{0} दिनों में समाप्त हो रहा है", + "montage-round-no-direction-given": "कोई निर्देश नहीं दिया गया", + "montage-round-ranking": "रैंकिंग", + "montage-round-rating": "रेटिंग", + "montage-round-yesno": "हां/नहीं", + "montage-round-file-info": "राउंड फाइल जानकारी", + "montage-round-file-type": "फाइल प्रकार", + "montage-round-voting-details": "मतदान का विवरण", + "montage-round-activate": "सक्रिय करें", + "montage-round-activated": "राउंड सफलतापूर्वक सक्रिय किया गया।", + "montage-round-pause": "रोकें", + "montage-round-paused": "राउंड सफलतापूर्वक रोका गया।", + "montage-round-added": "राउंड सफलतापूर्वक जोड़ा गया।", + "montage-round-finalize": "अंतिम रूप दें", + "montage-round-finalized": "राउंड सफलतापूर्वक अंतिम रूप दिया गया।", + "montage-round-download-results": "परिणाम डाउनलोड करें", + "montage-round-download-entries": "प्रविष्टियां डाउनलोड करें", + "montage-round-source": "स्रोत", + "montage-round-source-category": "विकिमीडिया कॉमन्स पर श्रेणी", + "montage-round-source-csv": "फाइल सूची URL", + "montage-round-source-filelist": "फाइल सूची", + "montage-round-source-category-help": "विकिमीडिया कॉमन्स पर श्रेणी जो सभी प्रतियोगिता छवियों को एकत्र करती है। उदाहरण: विकि लव्स मॉन्यूमेंट्स 2017 इन घाना।", + "montage-round-source-csv-help": "CSV के रूप में सहेजी गई फाइलों की सूची और Google शीट या Gist पर अपलोड की गई।", + "montage-round-source-selected-help": "फाइलों की सूची, प्रत्येक पंक्ति में एक बिना 'File:' उपसर्ग के।", + "montage-round-category-placeholder": "श्रेणी दर्ज करें", + "montage-round-category-label": "श्रेणी दर्ज करें", + "montage-round-no-category": "कोई श्रेणियां नहीं मिलीं।", + "montage-round-file-url": "फाइल URL दर्ज करें", + "montage-round-file-list": "सूची (प्रत्येक पंक्ति में एक फाइल)", + "montage-round-threshold": "सीमा", + "montage-round-vote-method": "वोट विधि", + "montage-round-threshold-description": "फोटो के लिए न्यूनतम औसत रेटिंग", + "montage-round-threshold-default": "सीमा चुनें", + "montage-round-jurors-description": "उस जूरी सदस्य का उपयोगकर्ता नाम दर्ज करें जिसे आप इस राउंड में जोड़ना चाहते हैं।", + "montage-round-quorum-description": "प्रत्येक छवि पर मतदान करने वाले जूरी सदस्यों की संख्या", + "montage-no-results": "कोई परिणाम नहीं मिला", + "montage-vote-round-inactive": "राउंड सक्रिय नहीं है", + "montage-vote-contact-organizer": "यह राउंड सक्रिय नहीं है। कृपया आयोजक से संपर्क करें।", + "montage-vote-show-full-size": "पूर्ण आकार दिखाएं", + "montage-vote-commons-page": "कॉमन्स पृष्ठ", + "montage-vote-accept": "स्वीकार करें", + "montage-vote-decline": "अस्वीकृत करें", + "montage-vote-keyboard-instructions": "आप कीबोर्ड का उपयोग करके भी मतदान कर सकते हैं।", + "montage-vote-actions": "क्रियाएं", + "montage-vote-add-favorites": "पसंदीदा में जोड़ें", + "montage-vote-added-favorites": "छवि पसंदीदा में जोड़ी गई", + "montage-vote-remove-favorites": "पसंदीदा से निकालें", + "montage-vote-removed-favorites": "छवि पसंदीदा से निकाली गई", + "montage-vote-skip": "छोड़ें (बाद में मतदान करें)", + "montage-vote-description": "विवरण", + "montage-vote-version": "संस्करण", + "montage-vote-last-version": "{0} पर अंतिम संस्करण", + "montage-vote-all-done": "सभी काम पूरे!", + "montage-vote-no-images": "आपने इस राउंड में सभी छवियों पर मतदान किया है। आप अभी भी नीचे दिए गए बटन का उपयोग करके अपने पिछले मतों को संपादित कर सकते हैं।", + "montage-vote-hide-panel": "पैनल छुपाएं", + "montage-vote-show-panel": "पैनल दिखाएं", + "montage-vote-image": "छवि", + "montage-vote-image-remains": "{0} छवि शेष", + "montage-vote-rating-instructions": "एक से पांच सितारे", + "montage-vote-round-part-of-campaign": "{0} का हिस्सा", + "montage-vote-grid-size-large": "बड़ा", + "montage-vote-grid-size-medium": "मध्यम", + "montage-vote-grid-size-small": "छोटा", + "montage-vote-image-review": "छवि समीक्षा", + "montage-vote-ordinal-place": "{0} स्थान", + "montage-vote-order-by": "क्रमबद्ध करें:", + "montage-vote-gallery-size": "गैलरी आकार", + "montage-voted-time": "{0} मतदान किया", + "montage-no-votes-yet": "अभी तक कोई मत नहीं!", + "montage-no-votes-this-round": "आपने इस राउंड में किसी भी छवि पर मतदान नहीं किया।", + "montage-option-yes": "हां", + "montage-option-no": "नहीं", + "montage-vote-edit-for": "{0} के लिए वोट संपादित", + "montage-add-organizer": "अभियान आयोजक जोड़ें", + "montage-btn-add": "जोड़ें", + "montage-added-organizer": "आयोजक सफलतापूर्वक जोड़ा गया", + "montage-at-least-one-user": "कृपया कम से कम एक उपयोगकर्ता जोड़ें", + "montage-only-one-user": "कृपया केवल एक उपयोगकर्ता जोड़ें", + "montage-round-quorum-per-photo": "{0} फोटो प्रति जूरी सदस्य", + "montage-round-open-task-percentage": "खुले कार्यों का प्रतिशत", + "montage-round-cancelled-tasks": "रद्द किए गए कार्य", + "montage-round-disqualified-files": "अयोग्य फाइलें", + "montage-round-open-tasks": "खुले कार्य", + "montage-round-files": "फाइलें", + "montage-round-tasks": "कार्य", + "montage-round-uploaders": "अपलोडर्स" +} \ No newline at end of file diff --git a/frontend/src/i18n/qqq.json b/frontend/src/i18n/qqq.json new file mode 100644 index 00000000..208dfd43 --- /dev/null +++ b/frontend/src/i18n/qqq.json @@ -0,0 +1,186 @@ +{ + "@metadata": { + "authors": [ + "Jayprakash12345" + ] + }, + "montage-title": "The title of the application.", + "montage-about": "Label for the 'About' section in the application.", + "montage-credit": "Acknowledgment text for the Montage team.", + "montage-source-code": "Label for the link to the application's source code.", + "montage-login-heading": "Heading text displayed on the login page.", + "montage-login-description": "Description text explaining the login process and data privacy.", + "montage-login-account-instructions": "Instructions for creating a Wikimedia account, with a placeholder for a URL.", + "montage-login-metawiki": "Name of the Meta-Wiki, used in login instructions.", + "montage-login-button": "Label for the button to log in using a Wikimedia account.", + "montage-login-logout": "Label for the button to log out.", + "montage-active-campaigns": "Label for the section listing active campaigns.", + "montage-all-campaigns": "Label for the section listing all campaigns.", + "montage-new-campaign": "Label for the button to create a new campaign.", + "montage-manage-current": "Instruction to manage current campaigns.", + "montage-manage-all": "Instruction to view and manage all campaigns, active and archived.", + "montage-or": "Text used to provide an alternative option.", + "montage-view-all": "Instruction to view all campaigns and rounds.", + "montage-view-active": "Instruction to view only active campaigns and rounds.", + "montage-coordinator-campaigns": "Label for campaigns managed by coordinators.", + "montage-juror-campaigns": "Label for campaigns involving jurors.", + "montage-active-voting-round": "Label for the section listing active voting rounds.", + "montage-latest-round": "Label for the latest round in the application.", + "montage-coordinators": "Label for the section listing coordinators of a campaign.", + "montage-voting-deadline": "Label for the voting deadline field.", + "montage-directions": "Label for the section or field where directions are provided.", + "montage-your-progress": "Label for displaying the user's voting progress.", + "montage-vote": "Label for the voting button.", + "montage-edit-previous-vote": "Label for editing previously submitted votes.", + "montage-progress-status": "Template to display progress status with placeholders for completed and total tasks.", + "montage-new-campaig-heading": "Heading for the new campaign creation page.", + "montage-placeholder-campaign-name": "Placeholder text for the campaign name input field.", + "montage-description-campaign-name": "Description for the campaign name input field.", + "montage-required-campaign-name": "Validation message for a required campaign name field.", + "montage-placeholder-campaign-url": "Placeholder text for the campaign URL input field.", + "montage-description-campaign-url": "Description for the campaign URL input field.", + "montage-required-campaign-url": "Validation message for a required campaign URL field.", + "montage-invalid-campaign-url": "Validation message for an invalid campaign URL.", + "montage-label-date-range": "Label for the date range input field.", + "montage-description-date-range": "Description for the date range input field, explaining its purpose.", + "montage-label-open-date": "Label for the open date input field.", + "montage-required-open-date": "Validation message for a required open date field.", + "montage-label-open-time": "Label for the open time input field.", + "montage-required-open-time": "Validation message for a required open time field.", + "montage-label-close-date": "Label for the close date input field.", + "montage-required-close-date": "Validation message for a required close date field.", + "montage-label-close-time": "Label for the close time input field.", + "montage-required-close-time": "Validation message for a required close time field.", + "montage-label-campaign-coordinators": "Label for the campaign coordinators input field.", + "montage-description-campaign-coordinators": "Description for the campaign coordinators input field, explaining their role.", + "montage-required-campaign-coordinators": "Validation message for a required campaign coordinator field.", + "montage-campaign-added-success": "Notification message indicating a campaign was successfully added.", + "montage-something-went-wrong": "Error message indicating an unspecified problem occurred.", + "montage-btn-create-campaign": "Label for the button to create a campaign.", + "montage-btn-save": "Label for the button to save changes.", + "montage-btn-cancel": "Label for the button to cancel an action.", + "montage-close-campaign": "Label for the action to close a campaign.", + "montage-archive": "Label for the action to archive a campaign or round.", + "montage-unarchive": "Label for the action to unarchive a campaign or round.", + "montage-edit-campaign": "Label for the action to edit a campaign.", + "montage-round-add": "Label for the action to add a new round.", + "montage-round-open-date": "Label for the open date input field for a round.", + "montage-round-open-time": "Label for the open time input field for a round.", + "montage-round-close-date": "Label for the close date input field for a round.", + "montage-round-close-time": "Label for the close time input field for a round.", + "montage-round-name": "Label for the round name input field.", + "montage-round-deadline": "Label for the voting deadline field in a round.", + "montage-label-round-stats": "Label for the option to show round statistics.", + "montage-description-round-stats": "Description for the option to show round statistics, explaining what is displayed.", + "montage-label-round-quorum": "Label for the quorum field for a round.", + "montage-description-round-quorum": "Description for the quorum field, explaining the required juror votes per image.", + "montage-label-round-jurors": "Label for the jurors input field for a round.", + "montage-description-round-jurors": "Description for the jurors input field, explaining how to add jurors to a round.", + "montage-round-stats": "Label for the statistics display in a round.", + "montage-round-stats-description": "Description for the round statistics, showing accepted or declined image counts.", + "montage-round-file-setting": "Label for the file settings in a round.", + "montage-round-delete": "Label for the action to delete a round.", + "montage-round-edit": "Label for the action to edit a round.", + "montage-round-save": "Label for the action to save changes to a round.", + "montage-round-delete-confirm": "Confirmation message for deleting a round.", + "montage-round-allowed-filetypes": "Label for the allowed file types setting in a round.", + "montage-round-dq-by-filetype": "Label for the disqualification rule based on file types.", + "montage-round-dq-by-resolution": "Label for the disqualification rule based on image resolution.", + "montage-round-dq-by-upload-date": "Label for the disqualification rule based on upload date.", + "montage-round-dq-by-uploader": "Label for the disqualification rule targeting specific uploaders.", + "montage-round-dq-coords": "Label for the disqualification rule targeting coordinators.", + "montage-round-dq-maintainers": "Label for the disqualification rule targeting maintainers.", + "montage-round-dq-organizers": "Label for the disqualification rule targeting organizers.", + "montage-round-final-threshold": "Label for the final threshold setting in a round.", + "montage-round-min-resolution": "Label for the minimum resolution field in a round.", + "montage-round-show-filename": "Label for the option to show the filename in a round.", + "montage-round-show-link": "Label for the option to show a link in a round.", + "montage-round-show-resolution": "Label for the option to show the resolution in a round.", + "montage-round-vote-ending": "Template for displaying when voting will end in a round, with a placeholder for days remaining.", + "montage-round-no-direction-given": "Message displayed when no directions are provided for a round.", + "montage-round-ranking": "Label for the ranking voting method in a round.", + "montage-round-rating": "Label for the rating voting method in a round.", + "montage-round-yesno": "Label for the Yes/No voting method in a round.", + "montage-round-file-info": "Label for the file information section in a round.", + "montage-round-file-type": "Label for the file type field in a round.", + "montage-round-voting-details": "Label for the voting details section in a round.", + "montage-round-activate": "Label for the action to activate a round.", + "montage-round-activated": "Notification message indicating a round has been successfully activated.", + "montage-round-pause": "Label for the action to pause a round.", + "montage-round-paused": "Notification message indicating a round has been successfully paused.", + "montage-round-finalize": "Label for the action to finalize a round.", + "montage-round-finalized": "Notification message indicating a round has been successfully finalized.", + "montage-round-download-results": "Label for the action to download round results.", + "montage-round-download-entries": "Label for the action to download entries for a round.", + "montage-round-source": "Label for the source field in a round.", + "montage-round-source-category": "Label for the Wikimedia Commons category source option.", + "montage-round-source-csv": "Label for the CSV file list source option.", + "montage-round-source-filelist": "Label for the file list source option.", + "montage-round-source-category-help": "Help text explaining the Wikimedia Commons category source option.", + "montage-round-source-csv-help": "Help text explaining the CSV file list source option.", + "montage-round-source-selected-help": "Help text explaining the selected file list source option.", + "montage-round-category-placeholder": "Placeholder text for the category input field.", + "montage-round-category-label": "Label for the category input field.", + "montage-round-no-category": "Message displayed when no categories are found.", + "montage-round-file-url": "Label for the file URL input field.", + "montage-round-file-list": "Label for the file list input field.", + "montage-round-threshold": "Label for the threshold setting in a round.", + "montage-round-vote-method": "Label for the vote method setting in a round.", + "montage-round-threshold-description": "Description for the threshold setting, explaining its purpose.", + "montage-round-threshold-default": "Placeholder text for the threshold setting.", + "montage-round-jurors-description": "Description for the jurors input field, explaining its purpose.", + "montage-round-quorum-description": "Description for the quorum input field, explaining its purpose.", + "montage-no-results": "Message displayed when no results are found.", + "montage-vote-round-inactive": "Message displayed when a round is inactive.", + "montage-vote-contact-organizer": "Message instructing the user to contact the organizer for an inactive round.", + "montage-vote-show-full-size": "Label for the action to view an image in full size.", + "montage-vote-commons-page": "Label for the action to view the Commons page for an image.", + "montage-vote-accept": "Label for the action to accept an image.", + "montage-vote-decline": "Label for the action to decline an image.", + "montage-vote-keyboard-instructions": "Instructions for voting using the keyboard.", + "montage-vote-actions": "Label for the actions section during voting.", + "montage-vote-add-favorites": "Label for the action to add an image to favorites.", + "montage-vote-added-favorites": "Notification message indicating an image has been added to favorites.", + "montage-vote-remove-favorites": "Label for the action to remove an image from favorites.", + "montage-vote-removed-favorites": "Notification message indicating an image has been removed from favorites.", + "montage-vote-skip": "Label for the action to skip voting on an image.", + "montage-vote-description": "Label for the description section for an image.", + "montage-vote-version": "Label for the version field for an image.", + "montage-vote-last-version": "Template for displaying the last version of an image with a timestamp.", + "montage-vote-all-done": "Message displayed when the user has finished voting on all images.", + "montage-vote-no-images": "Message displayed when there are no images left to vote on.", + "montage-vote-hide-panel": "Label for the action to hide the voting panel.", + "montage-vote-show-panel": "Label for the action to show the voting panel.", + "montage-vote-image": "Label for the image field during voting.", + "montage-vote-image-remains": "Template for displaying the number of images remaining to be voted on.", + "montage-vote-rating-instructions": "Instructions for rating images, explaining the rating scale.", + "montage-vote-round-part-of-campaign": "Template indicating the campaign a round is part of.", + "montage-vote-grid-size-large": "Label for selecting a large grid size.", + "montage-vote-grid-size-medium": "Label for selecting a medium grid size.", + "montage-vote-grid-size-small": "Label for selecting a small grid size.", + "montage-vote-image-review": "Label for the image review section.", + "montage-vote-ordinal-place": "Template for displaying an ordinal place, with a placeholder for the rank.", + "montage-vote-order-by": "Label for the order-by field during voting.", + "montage-vote-gallery-size": "Label for the gallery size field during voting.", + "montage-voted-time": "Template for displaying the time an image was voted on.", + "montage-no-votes-yet": "Message displayed when no votes have been cast yet.", + "montage-no-votes-this-round": "Message displayed when no votes have been cast in the current round.", + "montage-round-added": "Message displayed when a round is successfully added.", + "montage-round-show-stats": "Label to indicate the option to show statistical information.", + "montage-option-yes": "Label for a 'Yes' option in various UI contexts.", + "montage-option-no": "Label for a 'No' option in various UI contexts.", + "montage-vote-edit-for": "Used in the heading to indicate that votes are being edited for a specific round. {0} is the placeholder for the round name.", + "montage-add-organizer": "Label for a button or dialog title to add a new campaign organizer.", + "montage-btn-add": "Text for a button to add an item or user.", + "montage-added-organizer": "Notification message shown when an organizer is successfully added.", + "montage-at-least-one-user": "Error message prompting the user to add at least one user.", + "montage-only-one-user": "Error message indicating that only one user can be added.", + "montage-round-quorum-per-photo": "Text indicating the number of jurors required per photo in a round. {0} is a placeholder for the number of jurors.", + "montage-round-open-task-percentage": "Label for the percentage of open tasks in a round.", + "montage-round-cancelled-tasks": "Label for the number of tasks canceled in a round.", + "montage-round-disqualified-files": "Label for the number of files disqualified in a round.", + "montage-round-open-tasks": "Label for the number of tasks currently open in a round.", + "montage-round-files": "Label for the number of files in a round.", + "montage-round-tasks": "Label for the number of tasks in a round.", + "montage-round-uploaders": "Label for the number of uploaders contributing files to a round." + } \ No newline at end of file diff --git a/frontend/src/main.js b/frontend/src/main.js new file mode 100644 index 00000000..896b4f3d --- /dev/null +++ b/frontend/src/main.js @@ -0,0 +1,28 @@ +import { createApp } from 'vue' +import { createPinia } from 'pinia' + +import App from './App.vue' +import router from './router' +import initializeI18n from './i18n' +import Toast from 'vue-toastification' +import 'vue-toastification/dist/index.css' + +import { CdxTooltip } from '@wikimedia/codex' +import ClipLoader from 'vue-spinner/src/ClipLoader.vue' + +(async () => { + // Wait for i18n initialization + const i18n = await initializeI18n() + const app = createApp(App) + + app.use(createPinia()) + app.use(router) + app.use(Toast) + app.use(i18n) + + // Global directive + app.directive('tooltip', CdxTooltip) + app.component('clip-loader', ClipLoader) + + app.mount('#app') +})() diff --git a/frontend/src/router/index.js b/frontend/src/router/index.js new file mode 100644 index 00000000..588f2a70 --- /dev/null +++ b/frontend/src/router/index.js @@ -0,0 +1,68 @@ +import '@wikimedia/codex/dist/codex.style.css' +import '@/assets/main.css' + +import { createRouter, createWebHashHistory } from 'vue-router' +import { useUserStore } from '@/stores/user' + +import HomeView from '../views/HomeView.vue' +import NewCampaignView from '@/views/NewCampaignView.vue' +import CampaignView from '@/views/CampaignView.vue' +import VoteView from '@/views/VoteView.vue' +import VoteEditView from '@/views/VoteEditView.vue' +import AllCampaignView from '@/views/AllCampaignView.vue' + +const routes = [ + { + path: '/', + name: 'home', + component: HomeView + }, + { + path: '/campaign/all', + name: 'campaign-all', + component: AllCampaignView, + meta: { requiresAuth: true } + }, + { + path: '/campaign/new', + name: 'new-campaign', + component: NewCampaignView, + meta: { requiresAuth: true } + }, + { + path: '/campaign/:id', + name: 'campaign', + component: CampaignView, + meta: { requiresAuth: true } + }, + { + path: '/vote/:id', + name: 'vote', + component: VoteView, + meta: { requiresAuth: true } + }, + { + path: '/vote/:id/edit', + name: 'vote-edit', + component: VoteEditView, + meta: { requiresAuth: true } + } +] + +const router = createRouter({ + history: createWebHashHistory(import.meta.env.BASE_URL), + routes +}) + +router.beforeEach(async (to, from, next) => { + const userStore = useUserStore() + await userStore.checkAuth() + + if (to.meta.requiresAuth && userStore.user === null) { + return next({ name: 'home' }) + } + + next() +}) + +export default router diff --git a/frontend/src/services/adminService.js b/frontend/src/services/adminService.js new file mode 100644 index 00000000..75b3ca08 --- /dev/null +++ b/frontend/src/services/adminService.js @@ -0,0 +1,56 @@ +import { apiBackend } from './api' + +const adminService = { + get: () => apiBackend.get('admin'), + + allCampaigns: () => apiBackend.get('admin/campaigns/all'), + + getCampaign: (id) => apiBackend.get(`admin/campaign/${id}`), + + getRound: (id) => apiBackend.get(`admin/round/${id}`), + + getReviews: (id) => apiBackend.get(`admin/round/${id}/reviews`), + + addOrganizer: (data) => apiBackend.post('admin/add_organizer', data), + + addCampaign: (data) => apiBackend.post('admin/add_campaign', data), + + addRound: (id, data) => apiBackend.post(`admin/campaign/${id}/add_round`, data), + + finalizeCampaign: (id) => apiBackend.post(`admin/campaign/${id}/finalize`, { post: true }), + + addCoordinator: (id, username) => + apiBackend.post(`admin/campaign/${id}/add_coordinator`, { username }), + + removeCoordinator: (id, username) => + apiBackend.post(`admin/campaign/${id}/remove_coordinator`, { username }), + + activateRound: (id) => apiBackend.post(`admin/round/${id}/activate`, { post: true }), + + pauseRound: (id) => apiBackend.post(`admin/round/${id}/pause`, { post: true }), + + populateRound: (id, data) => apiBackend.post(`admin/round/${id}/import`, data), + + editCampaign: (id, data) => apiBackend.post(`admin/campaign/${id}/edit`, data), + + editRound: (id, data) => apiBackend.post(`admin/round/${id}/edit`, data), + + cancelRound: (id) => apiBackend.post(`admin/round/${id}/cancel`), + + getRoundFlags: (id) => apiBackend.get(`admin/round/${id}/flags`), + + getRoundReviews: (id) => apiBackend.get(`admin/round/${id}/reviews`), + + getRoundVotes: (id) => apiBackend.get(`admin/round/${id}/votes`), + + previewRound: (id) => apiBackend.get(`admin/round/${id}/preview_results`), + + advanceRound: (id, data) => apiBackend.post(`admin/round/${id}/advance`, data), + + // Direct download URLs (manual baseURL needed) + downloadRound: (id) => `${apiBackend.defaults.baseURL}admin/round/${id}/results/download`, + downloadEntries: (id) => `${apiBackend.defaults.baseURL}admin/round/${id}/entries/download`, + downloadReviews: (id) => `${apiBackend.defaults.baseURL}admin/round/${id}/reviews` +} + +export default adminService diff --git a/frontend/src/services/alertService.js b/frontend/src/services/alertService.js new file mode 100644 index 00000000..072e34cb --- /dev/null +++ b/frontend/src/services/alertService.js @@ -0,0 +1,28 @@ +import { useToast } from 'vue-toastification' + +const toast = useToast() + +const AlertService = { + success(text, time, callback) { + toast.success(text, { + timeout: time || 2000, + position: 'top-right', + onClose: () => { + callback && callback() + } + }) + }, + error(error, time) { + const message = error?.response?.data?.message || error?.message || 'An error occurred' + const detail = error?.response?.data?.detail + + const text = detail ? `${message}: ${detail}` : message + + toast.error(text, { + timeout: time || 5000, + position: 'top-right' + }) + } +} + +export default AlertService diff --git a/frontend/src/services/api.js b/frontend/src/services/api.js new file mode 100644 index 00000000..923cb5c9 --- /dev/null +++ b/frontend/src/services/api.js @@ -0,0 +1,60 @@ +import axios from 'axios' +import { useLoadingStore } from '@/stores/loading' + +// Create Axios instance for Backend API +const apiBackend = axios.create({ + baseURL: '/v1/', + timeout: 10000, + headers: { + 'Content-Type': 'application/json', + Accept: 'application/json' + } +}) + +// Create Axios instance for Commons API +const apiCommons = axios.create({ + baseURL: 'https://commons.wikimedia.org/w/api.php', + timeout: 10000, + headers: { + 'Content-Type': 'application/json', + Accept: 'application/json' + } +}) + +const addInterceptors = (instance) => { + instance.interceptors.request.use( + (config) => { + const loadingStore = useLoadingStore() + loadingStore.setLoading(true) + + return config + }, + (error) => { + const loadingStore = useLoadingStore() + loadingStore.setLoading(false) + + return Promise.reject(error) + } + ) + + // Response Interceptor + instance.interceptors.response.use( + (response) => { + const loadingStore = useLoadingStore() + loadingStore.setLoading(false) + + return response['data'] + }, + (error) => { + const loadingStore = useLoadingStore() + loadingStore.setLoading(false) + + return Promise.reject(error) + } + ) +} + +addInterceptors(apiBackend) +addInterceptors(apiCommons) + +export { apiBackend, apiCommons } diff --git a/frontend/src/services/dataService.js b/frontend/src/services/dataService.js new file mode 100644 index 00000000..a15b9852 --- /dev/null +++ b/frontend/src/services/dataService.js @@ -0,0 +1,75 @@ +import { apiCommons } from './api' + +const dataService = { + async getImageInfo(images) { + const parts = Math.ceil(images.length / 50) + let promises = [] + + for (let i = 0; i < parts; i++) { + const part = images.slice(50 * i, 50 * i + 50) + promises.push( + apiCommons({ + method: 'GET', + params: { + action: 'query', + prop: 'imageinfo', + titles: part.map((image) => 'File:' + image).join('|'), + format: 'json', + iiprop: 'timestamp|user|userid|size|dimensions|url', + iilimit: '10', + origin: '*' + } + }) + ) + } + + try { + return await Promise.all(promises) + } catch (error) { + console.error('Error fetching image info:', error) + throw error + } + }, + + async searchUser(username) { + try { + const response = await apiCommons({ + method: 'GET', + params: { + action: 'query', + list: 'globalallusers', + format: 'json', + rawcontinue: 'true', + agufrom: username, + origin: '*' + } + }) + return response + } catch (error) { + console.error('Error searching for user:', error) + throw error + } + }, + + async searchCategory(category) { + try { + const response = await apiCommons({ + method: 'GET', + params: { + action: 'opensearch', + format: 'json', + namespace: '14', + limit: '10', + search: category, + origin: '*' + } + }) + return response + } catch (error) { + console.error('Error searching for category:', error) + throw error + } + } +} + +export default dataService diff --git a/frontend/src/services/dialogService.js b/frontend/src/services/dialogService.js new file mode 100644 index 00000000..74b98928 --- /dev/null +++ b/frontend/src/services/dialogService.js @@ -0,0 +1,67 @@ +import { ref, defineComponent, h, render, getCurrentInstance } from 'vue'; +import { CdxDialog } from '@wikimedia/codex'; + +const dialogService = () => { + const open = ref(false); + const dialogConfig = ref({}); + + const show = (config) => { + dialogConfig.value = config; + open.value = true; + }; + + const DialogComponent = defineComponent({ + setup() { + const { appContext } = getCurrentInstance(); + + const onPrimaryAction = () => { + open.value = false; + if (dialogConfig.value.onPrimary) { + dialogConfig.value.onPrimary(); + } + }; + + const onDefaultAction = () => { + open.value = false; + if (dialogConfig.value.onDefault) { + dialogConfig.value.onDefault(); + } + }; + + return () => h(CdxDialog, { + appContext: appContext, + open: open.value, + 'onUpdate:open': (value) => open.value = value, + title: dialogConfig.value.title, + useCloseButton: true, + primaryAction: dialogConfig.value.primaryAction, + defaultAction: dialogConfig.value.defaultAction, + onPrimary: onPrimaryAction, + onDefault: onDefaultAction + }, { + default: () => { + if (typeof dialogConfig.value.content === 'string') { + return h('div', { innerHTML: dialogConfig.value.content }); + } else if (dialogConfig.value.content) { + return h(dialogConfig.value.content, dialogConfig.value.props || {}); + } + return null; + } + }); + } + }); + + const mountDialog = () => { + const container = document.createElement('div'); + document.body.appendChild(container); + render(h(DialogComponent), container); + }; + + mountDialog(); + + return { + show + }; +}; + +export default dialogService; \ No newline at end of file diff --git a/frontend/src/services/jurorService.js b/frontend/src/services/jurorService.js new file mode 100644 index 00000000..01422374 --- /dev/null +++ b/frontend/src/services/jurorService.js @@ -0,0 +1,58 @@ +import { apiBackend } from './api' +import _ from 'lodash' +import dataService from './dataService' + +const jurorService = { + get: () => apiBackend.get('juror'), + + getCampaign: (id) => apiBackend.get(`juror/campaign/${id}`), + + allCampaigns: () => apiBackend.get('juror/campaigns/all'), + + getPastVotes: (id, offset = 0, orderBy = 'date', sort = 'desc') => + apiBackend.get(`juror/round/${id}/votes?offset=${offset}&order_by=${orderBy}&sort=${sort}`), + + getPastRanking: (id) => apiBackend.get(`juror/round/${id}/rankings`), + + getFaves: () => apiBackend.get('juror/faves'), + + getRound: (id) => apiBackend.get(`juror/round/${id}`), + + getRoundVotesStats: (id) => apiBackend.get(`juror/round/${id}/votes-stats`), + + faveImage: (roundId, entryId) => apiBackend.post(`juror/round/${roundId}/${entryId}/fave`, {}), + + unfaveImage: (roundId, entryId) => + apiBackend.post(`juror/round/${roundId}/${entryId}/unfave`, {}), + + flagImage: (roundId, entryId, reason) => + apiBackend.post(`juror/round/${roundId}/${entryId}/flag`, { reason }), + + setRating: (id, data) => apiBackend.post(`juror/round/${id}/tasks/submit`, data), + + getRoundTasks: (id, offset = 0) => { + return apiBackend.get(`juror/round/${id}/tasks?count=10&offset=${offset}`).then((data) => { + const tasks = data.data.tasks + const files = tasks.map((task) => task.entry.name) + + return dataService.getImageInfo(files).then((responses) => { + if (!responses.length) return data + + const hists = _.values(responses[0].query.pages) + hists.forEach((element) => { + if (element && element.imageinfo) { + const image = _.find(tasks, { + entry: { url: element.imageinfo[0].url } + }) + if (image) { + image.history = element.imageinfo + } + } + }) + return data + }) + }) + } +} + +export default jurorService diff --git a/frontend/src/stores/loading.js b/frontend/src/stores/loading.js new file mode 100644 index 00000000..1640ac5c --- /dev/null +++ b/frontend/src/stores/loading.js @@ -0,0 +1,12 @@ +import { ref } from 'vue' +import { defineStore } from 'pinia' + +export const useLoadingStore = defineStore('loading', () => { + const loading = ref(null) + + function setLoading(val) { + loading.value = val + } + + return { loading, setLoading } +}) diff --git a/frontend/src/stores/user.js b/frontend/src/stores/user.js new file mode 100644 index 00000000..cf1c562b --- /dev/null +++ b/frontend/src/stores/user.js @@ -0,0 +1,33 @@ +import { ref } from 'vue' +import { defineStore } from 'pinia' +import adminService from '@/services/adminService' + +export const useUserStore = defineStore('user-store', () => { + const user = ref(null) + const isAuthenticated = ref(false) + + function login(userObj) { + if (!userObj) { + window.location = window.location.origin + '/login' + } + user.value = userObj + isAuthenticated.value = true + } + + function logout() { + window.location = window.location.origin + '/logout' + user.value = null + isAuthenticated.value = false + } + + async function checkAuth() { + if (!isAuthenticated.value) { + const res = await adminService.get() + if (res.status === 'success' && res.user) { + login(res.user) + } + } + } + + return { user, login, logout, checkAuth } +}) diff --git a/frontend/src/utils.js b/frontend/src/utils.js new file mode 100644 index 00000000..e18cb235 --- /dev/null +++ b/frontend/src/utils.js @@ -0,0 +1,63 @@ +export function formatDate(dateString) { + const options = { year: 'numeric', month: 'short', day: 'numeric' } + return new Date(dateString).toLocaleDateString('en-US', options) +} + +export function getVotingName(voting) { + const types = { + "yesno": "montage-round-yesno", + "rating": "montage-round-rating", + "ranking": "montage-round-ranking", + } + + return types[voting] +} + +export function getAvatarColor(username) { + const colors = [ + '#1abc9c', + '#2ecc71', + '#3498db', + '#9b59b6', + '#34495e', + '#16a085', + '#27ae60', + '#2980b9', + '#8e44ad', + '#2c3e50', + '#f1c40f', + '#e67e22', + '#e74c3c', + '#95a5a6', + '#f39c12', + '#d35400', + '#c0392b', + '#bdc3c7', + '#7f8c8d' + ] + + const sum = stringToColor(username) + const color = colors[sum % colors.length] + const rgba = hexToRgba(color, 0.5) + + return rgba +} + +function stringToColor(str) { + let hash = 0 + for (let char of str) { + hash = char.charCodeAt(0) + ((hash << 5) - hash) + } + return Math.abs(hash % 19) +} + +function hexToRgba(hex, alpha) { + const r = parseInt(cutHex(hex).substring(0, 2), 16) + const g = parseInt(cutHex(hex).substring(2, 4), 16) + const b = parseInt(cutHex(hex).substring(4, 6), 16) + return `rgba(${r}, ${g}, ${b}, ${alpha})` +} + +function cutHex(h) { + return h.startsWith('#') ? h.substring(1, 7) : h +} diff --git a/frontend/src/views/AllCampaignView.vue b/frontend/src/views/AllCampaignView.vue new file mode 100644 index 00000000..c56988de --- /dev/null +++ b/frontend/src/views/AllCampaignView.vue @@ -0,0 +1,7 @@ + + + diff --git a/frontend/src/views/CampaignView.vue b/frontend/src/views/CampaignView.vue new file mode 100644 index 00000000..3fe7bbb2 --- /dev/null +++ b/frontend/src/views/CampaignView.vue @@ -0,0 +1,9 @@ + + + diff --git a/frontend/src/views/HomeView.vue b/frontend/src/views/HomeView.vue new file mode 100644 index 00000000..e8fc6c90 --- /dev/null +++ b/frontend/src/views/HomeView.vue @@ -0,0 +1,14 @@ + + + diff --git a/frontend/src/views/NewCampaignView.vue b/frontend/src/views/NewCampaignView.vue new file mode 100644 index 00000000..1ed7622b --- /dev/null +++ b/frontend/src/views/NewCampaignView.vue @@ -0,0 +1,7 @@ + + + \ No newline at end of file diff --git a/frontend/src/views/VoteEditView.vue b/frontend/src/views/VoteEditView.vue new file mode 100644 index 00000000..f619cc94 --- /dev/null +++ b/frontend/src/views/VoteEditView.vue @@ -0,0 +1,15 @@ + + + + + diff --git a/frontend/src/views/VoteView.vue b/frontend/src/views/VoteView.vue new file mode 100644 index 00000000..afe7a401 --- /dev/null +++ b/frontend/src/views/VoteView.vue @@ -0,0 +1,16 @@ + + + + + diff --git a/frontend/vite.config.js b/frontend/vite.config.js new file mode 100644 index 00000000..5c45e1d9 --- /dev/null +++ b/frontend/vite.config.js @@ -0,0 +1,16 @@ +import { fileURLToPath, URL } from 'node:url' + +import { defineConfig } from 'vite' +import vue from '@vitejs/plugin-vue' + +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [ + vue(), + ], + resolve: { + alias: { + '@': fileURLToPath(new URL('./src', import.meta.url)) + } + } +}) diff --git a/montage/rdb.py b/montage/rdb.py index 6b59e2a2..3a6e4979 100644 --- a/montage/rdb.py +++ b/montage/rdb.py @@ -116,7 +116,7 @@ flags attribute. """ -MAINTAINERS = ['MahmoudHashemi', 'Slaporte', 'Yarl', 'LilyOfTheWest'] +MAINTAINERS = ['MahmoudHashemi', 'Slaporte', 'Yarl', 'LilyOfTheWest', 'Jayprakash12345'] """ class RDBSession(sessionmaker()): diff --git a/requirements.txt b/requirements.txt index 0d95e706..9422b154 100644 --- a/requirements.txt +++ b/requirements.txt @@ -4,7 +4,7 @@ # # pip-compile --annotation-style=line requirements.in # --e git+https://github.com/the-maldridge/python-vote-core.git@f0b01e7e24f80673c4c237ee9e6118e8986cf0bb#egg=python3_vote_core # via -r requirements.in +git+https://github.com/the-maldridge/python-vote-core.git@f0b01e7e24f80673c4c237ee9e6118e8986cf0bb#egg=python3_vote_core # via -r requirements.in ashes==24.0.0 # via -r requirements.in, chert, clastic attrs==23.2.0 # via clastic, glom boltons==24.0.0 # via -r requirements.in, chert, clastic, face, glom, lithoxyl