From a36f96c29061fa55c8ce2373bf9b159279b52a68 Mon Sep 17 00:00:00 2001 From: Shop Deploy Date: Thu, 11 Jun 2026 13:48:08 +0500 Subject: [PATCH] initial: client --- .gitignore | 15 + AGENTS.md | 39 + README.md | 23 + client/.editorconfig | 12 + client/.gitignore | 24 + client/.prettierignore | 6 + client/.prettierrc.json | 9 + client/README.md | 73 + client/eslint.config.js | 204 + client/index.html | 30 + client/package-lock.json | 10680 ++++++++++++++++ client/package.json | 82 + client/public/apple-touch-icon.png | Bin 0 -> 4629 bytes client/public/favicon-128.png | Bin 0 -> 2796 bytes client/public/favicon-32.png | Bin 0 -> 516 bytes client/public/favicon-48.png | Bin 0 -> 774 bytes client/public/favicon.svg | 1 + client/public/fonts/Outfit-Bold.woff2 | Bin 0 -> 14060 bytes client/public/fonts/Outfit-Medium.woff2 | Bin 0 -> 13528 bytes client/public/fonts/Outfit-Regular.woff2 | Bin 0 -> 14032 bytes client/public/fonts/Outfit-SemiBold.woff2 | Bin 0 -> 14140 bytes client/public/icons.svg | 24 + client/public/logo.webp | Bin 0 -> 23522 bytes client/public/politik.txt | 106 + client/public/robots.txt | 4 + client/public/sitemap.xml | 28 + client/src/app/App.tsx | 22 + client/src/app/layout/AppHeader.tsx | 202 + client/src/app/layout/MainLayout.tsx | 144 + .../app/layout/__tests__/MainLayout.test.tsx | 36 + client/src/app/providers/AppProviders.tsx | 361 + client/src/app/providers/SseProvider.tsx | 83 + .../providers/__tests__/SseProvider.test.tsx | 159 + client/src/app/providers/theme-controller.tsx | 113 + client/src/app/routes/index.tsx | 135 + client/src/app/styles/global.css | 45 + client/src/assets/hero.png | Bin 0 -> 13057 bytes client/src/assets/react.svg | 1 + client/src/assets/vite.svg | 1 + client/src/entities/cart/api/cart-api.ts | 21 + client/src/entities/cart/index.ts | 3 + .../src/entities/cart/lib/use-cart-query.ts | 14 + client/src/entities/cart/model/types.ts | 7 + .../catalog-slider/api/catalog-slider-api.ts | 19 + client/src/entities/catalog-slider/index.ts | 2 + .../entities/catalog-slider/model/types.ts | 10 + .../src/entities/gallery/api/gallery-api.ts | 52 + client/src/entities/gallery/index.ts | 3 + client/src/entities/gallery/model/types.ts | 7 + .../src/entities/gallery/ui/GalleryGrid.tsx | 100 + .../notification/api/notifications-api.ts | 54 + client/src/entities/notification/index.ts | 7 + .../src/entities/order/api/admin-order-api.ts | 96 + client/src/entities/order/api/order-api.ts | 103 + client/src/entities/order/index.ts | 9 + .../entities/product/api/admin-product-api.ts | 108 + .../src/entities/product/api/product-api.ts | 42 + client/src/entities/product/index.ts | 2 + client/src/entities/product/model/types.ts | 33 + .../src/entities/product/ui/ProductCard.tsx | 264 + .../entities/review/api/admin-review-api.ts | 32 + client/src/entities/review/api/reviews-api.ts | 78 + client/src/entities/review/index.ts | 12 + .../test-checklist/api/test-checklist-api.ts | 40 + client/src/entities/user/api/address-api.ts | 49 + client/src/entities/user/api/messages-api.ts | 24 + client/src/entities/user/api/user-api.ts | 45 + client/src/entities/user/index.ts | 10 + client/src/entities/user/model/types.ts | 23 + client/src/features/address-form/index.ts | 2 + .../address-form/ui/AddressFormDialog.tsx | 127 + .../address-map-picker/api/map-geocoding.ts | 28 + .../src/features/address-map-picker/index.ts | 1 + .../address-map-picker/model/types.ts | 3 + .../ui/AddressMapPicker.tsx | 144 + .../address-map-picker/ui/MapPickerMap.tsx | 177 + client/src/features/auth-code/index.ts | 1 + .../auth-code/ui/AuthCodeForm.test.tsx | 67 + .../features/auth-code/ui/AuthCodeForm.tsx | 115 + client/src/features/auth-forgot/index.ts | 1 + .../auth-forgot/ui/AuthForgotForm.tsx | 145 + .../__tests__/OAuthButtons.test.tsx | 19 + client/src/features/auth-oauth/index.ts | 1 + .../auth-oauth/lib/oauth-providers.ts | 24 + .../features/auth-oauth/ui/OAuthButtons.tsx | 28 + client/src/features/auth-password/index.ts | 1 + .../ui/AuthPasswordForm.test.tsx | 74 + .../auth-password/ui/AuthPasswordForm.tsx | 234 + client/src/features/cart/add-to-cart/index.ts | 1 + .../cart/add-to-cart/ui/AddToCartButton.tsx | 46 + .../ui/__tests__/AddToCartButton.test.tsx | 43 + client/src/features/cart/cart-badge/index.ts | 1 + .../features/cart/cart-badge/ui/CartBadge.tsx | 31 + .../features/cart/toggle-cart-icon/index.ts | 1 + .../toggle-cart-icon/ui/ToggleCartIcon.tsx | 77 + .../ui/__tests__/ToggleCartIcon.test.tsx | 74 + client/src/features/order-chat/index.ts | 1 + .../src/features/order-chat/ui/OrderChat.tsx | 87 + client/src/features/order-detail/index.ts | 2 + .../ui/DeliveryFeeAdjustmentForm.tsx | 55 + .../order-detail/ui/OrderDetailContent.tsx | 245 + .../ui/__tests__/OrderDetailContent.test.tsx | 211 + client/src/features/order-payment/index.ts | 1 + .../order-payment/ui/OrderPaymentSection.tsx | 65 + client/src/features/product-form/index.ts | 2 + .../use-product-form-helpers.test.ts | 24 + .../lib/use-product-form-helpers.ts | 30 + .../src/features/product-form/model/types.ts | 12 + .../product-form/ui/GalleryImagePicker.tsx | 142 + .../product-form/ui/ProductFormFields.tsx | 235 + client/src/features/product-review/index.ts | 3 + .../product-review/ui/ProductReviewsList.tsx | 110 + .../product-review/ui/ReviewDialog.tsx | 164 + .../product-review/ui/ReviewSection.tsx | 101 + client/src/features/user/user-menu/index.ts | 1 + .../features/user/user-menu/ui/UserMenu.tsx | 72 + client/src/main.tsx | 13 + client/src/pages/about/index.ts | 1 + client/src/pages/about/ui/AboutMap.tsx | 106 + client/src/pages/about/ui/AboutPage.tsx | 58 + client/src/pages/admin-categories/index.ts | 1 + .../ui/AdminCategoriesPage.tsx | 295 + client/src/pages/admin-gallery/index.ts | 1 + .../admin-gallery/ui/AdminGalleryPage.tsx | 252 + .../admin-gallery/ui/GallerySliderSection.tsx | 190 + client/src/pages/admin-layout/index.ts | 1 + .../pages/admin-layout/ui/AdminLayoutPage.tsx | 218 + .../ui/AdminNotificationsPage.tsx | 132 + client/src/pages/admin-orders/index.ts | 1 + .../pages/admin-orders/ui/AdminOrdersPage.tsx | 194 + .../ui/__tests__/AdminOrdersPage.test.tsx | 126 + client/src/pages/admin-products/index.ts | 1 + .../admin-products/ui/AdminProductsPage.tsx | 272 + .../ui/__tests__/AdminProductsPage.test.tsx | 77 + client/src/pages/admin-reviews/index.ts | 1 + .../admin-reviews/ui/AdminReviewsPage.tsx | 103 + client/src/pages/admin-settings/index.ts | 1 + .../admin-settings/ui/AdminSettingsPage.tsx | 216 + client/src/pages/admin-slider/index.ts | 1 + .../pages/admin-slider/ui/AdminSliderPage.tsx | 250 + .../src/pages/admin-test-checklist/index.ts | 1 + .../ui/AdminTestChecklistPage.tsx | 280 + client/src/pages/admin-users/index.ts | 1 + .../pages/admin-users/ui/AdminUsersPage.tsx | 269 + .../pages/auth/__tests__/AuthPage.test.tsx | 64 + client/src/pages/auth/index.ts | 2 + client/src/pages/auth/ui/AuthCallbackPage.tsx | 28 + client/src/pages/auth/ui/AuthPage.tsx | 190 + client/src/pages/cart/index.ts | 1 + client/src/pages/cart/ui/CartPage.tsx | 184 + client/src/pages/checkout/index.ts | 1 + client/src/pages/checkout/ui/CheckoutPage.tsx | 273 + client/src/pages/home/index.ts | 1 + .../src/pages/home/lib/use-product-filters.ts | 94 + client/src/pages/home/ui/HomePage.tsx | 160 + client/src/pages/home/ui/ProductFilters.tsx | 188 + client/src/pages/info/index.ts | 1 + client/src/pages/info/ui/InfoPage.tsx | 70 + .../info/ui/sections/DeliverySection.tsx | 172 + .../info/ui/sections/HowToOrderSection.tsx | 233 + .../info/ui/sections/OrderStatusesSection.tsx | 247 + .../pages/info/ui/sections/PaymentSection.tsx | 105 + .../pages/info/ui/sections/ReturnsSection.tsx | 71 + client/src/pages/me/index.ts | 1 + client/src/pages/me/ui/MeLayoutPage.tsx | 202 + .../pages/me/ui/sections/AddressesPage.tsx | 223 + .../me/ui/sections/AuthMethodsSection.tsx | 296 + .../pages/me/ui/sections/AvatarSection.tsx | 119 + .../me/ui/sections/DeleteAccountSection.tsx | 92 + .../src/pages/me/ui/sections/MessagesPage.tsx | 228 + .../me/ui/sections/NotificationsPage.tsx | 125 + .../pages/me/ui/sections/OrderDetailPage.tsx | 307 + .../src/pages/me/ui/sections/OrdersPage.tsx | 81 + .../pages/me/ui/sections/ProfileSection.tsx | 60 + .../src/pages/me/ui/sections/SettingsPage.tsx | 45 + .../__tests__/AuthMethodsSection.test.tsx | 48 + .../sections/__tests__/AvatarSection.test.tsx | 36 + .../__tests__/ProfileSection.test.tsx | 23 + client/src/pages/not-found/index.ts | 1 + .../src/pages/not-found/ui/NotFoundPage.tsx | 64 + .../content/privacy-policy.template.txt | 106 + client/src/pages/privacy-policy/index.ts | 1 + .../privacy-policy/ui/PrivacyPolicyPage.tsx | 151 + client/src/pages/product/index.ts | 1 + client/src/pages/product/ui/ProductPage.tsx | 231 + client/src/pages/terms/index.ts | 1 + client/src/pages/terms/ui/TermsPage.tsx | 207 + client/src/shared/api/client.ts | 34 + client/src/shared/assets/vk-logo.svg | 11 + client/src/shared/config/index.ts | 27 + .../shared/constants/__tests__/order.test.ts | 17 + .../src/shared/constants/delivery-carrier.ts | 19 + client/src/shared/constants/order.ts | 17 + client/src/shared/constants/pickup-point.ts | 9 + client/src/shared/constants/upload-limits.ts | 13 + .../shared/lib/__tests__/format-price.test.ts | 22 + .../__tests__/get-api-error-message.test.ts | 35 + .../lib/__tests__/get-error-message.test.ts | 20 + .../__tests__/group-orders-by-status.test.ts | 33 + .../lib/__tests__/is-synthetic-email.test.ts | 24 + .../order-requires-price-approval.test.ts | 44 + .../use-mutation-with-toast.test.tsx | 75 + client/src/shared/lib/avatar-styles.ts | 54 + client/src/shared/lib/create-error-store.ts | 10 + client/src/shared/lib/format-price.ts | 7 + .../src/shared/lib/get-api-error-message.ts | 23 + client/src/shared/lib/get-error-message.ts | 8 + .../src/shared/lib/get-original-webp-url.ts | 14 + .../src/shared/lib/group-orders-by-status.ts | 24 + .../src/shared/lib/invalidate-query-keys.ts | 5 + client/src/shared/lib/is-synthetic-email.ts | 5 + client/src/shared/lib/oauth-authorize-url.ts | 7 + .../src/shared/lib/order-address-snapshot.ts | 19 + .../lib/order-requires-price-approval.ts | 11 + client/src/shared/lib/order-status-data.ts | 79 + client/src/shared/lib/persist-token.ts | 27 + client/src/shared/lib/reviews-count-ru.ts | 12 + client/src/shared/lib/sse.ts | 3 + .../src/shared/lib/use-edit-dialog-state.ts | 28 + .../src/shared/lib/use-mutation-with-toast.ts | 42 + client/src/shared/lib/use-page-title.ts | 24 + .../model/__tests__/notification.test.ts | 97 + client/src/shared/model/auth.ts | 122 + client/src/shared/model/notification.ts | 41 + client/src/shared/model/theme.ts | 5 + .../src/shared/ui/AdminDialog/AdminDialog.tsx | 36 + .../src/shared/ui/AdminTable/AdminTable.tsx | 61 + client/src/shared/ui/AdminTable/index.ts | 2 + client/src/shared/ui/BearLogo.tsx | 34 + client/src/shared/ui/ChatMessageBubble.tsx | 46 + client/src/shared/ui/CookieConsentBanner.tsx | 72 + client/src/shared/ui/DemoBanner.tsx | 23 + client/src/shared/ui/DemoOverlay.tsx | 56 + client/src/shared/ui/EntityRowActions.tsx | 43 + .../shared/ui/ErrorBoundary/ErrorBoundary.tsx | 56 + .../__tests__/ErrorBoundary.test.tsx | 61 + client/src/shared/ui/ErrorBoundary/index.ts | 1 + .../shared/ui/ModeSwitcher/ModeSwitcher.tsx | 33 + client/src/shared/ui/ModeSwitcher/index.ts | 1 + client/src/shared/ui/NoiseOverlay.tsx | 25 + .../NotificationStack.test.tsx | 74 + .../NotificationStack/NotificationStack.tsx | 106 + .../src/shared/ui/NotificationStack/index.ts | 1 + client/src/shared/ui/OptimizedImage.tsx | 82 + client/src/shared/ui/OrderMessageBody.tsx | 26 + client/src/shared/ui/OrderStatusChip.tsx | 221 + .../shared/ui/RichTextMessageContent.lazy.tsx | 26 + .../src/shared/ui/RichTextMessageContent.tsx | 76 + .../shared/ui/RichTextMessageEditor.lazy.tsx | 28 + .../src/shared/ui/RichTextMessageEditor.tsx | 106 + .../ui/SchemeSwitcher/SchemeSwitcher.tsx | 61 + client/src/shared/ui/SchemeSwitcher/index.ts | 1 + client/src/shared/ui/ScrollOnNavigate.tsx | 12 + client/src/shared/ui/ScrollToTop.tsx | 31 + client/src/shared/ui/SkeletonPage.tsx | 14 + client/src/shared/ui/UserAvatar.tsx | 50 + .../shared/ui/__tests__/DemoOverlay.test.tsx | 44 + .../ui/__tests__/OptimizedImage.test.tsx | 71 + client/src/testing/setup.ts | 1 + client/src/vite-env.d.ts | 23 + client/src/widgets/catalog-slider/index.ts | 1 + .../catalog-slider/ui/CatalogSlider.tsx | 187 + client/src/widgets/navigation-drawer/index.ts | 1 + .../navigation-drawer/ui/NavigationDrawer.tsx | 102 + client/src/widgets/reviews-block/index.ts | 1 + .../widgets/reviews-block/ui/ReviewsBlock.tsx | 160 + client/tsconfig.app.json | 33 + client/tsconfig.json | 4 + client/tsconfig.node.json | 24 + client/vite.config.ts | 56 + client/vitest.config.ts | 13 + 271 files changed, 28009 insertions(+) create mode 100644 .gitignore create mode 100644 AGENTS.md create mode 100644 README.md create mode 100755 client/.editorconfig create mode 100755 client/.gitignore create mode 100755 client/.prettierignore create mode 100755 client/.prettierrc.json create mode 100755 client/README.md create mode 100755 client/eslint.config.js create mode 100755 client/index.html create mode 100755 client/package-lock.json create mode 100755 client/package.json create mode 100755 client/public/apple-touch-icon.png create mode 100755 client/public/favicon-128.png create mode 100755 client/public/favicon-32.png create mode 100755 client/public/favicon-48.png create mode 100755 client/public/favicon.svg create mode 100755 client/public/fonts/Outfit-Bold.woff2 create mode 100755 client/public/fonts/Outfit-Medium.woff2 create mode 100755 client/public/fonts/Outfit-Regular.woff2 create mode 100755 client/public/fonts/Outfit-SemiBold.woff2 create mode 100755 client/public/icons.svg create mode 100755 client/public/logo.webp create mode 100755 client/public/politik.txt create mode 100755 client/public/robots.txt create mode 100755 client/public/sitemap.xml create mode 100755 client/src/app/App.tsx create mode 100755 client/src/app/layout/AppHeader.tsx create mode 100755 client/src/app/layout/MainLayout.tsx create mode 100755 client/src/app/layout/__tests__/MainLayout.test.tsx create mode 100755 client/src/app/providers/AppProviders.tsx create mode 100755 client/src/app/providers/SseProvider.tsx create mode 100755 client/src/app/providers/__tests__/SseProvider.test.tsx create mode 100755 client/src/app/providers/theme-controller.tsx create mode 100755 client/src/app/routes/index.tsx create mode 100755 client/src/app/styles/global.css create mode 100755 client/src/assets/hero.png create mode 100755 client/src/assets/react.svg create mode 100755 client/src/assets/vite.svg create mode 100755 client/src/entities/cart/api/cart-api.ts create mode 100755 client/src/entities/cart/index.ts create mode 100755 client/src/entities/cart/lib/use-cart-query.ts create mode 100755 client/src/entities/cart/model/types.ts create mode 100755 client/src/entities/catalog-slider/api/catalog-slider-api.ts create mode 100755 client/src/entities/catalog-slider/index.ts create mode 100755 client/src/entities/catalog-slider/model/types.ts create mode 100755 client/src/entities/gallery/api/gallery-api.ts create mode 100755 client/src/entities/gallery/index.ts create mode 100755 client/src/entities/gallery/model/types.ts create mode 100755 client/src/entities/gallery/ui/GalleryGrid.tsx create mode 100755 client/src/entities/notification/api/notifications-api.ts create mode 100755 client/src/entities/notification/index.ts create mode 100755 client/src/entities/order/api/admin-order-api.ts create mode 100755 client/src/entities/order/api/order-api.ts create mode 100755 client/src/entities/order/index.ts create mode 100755 client/src/entities/product/api/admin-product-api.ts create mode 100755 client/src/entities/product/api/product-api.ts create mode 100755 client/src/entities/product/index.ts create mode 100755 client/src/entities/product/model/types.ts create mode 100755 client/src/entities/product/ui/ProductCard.tsx create mode 100755 client/src/entities/review/api/admin-review-api.ts create mode 100755 client/src/entities/review/api/reviews-api.ts create mode 100755 client/src/entities/review/index.ts create mode 100755 client/src/entities/test-checklist/api/test-checklist-api.ts create mode 100755 client/src/entities/user/api/address-api.ts create mode 100755 client/src/entities/user/api/messages-api.ts create mode 100755 client/src/entities/user/api/user-api.ts create mode 100755 client/src/entities/user/index.ts create mode 100755 client/src/entities/user/model/types.ts create mode 100755 client/src/features/address-form/index.ts create mode 100755 client/src/features/address-form/ui/AddressFormDialog.tsx create mode 100755 client/src/features/address-map-picker/api/map-geocoding.ts create mode 100755 client/src/features/address-map-picker/index.ts create mode 100755 client/src/features/address-map-picker/model/types.ts create mode 100755 client/src/features/address-map-picker/ui/AddressMapPicker.tsx create mode 100755 client/src/features/address-map-picker/ui/MapPickerMap.tsx create mode 100755 client/src/features/auth-code/index.ts create mode 100755 client/src/features/auth-code/ui/AuthCodeForm.test.tsx create mode 100755 client/src/features/auth-code/ui/AuthCodeForm.tsx create mode 100755 client/src/features/auth-forgot/index.ts create mode 100755 client/src/features/auth-forgot/ui/AuthForgotForm.tsx create mode 100755 client/src/features/auth-oauth/__tests__/OAuthButtons.test.tsx create mode 100755 client/src/features/auth-oauth/index.ts create mode 100755 client/src/features/auth-oauth/lib/oauth-providers.ts create mode 100755 client/src/features/auth-oauth/ui/OAuthButtons.tsx create mode 100755 client/src/features/auth-password/index.ts create mode 100755 client/src/features/auth-password/ui/AuthPasswordForm.test.tsx create mode 100755 client/src/features/auth-password/ui/AuthPasswordForm.tsx create mode 100755 client/src/features/cart/add-to-cart/index.ts create mode 100755 client/src/features/cart/add-to-cart/ui/AddToCartButton.tsx create mode 100755 client/src/features/cart/add-to-cart/ui/__tests__/AddToCartButton.test.tsx create mode 100755 client/src/features/cart/cart-badge/index.ts create mode 100755 client/src/features/cart/cart-badge/ui/CartBadge.tsx create mode 100755 client/src/features/cart/toggle-cart-icon/index.ts create mode 100755 client/src/features/cart/toggle-cart-icon/ui/ToggleCartIcon.tsx create mode 100755 client/src/features/cart/toggle-cart-icon/ui/__tests__/ToggleCartIcon.test.tsx create mode 100755 client/src/features/order-chat/index.ts create mode 100755 client/src/features/order-chat/ui/OrderChat.tsx create mode 100755 client/src/features/order-detail/index.ts create mode 100755 client/src/features/order-detail/ui/DeliveryFeeAdjustmentForm.tsx create mode 100755 client/src/features/order-detail/ui/OrderDetailContent.tsx create mode 100755 client/src/features/order-detail/ui/__tests__/OrderDetailContent.test.tsx create mode 100755 client/src/features/order-payment/index.ts create mode 100755 client/src/features/order-payment/ui/OrderPaymentSection.tsx create mode 100755 client/src/features/product-form/index.ts create mode 100755 client/src/features/product-form/lib/__tests__/use-product-form-helpers.test.ts create mode 100755 client/src/features/product-form/lib/use-product-form-helpers.ts create mode 100755 client/src/features/product-form/model/types.ts create mode 100755 client/src/features/product-form/ui/GalleryImagePicker.tsx create mode 100755 client/src/features/product-form/ui/ProductFormFields.tsx create mode 100755 client/src/features/product-review/index.ts create mode 100755 client/src/features/product-review/ui/ProductReviewsList.tsx create mode 100755 client/src/features/product-review/ui/ReviewDialog.tsx create mode 100755 client/src/features/product-review/ui/ReviewSection.tsx create mode 100755 client/src/features/user/user-menu/index.ts create mode 100755 client/src/features/user/user-menu/ui/UserMenu.tsx create mode 100755 client/src/main.tsx create mode 100755 client/src/pages/about/index.ts create mode 100755 client/src/pages/about/ui/AboutMap.tsx create mode 100755 client/src/pages/about/ui/AboutPage.tsx create mode 100755 client/src/pages/admin-categories/index.ts create mode 100755 client/src/pages/admin-categories/ui/AdminCategoriesPage.tsx create mode 100755 client/src/pages/admin-gallery/index.ts create mode 100755 client/src/pages/admin-gallery/ui/AdminGalleryPage.tsx create mode 100755 client/src/pages/admin-gallery/ui/GallerySliderSection.tsx create mode 100755 client/src/pages/admin-layout/index.ts create mode 100755 client/src/pages/admin-layout/ui/AdminLayoutPage.tsx create mode 100755 client/src/pages/admin-layout/ui/AdminNotificationsPage.tsx create mode 100755 client/src/pages/admin-orders/index.ts create mode 100755 client/src/pages/admin-orders/ui/AdminOrdersPage.tsx create mode 100755 client/src/pages/admin-orders/ui/__tests__/AdminOrdersPage.test.tsx create mode 100755 client/src/pages/admin-products/index.ts create mode 100755 client/src/pages/admin-products/ui/AdminProductsPage.tsx create mode 100755 client/src/pages/admin-products/ui/__tests__/AdminProductsPage.test.tsx create mode 100755 client/src/pages/admin-reviews/index.ts create mode 100755 client/src/pages/admin-reviews/ui/AdminReviewsPage.tsx create mode 100755 client/src/pages/admin-settings/index.ts create mode 100755 client/src/pages/admin-settings/ui/AdminSettingsPage.tsx create mode 100755 client/src/pages/admin-slider/index.ts create mode 100755 client/src/pages/admin-slider/ui/AdminSliderPage.tsx create mode 100755 client/src/pages/admin-test-checklist/index.ts create mode 100755 client/src/pages/admin-test-checklist/ui/AdminTestChecklistPage.tsx create mode 100755 client/src/pages/admin-users/index.ts create mode 100755 client/src/pages/admin-users/ui/AdminUsersPage.tsx create mode 100755 client/src/pages/auth/__tests__/AuthPage.test.tsx create mode 100755 client/src/pages/auth/index.ts create mode 100755 client/src/pages/auth/ui/AuthCallbackPage.tsx create mode 100755 client/src/pages/auth/ui/AuthPage.tsx create mode 100755 client/src/pages/cart/index.ts create mode 100755 client/src/pages/cart/ui/CartPage.tsx create mode 100755 client/src/pages/checkout/index.ts create mode 100755 client/src/pages/checkout/ui/CheckoutPage.tsx create mode 100755 client/src/pages/home/index.ts create mode 100755 client/src/pages/home/lib/use-product-filters.ts create mode 100755 client/src/pages/home/ui/HomePage.tsx create mode 100755 client/src/pages/home/ui/ProductFilters.tsx create mode 100755 client/src/pages/info/index.ts create mode 100755 client/src/pages/info/ui/InfoPage.tsx create mode 100755 client/src/pages/info/ui/sections/DeliverySection.tsx create mode 100755 client/src/pages/info/ui/sections/HowToOrderSection.tsx create mode 100755 client/src/pages/info/ui/sections/OrderStatusesSection.tsx create mode 100755 client/src/pages/info/ui/sections/PaymentSection.tsx create mode 100755 client/src/pages/info/ui/sections/ReturnsSection.tsx create mode 100755 client/src/pages/me/index.ts create mode 100755 client/src/pages/me/ui/MeLayoutPage.tsx create mode 100755 client/src/pages/me/ui/sections/AddressesPage.tsx create mode 100755 client/src/pages/me/ui/sections/AuthMethodsSection.tsx create mode 100755 client/src/pages/me/ui/sections/AvatarSection.tsx create mode 100755 client/src/pages/me/ui/sections/DeleteAccountSection.tsx create mode 100755 client/src/pages/me/ui/sections/MessagesPage.tsx create mode 100755 client/src/pages/me/ui/sections/NotificationsPage.tsx create mode 100755 client/src/pages/me/ui/sections/OrderDetailPage.tsx create mode 100755 client/src/pages/me/ui/sections/OrdersPage.tsx create mode 100755 client/src/pages/me/ui/sections/ProfileSection.tsx create mode 100755 client/src/pages/me/ui/sections/SettingsPage.tsx create mode 100755 client/src/pages/me/ui/sections/__tests__/AuthMethodsSection.test.tsx create mode 100755 client/src/pages/me/ui/sections/__tests__/AvatarSection.test.tsx create mode 100755 client/src/pages/me/ui/sections/__tests__/ProfileSection.test.tsx create mode 100755 client/src/pages/not-found/index.ts create mode 100755 client/src/pages/not-found/ui/NotFoundPage.tsx create mode 100755 client/src/pages/privacy-policy/content/privacy-policy.template.txt create mode 100755 client/src/pages/privacy-policy/index.ts create mode 100755 client/src/pages/privacy-policy/ui/PrivacyPolicyPage.tsx create mode 100755 client/src/pages/product/index.ts create mode 100755 client/src/pages/product/ui/ProductPage.tsx create mode 100755 client/src/pages/terms/index.ts create mode 100755 client/src/pages/terms/ui/TermsPage.tsx create mode 100755 client/src/shared/api/client.ts create mode 100755 client/src/shared/assets/vk-logo.svg create mode 100755 client/src/shared/config/index.ts create mode 100755 client/src/shared/constants/__tests__/order.test.ts create mode 100755 client/src/shared/constants/delivery-carrier.ts create mode 100755 client/src/shared/constants/order.ts create mode 100755 client/src/shared/constants/pickup-point.ts create mode 100755 client/src/shared/constants/upload-limits.ts create mode 100755 client/src/shared/lib/__tests__/format-price.test.ts create mode 100755 client/src/shared/lib/__tests__/get-api-error-message.test.ts create mode 100755 client/src/shared/lib/__tests__/get-error-message.test.ts create mode 100755 client/src/shared/lib/__tests__/group-orders-by-status.test.ts create mode 100755 client/src/shared/lib/__tests__/is-synthetic-email.test.ts create mode 100755 client/src/shared/lib/__tests__/order-requires-price-approval.test.ts create mode 100755 client/src/shared/lib/__tests__/use-mutation-with-toast.test.tsx create mode 100755 client/src/shared/lib/avatar-styles.ts create mode 100755 client/src/shared/lib/create-error-store.ts create mode 100755 client/src/shared/lib/format-price.ts create mode 100755 client/src/shared/lib/get-api-error-message.ts create mode 100755 client/src/shared/lib/get-error-message.ts create mode 100755 client/src/shared/lib/get-original-webp-url.ts create mode 100755 client/src/shared/lib/group-orders-by-status.ts create mode 100755 client/src/shared/lib/invalidate-query-keys.ts create mode 100755 client/src/shared/lib/is-synthetic-email.ts create mode 100755 client/src/shared/lib/oauth-authorize-url.ts create mode 100755 client/src/shared/lib/order-address-snapshot.ts create mode 100755 client/src/shared/lib/order-requires-price-approval.ts create mode 100755 client/src/shared/lib/order-status-data.ts create mode 100755 client/src/shared/lib/persist-token.ts create mode 100755 client/src/shared/lib/reviews-count-ru.ts create mode 100755 client/src/shared/lib/sse.ts create mode 100755 client/src/shared/lib/use-edit-dialog-state.ts create mode 100755 client/src/shared/lib/use-mutation-with-toast.ts create mode 100755 client/src/shared/lib/use-page-title.ts create mode 100755 client/src/shared/model/__tests__/notification.test.ts create mode 100755 client/src/shared/model/auth.ts create mode 100755 client/src/shared/model/notification.ts create mode 100755 client/src/shared/model/theme.ts create mode 100755 client/src/shared/ui/AdminDialog/AdminDialog.tsx create mode 100755 client/src/shared/ui/AdminTable/AdminTable.tsx create mode 100755 client/src/shared/ui/AdminTable/index.ts create mode 100755 client/src/shared/ui/BearLogo.tsx create mode 100755 client/src/shared/ui/ChatMessageBubble.tsx create mode 100755 client/src/shared/ui/CookieConsentBanner.tsx create mode 100755 client/src/shared/ui/DemoBanner.tsx create mode 100755 client/src/shared/ui/DemoOverlay.tsx create mode 100755 client/src/shared/ui/EntityRowActions.tsx create mode 100755 client/src/shared/ui/ErrorBoundary/ErrorBoundary.tsx create mode 100755 client/src/shared/ui/ErrorBoundary/__tests__/ErrorBoundary.test.tsx create mode 100755 client/src/shared/ui/ErrorBoundary/index.ts create mode 100755 client/src/shared/ui/ModeSwitcher/ModeSwitcher.tsx create mode 100755 client/src/shared/ui/ModeSwitcher/index.ts create mode 100755 client/src/shared/ui/NoiseOverlay.tsx create mode 100755 client/src/shared/ui/NotificationStack/NotificationStack.test.tsx create mode 100755 client/src/shared/ui/NotificationStack/NotificationStack.tsx create mode 100755 client/src/shared/ui/NotificationStack/index.ts create mode 100755 client/src/shared/ui/OptimizedImage.tsx create mode 100755 client/src/shared/ui/OrderMessageBody.tsx create mode 100755 client/src/shared/ui/OrderStatusChip.tsx create mode 100755 client/src/shared/ui/RichTextMessageContent.lazy.tsx create mode 100755 client/src/shared/ui/RichTextMessageContent.tsx create mode 100755 client/src/shared/ui/RichTextMessageEditor.lazy.tsx create mode 100755 client/src/shared/ui/RichTextMessageEditor.tsx create mode 100755 client/src/shared/ui/SchemeSwitcher/SchemeSwitcher.tsx create mode 100755 client/src/shared/ui/SchemeSwitcher/index.ts create mode 100755 client/src/shared/ui/ScrollOnNavigate.tsx create mode 100755 client/src/shared/ui/ScrollToTop.tsx create mode 100755 client/src/shared/ui/SkeletonPage.tsx create mode 100755 client/src/shared/ui/UserAvatar.tsx create mode 100755 client/src/shared/ui/__tests__/DemoOverlay.test.tsx create mode 100755 client/src/shared/ui/__tests__/OptimizedImage.test.tsx create mode 100755 client/src/testing/setup.ts create mode 100755 client/src/vite-env.d.ts create mode 100755 client/src/widgets/catalog-slider/index.ts create mode 100755 client/src/widgets/catalog-slider/ui/CatalogSlider.tsx create mode 100755 client/src/widgets/navigation-drawer/index.ts create mode 100755 client/src/widgets/navigation-drawer/ui/NavigationDrawer.tsx create mode 100755 client/src/widgets/reviews-block/index.ts create mode 100755 client/src/widgets/reviews-block/ui/ReviewsBlock.tsx create mode 100755 client/tsconfig.app.json create mode 100755 client/tsconfig.json create mode 100755 client/tsconfig.node.json create mode 100755 client/vite.config.ts create mode 100755 client/vitest.config.ts diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..054287d --- /dev/null +++ b/.gitignore @@ -0,0 +1,15 @@ +node_modules +dist +dist-ssr +*.log +*.local +.env +.vscode/* +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? diff --git a/AGENTS.md b/AGENTS.md new file mode 100644 index 0000000..e1fd9d9 --- /dev/null +++ b/AGENTS.md @@ -0,0 +1,39 @@ +# AGENTS.md — shop-client + +## Project structure + +FSD architecture: app/pages/widgets/features/entities/shared + +## Developer commands + +| Command | What it does | +|---|---| +| `npm run dev` | Vite dev server on :5173, proxies /api and /uploads to 127.0.0.1:3333 | +| `npm run build` | Runs tsc -b first, then vite build | +| `npm run lint` | ESLint (flat config) | +| `npm run lint:fix` | ESLint with --fix | +| `npm run format` | Prettier write all | +| `npm run format:check` | Prettier check only | +| `npm test` | vitest run | +| `npm run test:watch` | vitest watch mode | + +## Conventions + +- **Language**: Отвечай пользователю на русском. +- **Single quotes**, no semicolons, trailing commas, 120 print width (Prettier + ESLint enforce). +- **FSD import boundaries** enforced by eslint-plugin-boundaries. Lower layers cannot import upper layers. +- **Aliases**: @/ → client/src/, @shared/ → ../../shop-server/shared/ +- **API requests**: Use apiClient (axios wrapper from shared/api/) with @tanstack/react-query. Invalidate queries after mutations. +- **UI**: Prefer MUI components over custom HTML/CSS. +- **no-console**: ESLint error; use console.warn/error/info only. + +## Testing + +- vitest + jsdom + @testing-library/react +- Setup file: src/testing/setup.ts +- Test files live in __tests__/ directories next to the code they test. + +## Important + +- shop-server must be cloned alongside shop-client for @shared/ alias to resolve. +- Vite dev server relies on backend running at 127.0.0.1:3333. Start server first. diff --git a/README.md b/README.md new file mode 100644 index 0000000..01f1f08 --- /dev/null +++ b/README.md @@ -0,0 +1,23 @@ +# shop-client — витрина и админка магазина ручной работы + +React + Vite + TypeScript + MUI. FSD-архитектура, @tanstack/react-query. + +## Запуск + +npm ci +npm run dev + +Откройте http://localhost:5173. API проксируется на http://127.0.0.1:3333. + +Требуется shop-server рядом (../../shop-server/shared/ для @shared/ alias). + +## Команды + +| Command | Description | +|---|---| +| npm run dev | Vite dev server | +| npm run build | tsc + vite build | +| npm run lint | ESLint | +| npm run lint:fix | ESLint --fix | +| npm run format | Prettier | +| npm test | vitest run | diff --git a/client/.editorconfig b/client/.editorconfig new file mode 100755 index 0000000..d775357 --- /dev/null +++ b/client/.editorconfig @@ -0,0 +1,12 @@ +root = true + +[*] +charset = utf-8 +end_of_line = lf +insert_final_newline = true +indent_style = space +indent_size = 2 +trim_trailing_whitespace = true + +[{*.md,*.mdx}] +trim_trailing_whitespace = false diff --git a/client/.gitignore b/client/.gitignore new file mode 100755 index 0000000..a547bf3 --- /dev/null +++ b/client/.gitignore @@ -0,0 +1,24 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* +lerna-debug.log* + +node_modules +dist +dist-ssr +*.local + +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea +.DS_Store +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? diff --git a/client/.prettierignore b/client/.prettierignore new file mode 100755 index 0000000..8559dd5 --- /dev/null +++ b/client/.prettierignore @@ -0,0 +1,6 @@ +node_modules +dist +.vite +coverage +*.min.* +package-lock.json diff --git a/client/.prettierrc.json b/client/.prettierrc.json new file mode 100755 index 0000000..61a24f7 --- /dev/null +++ b/client/.prettierrc.json @@ -0,0 +1,9 @@ +{ + "singleQuote": true, + "semi": false, + "printWidth": 120, + "trailingComma": "all", + "endOfLine": "lf", + "jsxSingleQuote": false, + "arrowParens": "always" +} diff --git a/client/README.md b/client/README.md new file mode 100755 index 0000000..7dbf7eb --- /dev/null +++ b/client/README.md @@ -0,0 +1,73 @@ +# React + TypeScript + Vite + +This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. + +Currently, two official plugins are available: + +- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Oxc](https://oxc.rs) +- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/) + +## React Compiler + +The React Compiler is not enabled on this template because of its impact on dev & build performances. To add it, see [this documentation](https://react.dev/learn/react-compiler/installation). + +## Expanding the ESLint configuration + +If you are developing a production application, we recommend updating the configuration to enable type-aware lint rules: + +```js +export default defineConfig([ + globalIgnores(['dist']), + { + files: ['**/*.{ts,tsx}'], + extends: [ + // Other configs... + + // Remove tseslint.configs.recommended and replace with this + tseslint.configs.recommendedTypeChecked, + // Alternatively, use this for stricter rules + tseslint.configs.strictTypeChecked, + // Optionally, add this for stylistic rules + tseslint.configs.stylisticTypeChecked, + + // Other configs... + ], + languageOptions: { + parserOptions: { + project: ['./tsconfig.node.json', './tsconfig.app.json'], + tsconfigRootDir: import.meta.dirname, + }, + // other options... + }, + }, +]) +``` + +You can also install [eslint-plugin-react-x](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-x) and [eslint-plugin-react-dom](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-dom) for React-specific lint rules: + +```js +// eslint.config.js +import reactX from 'eslint-plugin-react-x' +import reactDom from 'eslint-plugin-react-dom' + +export default defineConfig([ + globalIgnores(['dist']), + { + files: ['**/*.{ts,tsx}'], + extends: [ + // Other configs... + // Enable lint rules for React + reactX.configs['recommended-typescript'], + // Enable lint rules for React DOM + reactDom.configs.recommended, + ], + languageOptions: { + parserOptions: { + project: ['./tsconfig.node.json', './tsconfig.app.json'], + tsconfigRootDir: import.meta.dirname, + }, + // other options... + }, + }, +]) +``` diff --git a/client/eslint.config.js b/client/eslint.config.js new file mode 100755 index 0000000..7786ff6 --- /dev/null +++ b/client/eslint.config.js @@ -0,0 +1,204 @@ +import eslint from '@eslint/js' +import tseslint from 'typescript-eslint' +import importX from 'eslint-plugin-import-x' +import react from 'eslint-plugin-react' +import reactHooks from 'eslint-plugin-react-hooks' +import jsxA11y from 'eslint-plugin-jsx-a11y' +import eslintConfigPrettier from 'eslint-config-prettier' +import eslintPluginPrettier from 'eslint-plugin-prettier' +import globals from 'globals' +import boundaries from 'eslint-plugin-boundaries' +import reactRefresh from 'eslint-plugin-react-refresh' + +const fsdPathGroups = [ + { pattern: 'app/**', group: 'internal', position: 'before' }, + { pattern: 'pages/**', group: 'internal', position: 'before' }, + { pattern: 'widgets/**', group: 'internal', position: 'before' }, + { pattern: 'features/**', group: 'internal', position: 'before' }, + { pattern: 'entities/**', group: 'internal', position: 'before' }, + { pattern: 'shared/**', group: 'internal', position: 'before' }, + // alias вида "@/shared/..." + { pattern: '@/**', group: 'internal', position: 'before' }, +] + +/** Правила + FSD-границы. */ +export default tseslint.config( + { + ignores: ['dist/**', 'node_modules/**'], + }, + { + name: 'react-plugin-settings', + settings: { react: { version: '19' } }, + }, + eslint.configs.recommended, + ...tseslint.configs.recommended, + importX.flatConfigs.recommended, + importX.flatConfigs.typescript, + react.configs.flat.recommended, + react.configs.flat['jsx-runtime'], + reactHooks.configs.flat.recommended, + reactRefresh.configs.vite, + jsxA11y.flatConfigs.recommended, + { + files: ['**/*.{ts,tsx}'], + languageOptions: { + globals: { ...globals.browser, ...globals.es2021 }, + }, + settings: { + 'import/internal-regex': '^(@/)?(app|pages|widgets|features|entities|shared)(/|$)', + 'import/resolver': { + typescript: { project: './tsconfig.json' }, + node: true, + }, + }, + rules: { + 'no-console': ['error', { allow: ['warn', 'error', 'info'] }], + quotes: ['error', 'single', { avoidEscape: true, allowTemplateLiterals: false }], + 'max-len': [ + 'warn', + { + code: 120, + ignoreStrings: true, + ignoreTrailingComments: true, + ignoreTemplateLiterals: true, + ignoreComments: true, + }, + ], + 'import-x/extensions': [ + 'warn', + { + js: 'never', + jsx: 'never', + ts: 'never', + tsx: 'never', + json: 'always', + svg: 'always', + }, + ], + 'import-x/prefer-default-export': 'off', + 'import-x/no-extraneous-dependencies': 'off', + 'import-x/no-cycle': 'warn', + 'import-x/order': [ + 'warn', + { + groups: ['builtin', 'external', 'internal', 'parent', 'sibling', 'index', 'object', 'type'], + pathGroups: [ + { pattern: 'react', group: 'external', position: 'before' }, + { pattern: 'react-dom', group: 'external', position: 'before' }, + { pattern: '@mui/**', group: 'external', position: 'before' }, + ...fsdPathGroups, + ], + pathGroupsExcludedImportTypes: ['react'], + 'newlines-between': 'never', + alphabetize: { order: 'asc', caseInsensitive: true }, + }, + ], + 'react/prop-types': 'off', + 'react/jsx-curly-brace-presence': ['error', { props: 'never', children: 'ignore' }], + 'react/display-name': 'off', + 'react/no-unescaped-entities': 'warn', + '@typescript-eslint/no-explicit-any': 'warn', + '@typescript-eslint/no-unused-vars': ['error', { args: 'none' }], + 'no-unused-vars': 'off', + '@typescript-eslint/no-shadow': 'off', + 'no-shadow': 'off', + '@typescript-eslint/no-use-before-define': 'error', + 'no-use-before-define': 'off', + 'consistent-return': 'off', + '@typescript-eslint/no-empty-function': 'warn', + '@typescript-eslint/no-unnecessary-type-constraint': 'warn', + 'class-methods-use-this': 'warn', + }, + }, + { + files: ['**/*.{ts,tsx}'], + plugins: { prettier: eslintPluginPrettier }, + rules: { 'prettier/prettier': ['warn', { endOfLine: 'lf' }] }, + }, + eslintConfigPrettier, + { + files: ['**/*.{ts,tsx}'], + plugins: { boundaries }, + languageOptions: { + parserOptions: { + ecmaVersion: 'latest', + sourceType: 'module', + ecmaFeatures: { jsx: true }, + }, + }, + settings: { + 'import/resolver': { + typescript: { project: './tsconfig.json' }, + }, + 'boundaries/include': ['src/**/*'], + 'boundaries/elements': [ + { type: 'app', pattern: 'src/app/**' }, + { type: 'pages', pattern: 'src/pages/**' }, + { type: 'widgets', pattern: 'src/widgets/**' }, + { type: 'features', pattern: 'src/features/**' }, + { type: 'entities', pattern: 'src/entities/**' }, + { type: 'shared', pattern: 'src/shared/**' }, + ], + }, + rules: { + 'boundaries/no-unknown': 'off', + 'boundaries/no-unknown-files': 'off', + 'boundaries/dependencies': [ + 'error', + { + default: 'disallow', + checkUnknownLocals: true, + rules: [ + { from: { type: 'shared' }, allow: { to: { type: 'shared' } } }, + { + from: { type: 'entities' }, + allow: { to: { type: ['entities', 'shared'] } }, + }, + { + from: { type: 'features' }, + allow: { to: { type: ['features', 'entities', 'shared'] } }, + }, + { + from: { type: 'widgets' }, + allow: { + to: { type: ['widgets', 'features', 'entities', 'shared'] }, + }, + }, + { + from: { type: 'pages' }, + allow: { + to: { + type: ['pages', 'widgets', 'features', 'entities', 'shared'], + }, + }, + }, + { + from: { type: 'app' }, + allow: { + to: { + type: ['app', 'pages', 'widgets', 'features', 'entities', 'shared'], + }, + }, + }, + ], + }, + ], + }, + }, + { + files: ['src/app/providers/theme-controller.tsx'], + rules: { 'react-refresh/only-export-components': 'off' }, + }, + { + files: ['src/pages/**/ui/**/*.tsx'], + rules: { 'react-hooks/incompatible-library': 'off' }, + }, + { + files: ['eslint.config.js'], + rules: { + 'import-x/no-unresolved': 'off', + 'import-x/no-named-as-default': 'off', + 'import-x/no-named-as-default-member': 'off', + }, + }, +) diff --git a/client/index.html b/client/index.html new file mode 100755 index 0000000..d462d14 --- /dev/null +++ b/client/index.html @@ -0,0 +1,30 @@ + + + + + + + + + + + + + Любимый Креатив — Изделия ручной работы + + + + + + + + + + +
+ + + diff --git a/client/package-lock.json b/client/package-lock.json new file mode 100755 index 0000000..240aba7 --- /dev/null +++ b/client/package-lock.json @@ -0,0 +1,10680 @@ +{ + "name": "client", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "client", + "version": "0.0.0", + "dependencies": { + "@dicebear/adventurer": "^9.4.2", + "@dicebear/avataaars": "^9.4.2", + "@dicebear/big-ears": "^9.4.2", + "@dicebear/big-smile": "^9.4.2", + "@dicebear/bottts": "^9.4.2", + "@dicebear/core": "^9.4.2", + "@dicebear/croodles": "^9.4.2", + "@dicebear/fun-emoji": "^9.4.2", + "@dicebear/identicon": "^9.4.2", + "@dicebear/initials": "^9.4.2", + "@dicebear/lorelei": "^9.4.2", + "@dicebear/micah": "^9.4.2", + "@dicebear/notionists": "^9.4.2", + "@dicebear/pixel-art": "^9.4.2", + "@dicebear/rings": "^9.4.2", + "@dicebear/shapes": "^9.4.2", + "@dicebear/thumbs": "^9.4.2", + "@emotion/react": "^11.14.0", + "@emotion/styled": "^11.14.1", + "@mui/icons-material": "^9.0.0", + "@mui/material": "^9.0.0", + "@tanstack/react-query": "^5.100.5", + "@tiptap/extension-placeholder": "^3.22.5", + "@tiptap/react": "^3.22.5", + "@tiptap/starter-kit": "^3.22.5", + "axios": "^1.15.2", + "effector": "^23.4.4", + "effector-react": "^23.3.0", + "lucide-react": "^1.14.0", + "maplibre-gl": "^5.24.0", + "react": "^19.2.5", + "react-dom": "^19.2.5", + "react-hook-form": "^7.74.0", + "react-map-gl": "^8.1.1", + "react-router-dom": "^7.14.2", + "swiper": "^12.1.3" + }, + "devDependencies": { + "@eslint/js": "^9.39.4", + "@testing-library/jest-dom": "^6.9.1", + "@testing-library/react": "^16.3.2", + "@testing-library/user-event": "^14.6.1", + "@types/node": "^24.12.2", + "@types/react": "^19.2.14", + "@types/react-dom": "^19.2.3", + "@vitejs/plugin-react": "^6.0.1", + "eslint": "^9.39.4", + "eslint-config-prettier": "^10.1.8", + "eslint-import-resolver-typescript": "^4.4.4", + "eslint-plugin-boundaries": "^6.0.2", + "eslint-plugin-import-x": "^4.16.2", + "eslint-plugin-jsx-a11y": "^6.10.2", + "eslint-plugin-prettier": "^5.5.5", + "eslint-plugin-react": "^7.37.5", + "eslint-plugin-react-hooks": "^7.1.1", + "eslint-plugin-react-refresh": "^0.5.2", + "globals": "^17.5.0", + "jsdom": "^26.1.0", + "prettier": "^3.8.3", + "typescript": "~6.0.2", + "typescript-eslint": "^8.58.2", + "vite": "^8.0.10", + "vitest": "^3.2.4" + } + }, + "node_modules/@adobe/css-tools": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/@adobe/css-tools/-/css-tools-4.5.0.tgz", + "integrity": "sha512-6OzddxPio9UiWTCemp4N8cYLV2ZN1ncRnV1cVGtve7dhPOtRkleRyx32GQCYSwDYgaHU3USMm84tNsvKzRCa1Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/@asamuzakjp/css-color": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@asamuzakjp/css-color/-/css-color-3.2.0.tgz", + "integrity": "sha512-K1A6z8tS3XsmCMM86xoWdn7Fkdn9m6RSVtocUrJYIwZnFVkng/PvkEoWtOWmP+Scc6saYWHWZYbndEEXxl24jw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@csstools/css-calc": "^2.1.3", + "@csstools/css-color-parser": "^3.0.9", + "@csstools/css-parser-algorithms": "^3.0.4", + "@csstools/css-tokenizer": "^3.0.3", + "lru-cache": "^10.4.3" + } + }, + "node_modules/@asamuzakjp/css-color/node_modules/lru-cache": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", + "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/@babel/code-frame": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.7.tgz", + "integrity": "sha512-Aup7aUOfpbAUg2ROOJN6Iw5f9DMBlzu0mIkm/malLQFN/YQgO48wCj0Kxa3sEHJvPVFg7siR+qRInwXd2qhQKw==", + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.29.7", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.7.tgz", + "integrity": "sha512-locTkQyKvwIEgBzVrn8693ebc97F2U8ZHjbXwDXJ5Fn2TCpNwTlKcaKLkdHop5c/icOFE7qt7Q9JC5hnKNa6Gg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.7.tgz", + "integrity": "sha512-RgHBCvtjbOK2gXSNBNIkNoEc9qoVEtau3hj8gEqKQuL3HZAibKarWFEI3Lfm6EYKkLalOh8eSrj9b+ch9H/VBA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.29.7", + "@babel/generator": "^7.29.7", + "@babel/helper-compilation-targets": "^7.29.7", + "@babel/helper-module-transforms": "^7.29.7", + "@babel/helpers": "^7.29.7", + "@babel/parser": "^7.29.7", + "@babel/template": "^7.29.7", + "@babel/traverse": "^7.29.7", + "@babel/types": "^7.29.7", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/core/node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@babel/core/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/generator": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.7.tgz", + "integrity": "sha512-DkXD5OJQaAQIdZ1bt3UZdEnHAn9Imd3IVBdX03UFe+ony9Ojw5pzr9YVKGDY1jt+Gcn/FnGkNf8r+Vj5NOJWtQ==", + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.7", + "@babel/types": "^7.29.7", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.29.7.tgz", + "integrity": "sha512-wem6WaBj4NaVYVdNhLPPVacES6ZJ+KBBfSkTMD3YZxbP3rm3Di85tJU5ljaUNhaOynt+Aj0xruhYuzQBt8n71g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.29.7", + "@babel/helper-validator-option": "^7.29.7", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.29.7.tgz", + "integrity": "sha512-3nQVUAtvkKH9zahfWgw96Jc/uFOmjACE1kQz82E2lqWmHBgjzbNlsC22nuQTfahmWeQtTq5nQ/4Nnd2A1wj4zA==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.29.7.tgz", + "integrity": "sha512-ejHwrQQYcm9xnTivShn2IDOlIzInN34AXskvq9QicvCtEzq1Vzclu/tKF8Jq1Cg8JG2GL6/EmjgsCT7lXepE3g==", + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.29.7", + "@babel/types": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.29.7.tgz", + "integrity": "sha512-UPUVSyXbOh627KiCIGQSgwWzGeBKLkaJ9PJEdrngIwMSzxLR4jS4+f1f1jb7VzBbg8nFLaYotvVPFCTqdrmTAg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.29.7", + "@babel/helper-validator-identifier": "^7.29.7", + "@babel/traverse": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.29.7.tgz", + "integrity": "sha512-Pb5ijPrZ89GDH8223L4UP8i6QApWxs04RbPQJTeWDV0/keR2E36MeKnyr6LYmUUvqRRI+Iv87SuF1W6ErINzYw==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.29.7.tgz", + "integrity": "sha512-qehxGkRj55h/ff8EMaJ+cYhyaKlHIxqYDn682wQD7RNp9UujOQsHog2uS0r2vzr4pW+sXf90NeeayjcNaX3fFg==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.29.7.tgz", + "integrity": "sha512-N9ZErrD+yW5geCDtBqnOoxmR8+tNKiGuxKlDpuJxfsqpa2dFcexaziGAE/qoHLiDDreVNMupxGmSoNlyvsA3gw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.7.tgz", + "integrity": "sha512-1k2lAGRMfHTcwuNYcCNUmaUffmQv8KWMfh2iJUUeRlwlwH4FdNG7mfPI10NPfLHJFThE4Tyr4mv7kTNZOiPuBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.29.7", + "@babel/types": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.7.tgz", + "integrity": "sha512-hnORnjP/1P/zFEndoeX+n+t1RwWRJiJpM/jO7FW32Kn9r5+sJB2JWOdYo4L6k78j15eCwY3Gm/7364B1EMwtNg==", + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.7" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.7.tgz", + "integrity": "sha512-Nq8OhGWiZIZGV6hLHoyAKLLcJihP/xFeBMGJoUrxTX2psI8dCifzLhZISFb+VWS3wFMRDmCGw5R+dOySCqPLhw==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/template": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.29.7.tgz", + "integrity": "sha512-puq+Gf35oI24FeN11LkoUQFqv9uwNeWpxXZi/Ji3rRIoKAzKnxRaZ+Gkj0vKS9ZCiTESfng1N9LyOyXvo+m+Gg==", + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.7", + "@babel/parser": "^7.29.7", + "@babel/types": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.7.tgz", + "integrity": "sha512-EhlfNQtZ+NK22w5BM61ciuiq1m58ed33Wr1Xan//ZRTy6hgjnwyCffRYwzsGXdASJSUJ1guZILsErh1eQcl+zw==", + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.7", + "@babel/generator": "^7.29.7", + "@babel/helper-globals": "^7.29.7", + "@babel/parser": "^7.29.7", + "@babel/template": "^7.29.7", + "@babel/types": "^7.29.7", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.7.tgz", + "integrity": "sha512-4zBIxpPzowiZpusoFkyGVwakdRJUyuH5PxQ/PrqghfdFWWasvnCdPfQXHrenDai+gyLARulZjZowCOj6fjT4pA==", + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.29.7", + "@babel/helper-validator-identifier": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@boundaries/elements": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@boundaries/elements/-/elements-2.0.1.tgz", + "integrity": "sha512-sAWO3D8PFP6pBXdxxW93SQi/KQqqhE2AAHo3AgWfdtJXwO6bfK6/wUN81XnOZk0qRC6vHzUEKhjwVD9dtDWvxg==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-import-resolver-node": "0.3.9", + "eslint-module-utils": "2.12.1", + "handlebars": "4.7.9", + "is-core-module": "2.16.1", + "micromatch": "4.0.8" + }, + "engines": { + "node": ">=18.18" + } + }, + "node_modules/@csstools/color-helpers": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/@csstools/color-helpers/-/color-helpers-5.1.0.tgz", + "integrity": "sha512-S11EXWJyy0Mz5SYvRmY8nJYTFFd1LCNV+7cXyAgQtOOuzb4EsgfqDufL+9esx72/eLhsRdGZwaldu/h+E4t4BA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT-0", + "engines": { + "node": ">=18" + } + }, + "node_modules/@csstools/css-calc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@csstools/css-calc/-/css-calc-2.1.4.tgz", + "integrity": "sha512-3N8oaj+0juUw/1H3YwmDDJXCgTB1gKU6Hc/bB502u9zR0q2vd786XJH9QfrKIEgFlZmhZiq6epXl4rHqhzsIgQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^3.0.5", + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/@csstools/css-color-parser": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@csstools/css-color-parser/-/css-color-parser-3.1.0.tgz", + "integrity": "sha512-nbtKwh3a6xNVIp/VRuXV64yTKnb1IjTAEEh3irzS+HkKjAOYLTGNb9pmVNntZ8iVBHcWDA2Dof0QtPgFI1BaTA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "dependencies": { + "@csstools/color-helpers": "^5.1.0", + "@csstools/css-calc": "^2.1.4" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^3.0.5", + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/@csstools/css-parser-algorithms": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@csstools/css-parser-algorithms/-/css-parser-algorithms-3.0.5.tgz", + "integrity": "sha512-DaDeUkXZKjdGhgYaHNJTV9pV7Y9B3b644jCLs9Upc3VeNGg6LWARAT6O+Q+/COo+2gg/bM5rhpMAtf70WqfBdQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "peer": true, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@csstools/css-tokenizer": "^3.0.4" + } + }, + "node_modules/@csstools/css-tokenizer": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@csstools/css-tokenizer/-/css-tokenizer-3.0.4.tgz", + "integrity": "sha512-Vd/9EVDiu6PPJt9yAh6roZP6El1xHrdvIVGjyBsHR0RYwNHgL7FJPyIIW4fANJNG6FtyZfvlRPpFI4ZM/lubvw==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "peer": true, + "engines": { + "node": ">=18" + } + }, + "node_modules/@dicebear/adventurer": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/adventurer/-/adventurer-9.4.2.tgz", + "integrity": "sha512-jqYp834ZmGDA9HBBDQAdgF1O2UTCwHF4vVrktXWa2Dppp1JczPL5HnVOWsjtrLmXNn61Wd6OLmBb2e6rhzp3ig==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/avataaars": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/avataaars/-/avataaars-9.4.2.tgz", + "integrity": "sha512-3x9jKFkOkFSPmpTbt9xvhiU2E1GX7beCSsX0tXRUShj8x6+5Ks9yBRT1VlkySbnXrZ/GglADGg7vJ/D2uIx1Yw==", + "license": "See LICENSE file", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/big-ears": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/big-ears/-/big-ears-9.4.2.tgz", + "integrity": "sha512-mNfz3ppNA7UBq0IO3nXCiV5pFPG7c1DfzRB0foNU2Wo1XXT8FIcSY2BvDlYqorZTOUOz7dHb0vx06hqvG0HP5w==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/big-smile": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/big-smile/-/big-smile-9.4.2.tgz", + "integrity": "sha512-hmT5i7rcPPhStjZyg28pbIhdTnnMBzK3RObI0vKCpY30EFrzaPkkdDL6Ck5fAFBdvDIW1EpOJkenyR0XPmhgbQ==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/bottts": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/bottts/-/bottts-9.4.2.tgz", + "integrity": "sha512-tsx+dII7EFUCVA8URj66G1GqORCCVduCAx4dY2prEY2IeFianVpkntXuFsWZ9BBGx1NZFndvDith5oTwKMQPbQ==", + "license": "See LICENSE file", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/core": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/core/-/core-9.4.2.tgz", + "integrity": "sha512-MF0042+Z3s8PGZKZLySfhft28bUa3B1iq0e5NSjCvY8gfMi5aIH/iRJGRJa1N9Jz1BNkxYb4yvJ/N9KO8Z6Y+w==", + "license": "MIT", + "peer": true, + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@dicebear/croodles": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/croodles/-/croodles-9.4.2.tgz", + "integrity": "sha512-6VoO0JviIf7dKKMBTL/SMXxWhnXHaZuzufX90G0nXxS77ELG1YkGNMaZzawizN4C09Gbya2gJkozqrWiJN/aGw==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/fun-emoji": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/fun-emoji/-/fun-emoji-9.4.2.tgz", + "integrity": "sha512-kqB6LPkdYCdEU/mwbyz34xLzoNUKL6ARcoo3fr5ASq9D6ZE07qIKybC3xv5+CPz7VmspJ1Q3c/VVWVMDRP7Twg==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/identicon": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/identicon/-/identicon-9.4.2.tgz", + "integrity": "sha512-JVDSmZsv11mSWqwAktK5x9Bslht2xY3TFUn8xzu6slAYe1Z7hEXZ76eb+UJ6F4qEzdwZ7xPWzAS6Nb0Y3A0pww==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/initials": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/initials/-/initials-9.4.2.tgz", + "integrity": "sha512-yePuIUasmwtl9IrtB6rEzE/zb5fImKP/neW0CdcTC2MwLgMuP1GLHEGRgg1zI8exIh+PMv1YdLGyyUuRTE2Qpw==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/lorelei": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/lorelei/-/lorelei-9.4.2.tgz", + "integrity": "sha512-YMv6vnriW6VLFDsreKuOnUFFno6SRe7+7X7R7zPY0rZ+MaHX9V3jcioIG+1PSjIHEDfOLUHpr5vd1JBWv8y7UA==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/micah": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/micah/-/micah-9.4.2.tgz", + "integrity": "sha512-e4D3W/OlChSsLo7Llwsy0J18vk0azJqF/uFoY+EKACCNHBc1HGNsqVvu2CTf+OWOA8wTyAK6UkjBN5p01r7D+g==", + "license": "(MIT AND CC-BY-4.0)", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/notionists": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/notionists/-/notionists-9.4.2.tgz", + "integrity": "sha512-ZCySq+nxcD/x4xyYgytcj2N9uY3gxrL+qpnmOdp2BdA221KacVrxlsUPpIgEMqxS2rMmBQXfxg129Pzn4ycIpA==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/pixel-art": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/pixel-art/-/pixel-art-9.4.2.tgz", + "integrity": "sha512-peHf7oKICDgBZ8dUyj+txPnS7VZEWgvKE+xW4mNQqBt6dYZIjmva2shOVHn0b1JU+FDxMx3uIkWVixKdUq4WGg==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/rings": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/rings/-/rings-9.4.2.tgz", + "integrity": "sha512-Pc3ymWrRDQPJFNrbbLt7RJrzGvUuuxUiDkrfLhoVE+B6mZWEL1PC78DPbS1yUWYLErJOpJuM2GSwXmTbVjWf+g==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/shapes": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/shapes/-/shapes-9.4.2.tgz", + "integrity": "sha512-AFL6jAaiLztvcqyq+ds+lWZu6Vbp3PlGWhJeJRm842jxtiluJpl6r4f6nUXP2fdMz7MNpDzXfLooQK9E04NbUQ==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@dicebear/thumbs": { + "version": "9.4.2", + "resolved": "https://registry.npmjs.org/@dicebear/thumbs/-/thumbs-9.4.2.tgz", + "integrity": "sha512-ccWvDBqbkWS5uzHbsg5L6uML6vBfX7jT3J3jHCQksvz8haHItxTK02w+6e1UavZUsvza4lG5X/XY3eji3siJ4Q==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@dicebear/core": "^9.0.0" + } + }, + "node_modules/@emnapi/wasi-threads": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.2.tgz", + "integrity": "sha512-c95qOXkHdydNKhscBTebqEC1CVAZpyqOfVfBzQ1qgzyl3gfeldUjIggDbIZgDKsHLgnsM+igH7TJ/eAasaVuMA==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@emotion/babel-plugin": { + "version": "11.13.5", + "resolved": "https://registry.npmjs.org/@emotion/babel-plugin/-/babel-plugin-11.13.5.tgz", + "integrity": "sha512-pxHCpT2ex+0q+HH91/zsdHkw/lXd468DIN2zvfvLtPKLLMo6gQj7oLObq8PhkrxOZb/gGCq03S3Z7PDhS8pduQ==", + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.16.7", + "@babel/runtime": "^7.18.3", + "@emotion/hash": "^0.9.2", + "@emotion/memoize": "^0.9.0", + "@emotion/serialize": "^1.3.3", + "babel-plugin-macros": "^3.1.0", + "convert-source-map": "^1.5.0", + "escape-string-regexp": "^4.0.0", + "find-root": "^1.1.0", + "source-map": "^0.5.7", + "stylis": "4.2.0" + } + }, + "node_modules/@emotion/cache": { + "version": "11.14.0", + "resolved": "https://registry.npmjs.org/@emotion/cache/-/cache-11.14.0.tgz", + "integrity": "sha512-L/B1lc/TViYk4DcpGxtAVbx0ZyiKM5ktoIyafGkH6zg/tj+mA+NE//aPYKG0k8kCHSHVJrpLpcAlOBEXQ3SavA==", + "license": "MIT", + "dependencies": { + "@emotion/memoize": "^0.9.0", + "@emotion/sheet": "^1.4.0", + "@emotion/utils": "^1.4.2", + "@emotion/weak-memoize": "^0.4.0", + "stylis": "4.2.0" + } + }, + "node_modules/@emotion/hash": { + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/@emotion/hash/-/hash-0.9.2.tgz", + "integrity": "sha512-MyqliTZGuOm3+5ZRSaaBGP3USLw6+EGykkwZns2EPC5g8jJ4z9OrdZY9apkl3+UP9+sdz76YYkwCKP5gh8iY3g==", + "license": "MIT" + }, + "node_modules/@emotion/is-prop-valid": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@emotion/is-prop-valid/-/is-prop-valid-1.4.0.tgz", + "integrity": "sha512-QgD4fyscGcbbKwJmqNvUMSE02OsHUa+lAWKdEUIJKgqe5IwRSKd7+KhibEWdaKwgjLj0DRSHA9biAIqGBk05lw==", + "license": "MIT", + "dependencies": { + "@emotion/memoize": "^0.9.0" + } + }, + "node_modules/@emotion/memoize": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/@emotion/memoize/-/memoize-0.9.0.tgz", + "integrity": "sha512-30FAj7/EoJ5mwVPOWhAyCX+FPfMDrVecJAM+Iw9NRoSl4BBAQeqj4cApHHUXOVvIPgLVDsCFoz/hGD+5QQD1GQ==", + "license": "MIT" + }, + "node_modules/@emotion/react": { + "version": "11.14.0", + "resolved": "https://registry.npmjs.org/@emotion/react/-/react-11.14.0.tgz", + "integrity": "sha512-O000MLDBDdk/EohJPFUqvnp4qnHeYkVP5B0xEG0D/L7cOKP9kefu2DXn8dj74cQfsEzUqh+sr1RzFqiL1o+PpA==", + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/runtime": "^7.18.3", + "@emotion/babel-plugin": "^11.13.5", + "@emotion/cache": "^11.14.0", + "@emotion/serialize": "^1.3.3", + "@emotion/use-insertion-effect-with-fallbacks": "^1.2.0", + "@emotion/utils": "^1.4.2", + "@emotion/weak-memoize": "^0.4.0", + "hoist-non-react-statics": "^3.3.1" + }, + "peerDependencies": { + "react": ">=16.8.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@emotion/serialize": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/@emotion/serialize/-/serialize-1.3.3.tgz", + "integrity": "sha512-EISGqt7sSNWHGI76hC7x1CksiXPahbxEOrC5RjmFRJTqLyEK9/9hZvBbiYn70dw4wuwMKiEMCUlR6ZXTSWQqxA==", + "license": "MIT", + "dependencies": { + "@emotion/hash": "^0.9.2", + "@emotion/memoize": "^0.9.0", + "@emotion/unitless": "^0.10.0", + "@emotion/utils": "^1.4.2", + "csstype": "^3.0.2" + } + }, + "node_modules/@emotion/sheet": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@emotion/sheet/-/sheet-1.4.0.tgz", + "integrity": "sha512-fTBW9/8r2w3dXWYM4HCB1Rdp8NLibOw2+XELH5m5+AkWiL/KqYX6dc0kKYlaYyKjrQ6ds33MCdMPEwgs2z1rqg==", + "license": "MIT" + }, + "node_modules/@emotion/styled": { + "version": "11.14.1", + "resolved": "https://registry.npmjs.org/@emotion/styled/-/styled-11.14.1.tgz", + "integrity": "sha512-qEEJt42DuToa3gurlH4Qqc1kVpNq8wO8cJtDzU46TjlzWjDlsVyevtYCRijVq3SrHsROS+gVQ8Fnea108GnKzw==", + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/runtime": "^7.18.3", + "@emotion/babel-plugin": "^11.13.5", + "@emotion/is-prop-valid": "^1.3.0", + "@emotion/serialize": "^1.3.3", + "@emotion/use-insertion-effect-with-fallbacks": "^1.2.0", + "@emotion/utils": "^1.4.2" + }, + "peerDependencies": { + "@emotion/react": "^11.0.0-rc.0", + "react": ">=16.8.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@emotion/unitless": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@emotion/unitless/-/unitless-0.10.0.tgz", + "integrity": "sha512-dFoMUuQA20zvtVTuxZww6OHoJYgrzfKM1t52mVySDJnMSEa08ruEvdYQbhvyu6soU+NeLVd3yKfTfT0NeV6qGg==", + "license": "MIT" + }, + "node_modules/@emotion/use-insertion-effect-with-fallbacks": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@emotion/use-insertion-effect-with-fallbacks/-/use-insertion-effect-with-fallbacks-1.2.0.tgz", + "integrity": "sha512-yJMtVdH59sxi/aVJBpk9FQq+OR8ll5GT8oWd57UpeaKEVGab41JWaCFA7FRLoMLloOZF/c/wsPoe+bfGmRKgDg==", + "license": "MIT", + "peerDependencies": { + "react": ">=16.8.0" + } + }, + "node_modules/@emotion/utils": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@emotion/utils/-/utils-1.4.2.tgz", + "integrity": "sha512-3vLclRofFziIa3J2wDh9jjbkUz9qk5Vi3IZ/FSTKViB0k+ef0fPV7dYrUIugbgupYDx7v9ud/SjrtEP8Y4xLoA==", + "license": "MIT" + }, + "node_modules/@emotion/weak-memoize": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/@emotion/weak-memoize/-/weak-memoize-0.4.0.tgz", + "integrity": "sha512-snKqtPW01tN0ui7yu9rGv69aJXr/a/Ywvl11sUjNtEcRc+ng/mQriFL0wLXMef74iHa/EkftbDzU9F8iFbH+zg==", + "license": "MIT" + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.27.7.tgz", + "integrity": "sha512-EKX3Qwmhz1eMdEJokhALr0YiD0lhQNwDqkPYyPhiSwKrh7/4KRjQc04sZ8db+5DVVnZ1LmbNDI1uAMPEUBnQPg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.27.7.tgz", + "integrity": "sha512-jbPXvB4Yj2yBV7HUfE2KHe4GJX51QplCN1pGbYjvsyCZbQmies29EoJbkEc+vYuU5o45AfQn37vZlyXy4YJ8RQ==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.27.7.tgz", + "integrity": "sha512-62dPZHpIXzvChfvfLJow3q5dDtiNMkwiRzPylSCfriLvZeq0a1bWChrGx/BbUbPwOrsWKMn8idSllklzBy+dgQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.27.7.tgz", + "integrity": "sha512-x5VpMODneVDb70PYV2VQOmIUUiBtY3D3mPBG8NxVk5CogneYhkR7MmM3yR/uMdITLrC1ml/NV1rj4bMJuy9MCg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.27.7.tgz", + "integrity": "sha512-5lckdqeuBPlKUwvoCXIgI2D9/ABmPq3Rdp7IfL70393YgaASt7tbju3Ac+ePVi3KDH6N2RqePfHnXkaDtY9fkw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.27.7.tgz", + "integrity": "sha512-rYnXrKcXuT7Z+WL5K980jVFdvVKhCHhUwid+dDYQpH+qu+TefcomiMAJpIiC2EM3Rjtq0sO3StMV/+3w3MyyqQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.27.7.tgz", + "integrity": "sha512-B48PqeCsEgOtzME2GbNM2roU29AMTuOIN91dsMO30t+Ydis3z/3Ngoj5hhnsOSSwNzS+6JppqWsuhTp6E82l2w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.27.7.tgz", + "integrity": "sha512-jOBDK5XEjA4m5IJK3bpAQF9/Lelu/Z9ZcdhTRLf4cajlB+8VEhFFRjWgfy3M1O4rO2GQ/b2dLwCUGpiF/eATNQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.27.7.tgz", + "integrity": "sha512-RkT/YXYBTSULo3+af8Ib0ykH8u2MBh57o7q/DAs3lTJlyVQkgQvlrPTnjIzzRPQyavxtPtfg0EopvDyIt0j1rA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.27.7.tgz", + "integrity": "sha512-RZPHBoxXuNnPQO9rvjh5jdkRmVizktkT7TCDkDmQ0W2SwHInKCAV95GRuvdSvA7w4VMwfCjUiPwDi0ZO6Nfe9A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.27.7.tgz", + "integrity": "sha512-GA48aKNkyQDbd3KtkplYWT102C5sn/EZTY4XROkxONgruHPU72l+gW+FfF8tf2cFjeHaRbWpOYa/uRBz/Xq1Pg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.27.7.tgz", + "integrity": "sha512-a4POruNM2oWsD4WKvBSEKGIiWQF8fZOAsycHOt6JBpZ+JN2n2JH9WAv56SOyu9X5IqAjqSIPTaJkqN8F7XOQ5Q==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.27.7.tgz", + "integrity": "sha512-KabT5I6StirGfIz0FMgl1I+R1H73Gp0ofL9A3nG3i/cYFJzKHhouBV5VWK1CSgKvVaG4q1RNpCTR2LuTVB3fIw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.27.7.tgz", + "integrity": "sha512-gRsL4x6wsGHGRqhtI+ifpN/vpOFTQtnbsupUF5R5YTAg+y/lKelYR1hXbnBdzDjGbMYjVJLJTd2OFmMewAgwlQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.27.7.tgz", + "integrity": "sha512-hL25LbxO1QOngGzu2U5xeXtxXcW+/GvMN3ejANqXkxZ/opySAZMrc+9LY/WyjAan41unrR3YrmtTsUpwT66InQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.27.7.tgz", + "integrity": "sha512-2k8go8Ycu1Kb46vEelhu1vqEP+UeRVj2zY1pSuPdgvbd5ykAw82Lrro28vXUrRmzEsUV0NzCf54yARIK8r0fdw==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.27.7.tgz", + "integrity": "sha512-hzznmADPt+OmsYzw1EE33ccA+HPdIqiCRq7cQeL1Jlq2gb1+OyWBkMCrYGBJ+sxVzve2ZJEVeePbLM2iEIZSxA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.27.7.tgz", + "integrity": "sha512-b6pqtrQdigZBwZxAn1UpazEisvwaIDvdbMbmrly7cDTMFnw/+3lVxxCTGOrkPVnsYIosJJXAsILG9XcQS+Yu6w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.27.7.tgz", + "integrity": "sha512-OfatkLojr6U+WN5EDYuoQhtM+1xco+/6FSzJJnuWiUw5eVcicbyK3dq5EeV/QHT1uy6GoDhGbFpprUiHUYggrw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.27.7.tgz", + "integrity": "sha512-AFuojMQTxAz75Fo8idVcqoQWEHIXFRbOc1TrVcFSgCZtQfSdc1RXgB3tjOn/krRHENUB4j00bfGjyl2mJrU37A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.27.7.tgz", + "integrity": "sha512-+A1NJmfM8WNDv5CLVQYJ5PshuRm/4cI6WMZRg1by1GwPIQPCTs1GLEUHwiiQGT5zDdyLiRM/l1G0Pv54gvtKIg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.27.7.tgz", + "integrity": "sha512-+KrvYb/C8zA9CU/g0sR6w2RBw7IGc5J2BPnc3dYc5VJxHCSF1yNMxTV5LQ7GuKteQXZtspjFbiuW5/dOj7H4Yw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.27.7.tgz", + "integrity": "sha512-ikktIhFBzQNt/QDyOL580ti9+5mL/YZeUPKU2ivGtGjdTYoqz6jObj6nOMfhASpS4GU4Q/Clh1QtxWAvcYKamA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.27.7.tgz", + "integrity": "sha512-7yRhbHvPqSpRUV7Q20VuDwbjW5kIMwTHpptuUzV+AA46kiPze5Z7qgt6CLCK3pWFrHeNfDd1VKgyP4O+ng17CA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.27.7.tgz", + "integrity": "sha512-SmwKXe6VHIyZYbBLJrhOoCJRB/Z1tckzmgTLfFYOfpMAx63BJEaL9ExI8x7v0oAO3Zh6D/Oi1gVxEYr5oUCFhw==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.27.7.tgz", + "integrity": "sha512-56hiAJPhwQ1R4i+21FVF7V8kSD5zZTdHcVuRFMW0hn753vVfQN8xlx4uOPT4xoGH0Z/oVATuR82AiqSTDIpaHg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/eslint-utils/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/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/config-array": { + "version": "0.21.2", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.2.tgz", + "integrity": "sha512-nJl2KGTlrf9GjLimgIru+V/mzgSK0ABCDQRvxw5BjURL7WfH5uoWmizbH7QB6MmnMBd8cIC9uceWnezL1VZWWw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.7", + "debug": "^4.3.1", + "minimatch": "^3.1.5" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/config-helpers": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz", + "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/core": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz", + "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.5.tgz", + "integrity": "sha512-4IlJx0X0qftVsN5E+/vGujTRIFtwuLbNsVUe7TO6zYPDR1O6nFwvwhIKEKSrl6dZchmYBITazxKoUYOjdtjlRg==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.14.0", + "debug": "^4.3.2", + "espree": "^10.0.1", + "globals": "^14.0.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.1", + "minimatch": "^3.1.5", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@eslint/js": { + "version": "9.39.4", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.4.tgz", + "integrity": "sha512-nE7DEIchvtiFTwBw4Lfbu59PG+kCofhjsKaCWzxTpt4lfRjRMqG6uMBzKXuEcyXhOHoUp9riAm7/aWYGhXZ9cw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.7", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz", + "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/plugin-kit": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz", + "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0", + "levn": "^0.4.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@floating-ui/core": { + "version": "1.7.5", + "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz", + "integrity": "sha512-1Ih4WTWyw0+lKyFMcBHGbb5U5FtuHJuujoyyr5zTaWS5EYMeT6Jb2AuDeftsCsEuchO+mM2ij5+q9crhydzLhQ==", + "license": "MIT", + "optional": true, + "dependencies": { + "@floating-ui/utils": "^0.2.11" + } + }, + "node_modules/@floating-ui/utils": { + "version": "0.2.11", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.11.tgz", + "integrity": "sha512-RiB/yIh78pcIxl6lLMG0CgBXAZ2Y0eVHqMPYugu+9U0AeT6YBeiJpf7lbdJNIugFP5SIjwNRgo4DhR1Qxi26Gg==", + "license": "MIT", + "optional": true + }, + "node_modules/@humanfs/core": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz", + "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/types": "^0.15.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.8", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz", + "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.2", + "@humanfs/types": "^0.15.0", + "@humanwhocodes/retry": "^0.4.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/types": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz", + "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.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/retry": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", + "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@mapbox/jsonlint-lines-primitives": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/@mapbox/jsonlint-lines-primitives/-/jsonlint-lines-primitives-2.0.2.tgz", + "integrity": "sha512-rY0o9A5ECsTQRVhv7tL/OyDpGAoUB4tTvLiW1DSzQGq4bvTPhNw1VpSNjDJc5GFZ2XuyOtSWSVN05qOtcD71qQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/@mapbox/point-geometry": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@mapbox/point-geometry/-/point-geometry-1.1.0.tgz", + "integrity": "sha512-YGcBz1cg4ATXDCM/71L9xveh4dynfGmcLDqufR+nQQy3fKwsAZsWd/x4621/6uJaeB9mwOHE6hPeDgXz9uViUQ==", + "license": "ISC" + }, + "node_modules/@mapbox/tiny-sdf": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@mapbox/tiny-sdf/-/tiny-sdf-2.2.0.tgz", + "integrity": "sha512-LVL4wgI9YAum5V+LNVQO6QgFBPw7/MIIY4XJPNsPDMrjEwcE+JfKk1LuIl8GnF197ejVdC9QdPaxrx5gfgdGXg==", + "license": "BSD-2-Clause" + }, + "node_modules/@mapbox/unitbezier": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/@mapbox/unitbezier/-/unitbezier-0.0.1.tgz", + "integrity": "sha512-nMkuDXFv60aBr9soUG5q+GvZYL+2KZHVvsqFCzqnkGEf46U2fvmytHaEVc1/YZbiLn8X+eR3QzX1+dwDO1lxlw==", + "license": "BSD-2-Clause" + }, + "node_modules/@mapbox/vector-tile": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@mapbox/vector-tile/-/vector-tile-2.0.5.tgz", + "integrity": "sha512-pXj8m7KTsqZt+1jsE0xIpGvqTSbblfkuEJL/NJmNePMtEwxO8V3XMDo9WMSfDeqHvCtBI9Lmt4mGcGR10zecmw==", + "license": "BSD-3-Clause", + "dependencies": { + "@mapbox/point-geometry": "~1.1.0", + "@types/geojson": "^7946.0.16", + "pbf": "^4.0.2" + } + }, + "node_modules/@mapbox/whoots-js": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@mapbox/whoots-js/-/whoots-js-3.1.0.tgz", + "integrity": "sha512-Es6WcD0nO5l+2BOQS4uLfNPYQaNDfbot3X1XUoloz+x0mPDS3eeORZJl06HXjwBG1fOGwCRnzK88LMdxKRrd6Q==", + "license": "ISC", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@maplibre/geojson-vt": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@maplibre/geojson-vt/-/geojson-vt-6.1.0.tgz", + "integrity": "sha512-2eIY4gZxeKIVOZVNkAMb+5NgXhgsMQpOveTQAvnp53LYqHGJZDidk7Ew0Tged9PThidpbS+NFTh0g4zivhPDzQ==", + "license": "ISC", + "dependencies": { + "kdbush": "^4.0.2" + } + }, + "node_modules/@maplibre/maplibre-gl-style-spec": { + "version": "24.8.5", + "resolved": "https://registry.npmjs.org/@maplibre/maplibre-gl-style-spec/-/maplibre-gl-style-spec-24.8.5.tgz", + "integrity": "sha512-EzEJmMt6thioRH7GI9LWS7ahXTcAhAPGWCe6oTP2Ps4YnsXOOAfeqx854lZaiDnwURfHmcCKV1mr6oo0i23x6w==", + "license": "ISC", + "dependencies": { + "@mapbox/jsonlint-lines-primitives": "~2.0.2", + "@mapbox/unitbezier": "^0.0.1", + "json-stringify-pretty-compact": "^4.0.0", + "minimist": "^1.2.8", + "quickselect": "^3.0.0", + "tinyqueue": "^3.0.0" + }, + "bin": { + "gl-style-format": "dist/gl-style-format.mjs", + "gl-style-migrate": "dist/gl-style-migrate.mjs", + "gl-style-validate": "dist/gl-style-validate.mjs" + } + }, + "node_modules/@maplibre/mlt": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/@maplibre/mlt/-/mlt-1.1.11.tgz", + "integrity": "sha512-dKvjKdITw9d0y3ndGkSqLUEpWCizMtdq8NB06cHohH/JZ2sJoM7dClR9wzJLUWykjbw9RXDFmhjjNBnNW27mzw==", + "license": "(MIT OR Apache-2.0)", + "dependencies": { + "@mapbox/point-geometry": "^1.1.0" + } + }, + "node_modules/@maplibre/vt-pbf": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/@maplibre/vt-pbf/-/vt-pbf-4.3.0.tgz", + "integrity": "sha512-jIvp8F5hQCcreqOOpEt42TJMUlsrEcpf/kI1T2v85YrQRV6PPXUcEXUg5karKtH6oh47XJZ4kHu56pUkOuqA7w==", + "license": "MIT", + "dependencies": { + "@mapbox/point-geometry": "^1.1.0", + "@mapbox/vector-tile": "^2.0.4", + "@maplibre/geojson-vt": "^5.0.4", + "@types/geojson": "^7946.0.16", + "@types/supercluster": "^7.1.3", + "pbf": "^4.0.1", + "supercluster": "^8.0.1" + } + }, + "node_modules/@maplibre/vt-pbf/node_modules/@maplibre/geojson-vt": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@maplibre/geojson-vt/-/geojson-vt-5.0.4.tgz", + "integrity": "sha512-KGg9sma45S+stfH9vPCJk1J0lSDLWZgCT9Y8u8qWZJyjFlP8MNP1WGTxIMYJZjDvVT3PDn05kN1C95Sut1HpgQ==", + "license": "ISC" + }, + "node_modules/@mui/core-downloads-tracker": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/core-downloads-tracker/-/core-downloads-tracker-9.0.1.tgz", + "integrity": "sha512-GzamIIhZ1bH77dq7eKaeyRgJdkypsxin4jBFq2EMs4lBWRR0LFO1CSVMsoebn/VvjcNrnrOrjy48MkrkQUK2iw==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + } + }, + "node_modules/@mui/icons-material": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/icons-material/-/icons-material-9.0.1.tgz", + "integrity": "sha512-5PRpQjVLTNLyV/2J9J53Yz4R0tVbodG0BQDN2zQI1QBG1OPYM25ar+4N20eyFOfJT6zKglLzsnU70+zdVLaTkw==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@mui/material": "^9.0.1", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/material": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/material/-/material-9.0.1.tgz", + "integrity": "sha512-voyCpeUxcSWLN7KPZuq0pGCIt726T9K6kiVM3XUcywZDAlZSarLHaUxJVQpospbjjOzN53hwyjo8s6KoWl6utw==", + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/runtime": "^7.29.2", + "@mui/core-downloads-tracker": "^9.0.1", + "@mui/system": "^9.0.1", + "@mui/types": "^9.0.0", + "@mui/utils": "^9.0.1", + "@popperjs/core": "^2.11.8", + "@types/react-transition-group": "^4.4.12", + "clsx": "^2.1.1", + "csstype": "^3.2.3", + "prop-types": "^15.8.1", + "react-is": "^19.2.4", + "react-transition-group": "^4.4.5" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.5.0", + "@emotion/styled": "^11.3.0", + "@mui/material-pigment-css": "^9.0.1", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react-dom": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + }, + "@mui/material-pigment-css": { + "optional": true + }, + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/private-theming": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/private-theming/-/private-theming-9.0.1.tgz", + "integrity": "sha512-pSIGq4Yw749KHEwlkYZWVERgHgwJELP6ODtBNUfV8V4oIb5H+h7IQDFXuk/b2oQccODK1enJAtiEzlgLZmq+8g==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2", + "@mui/utils": "^9.0.1", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/styled-engine": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/@mui/styled-engine/-/styled-engine-9.0.0.tgz", + "integrity": "sha512-9RLGdX4Jg0aQPRuvqh/OLzYSPlgd5zyEw5/1HIRfdavSiOd03WtUaGZH9/w1RoTYuRKwpgy0hpIFaMHIqPVIWg==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2", + "@emotion/cache": "^11.14.0", + "@emotion/serialize": "^1.3.3", + "@emotion/sheet": "^1.4.0", + "csstype": "^3.2.3", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.4.1", + "@emotion/styled": "^11.3.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + } + } + }, + "node_modules/@mui/system": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/system/-/system-9.0.1.tgz", + "integrity": "sha512-WvlioaLxk6ewUIOfh0StxUvOPDS1mCfzaulcudsL1brZNXuh0N9FMk7RpH7ImJKjEz412SEy/V/yvqmtxbqxCQ==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2", + "@mui/private-theming": "^9.0.1", + "@mui/styled-engine": "^9.0.0", + "@mui/types": "^9.0.0", + "@mui/utils": "^9.0.1", + "clsx": "^2.1.1", + "csstype": "^3.2.3", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@emotion/react": "^11.5.0", + "@emotion/styled": "^11.3.0", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/react": { + "optional": true + }, + "@emotion/styled": { + "optional": true + }, + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/types": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/@mui/types/-/types-9.0.0.tgz", + "integrity": "sha512-i1cuFCAWN44b3AJWO7mh7tuh1sqbQSeVr/94oG0TX5uXivac8XalgE4/6fQZcmGZigzbQ35IXxj/4jLpRIBYZg==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@mui/utils": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/@mui/utils/-/utils-9.0.1.tgz", + "integrity": "sha512-f3UO3jNN1pYg5zxqXC81Bvv8hx5ACcYc0387382ZI7M5ono1heIwHYLrKsz85myguWdeVKPRZGmDdynWUBjK2g==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.29.2", + "@mui/types": "^9.0.0", + "@types/prop-types": "^15.7.15", + "clsx": "^2.1.1", + "prop-types": "^15.8.1", + "react-is": "^19.2.4" + }, + "engines": { + "node": ">=14.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/mui-org" + }, + "peerDependencies": { + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@napi-rs/wasm-runtime": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz", + "integrity": "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@tybys/wasm-util": "^0.10.1" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Brooooooklyn" + }, + "peerDependencies": { + "@emnapi/core": "^1.7.1", + "@emnapi/runtime": "^1.7.1" + } + }, + "node_modules/@oxc-project/types": { + "version": "0.132.0", + "resolved": "https://registry.npmjs.org/@oxc-project/types/-/types-0.132.0.tgz", + "integrity": "sha512-FESMOxil5Se014ui/Eq8fT5uHJo6nIRwH0PfJrZJXs6Gek3ZVFOrpUv3YIZT20m+extU98Hg1Ym72U58rlsxUQ==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/Boshen" + } + }, + "node_modules/@package-json/types": { + "version": "0.0.12", + "resolved": "https://registry.npmjs.org/@package-json/types/-/types-0.0.12.tgz", + "integrity": "sha512-uu43FGU34B5VM9mCNjXCwLaGHYjXdNincqKLaraaCW+7S2+SmiBg1Nv8bPnmschrIfZmfKNY9f3fC376MRrObw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@pkgr/core": { + "version": "0.2.9", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.2.9.tgz", + "integrity": "sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/pkgr" + } + }, + "node_modules/@popperjs/core": { + "version": "2.11.8", + "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz", + "integrity": "sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/popperjs" + } + }, + "node_modules/@rolldown/binding-android-arm64": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.2.tgz", + "integrity": "sha512-ZS4D1JPGn/MYQN/SYDWftIE/nVsM8j/AFOYEzAoOE2O3NktQOZru+/vYXGbR/qtdLdIfGCP0lcoJiYVzsEz+iQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-darwin-arm64": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.2.tgz", + "integrity": "sha512-vdFA9+C/rekyGce7WqHs/xoT0ioZEWaOFyZLIV1mEeNFaFDUQrPIo8Vs2GvJ6eetb3rzDUtUBgzto3ExpXJB3w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-darwin-x64": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.2.tgz", + "integrity": "sha512-BewSOwTHazv77DTYiAZXSqqKZ4KP/KonFisDMVU7PImxoWfB2aepnPhd2E4SWz3zDzYgDNbs6jBmTdgNnF02GA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-freebsd-x64": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.2.tgz", + "integrity": "sha512-m41o7M0YWtUdqk61Tb+jnKb2rN++iRdIASlExkUoKfIAH30DOHCB8fVLzSUpbWHHU8esmEioY62PxzexE8MBuA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm-gnueabihf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.2.tgz", + "integrity": "sha512-jcojB9H7W/jS29pMKWAK1N+fU99vXodHDTatS3b3y/XSOCiHo0kkA74pL3jJmkoQtYpOCxDvaKs1fo2Ij/1X5w==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm64-gnu": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.2.tgz", + "integrity": "sha512-1jn6qDU5iiOgFgygDzKUuKP0maTi0/f1+sBLgvij/76C77Nm3ts6ufz9Bjg5q5dduxiUIxtq86JIoBvo1xQ4Ig==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm64-musl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.2.tgz", + "integrity": "sha512-QVLO/czFMdoMFSqlX3bcswcJNm/23r+qoa/jgtmFc/qEp6/jXmIkDjF/XIo8dPfGaiwy1xfQn8o77L79GeXFgw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-ppc64-gnu": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.2.tgz", + "integrity": "sha512-hgO5Abm0w5UL6FEa2iFnZqo2KlK7TQ5QhV5x09hujBf7t5KzHQ1VmfPuTpqRy/rNlSxua3eWH374xxiVrP+lcA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-s390x-gnu": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.2.tgz", + "integrity": "sha512-fy8rXxuYEu602abC8MUNaPjYLIFzReOaEIEMKMUa0rFEUxNpVXhs15KSSQ4qlqSaM7B6rcj9rDZgADh/IGDzLQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-x64-gnu": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.2.tgz", + "integrity": "sha512-0+bOkiQ779+r1WpoHOWHqncvyySci0vKph+myNDYb+im6meJAzHQXay6oEgnkHuUGouM1LKTZwqKpBow6Kj7CQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-x64-musl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.2.tgz", + "integrity": "sha512-mjSkrzZK5Qsl0a9d1JgILOiuZOSDTVdKENcSXBoqbzSrspLR/4/IRVDo5wd2GgZjNss/viBFJdeq+j7qH2nypw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-openharmony-arm64": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.2.tgz", + "integrity": "sha512-1v5vHasdfQAZoEHakBV72LIFAC9JjnymsiKxp+GEr/ma3+NJCPSaYK+qavInOovJkgwFrs7GccX2d6IgDA3Z5w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-wasm32-wasi": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.2.tgz", + "integrity": "sha512-mb1VobWn6NheziTk5/WEaR6AKVbrwT5sOi6C7zk3gy/pD1qtJfU1j4PgTo2NJnOtbL9Dl3Aeei8w9jJ7qC2jZQ==", + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "1.10.0", + "@emnapi/runtime": "1.10.0", + "@napi-rs/wasm-runtime": "^1.1.4" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-wasm32-wasi/node_modules/@emnapi/core": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", + "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/wasi-threads": "1.2.1", + "tslib": "^2.4.0" + } + }, + "node_modules/@rolldown/binding-wasm32-wasi/node_modules/@emnapi/runtime": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", + "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@rolldown/binding-wasm32-wasi/node_modules/@emnapi/wasi-threads": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", + "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@rolldown/binding-win32-arm64-msvc": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.2.tgz", + "integrity": "sha512-SqKonF56vA/L2yHwHYcEp2P34URpOZ7d1fS635cTkpDnUtEGdUbhI6NzsPdqeSWvAAeGDrxjWjNmibDIdFf9/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz", + "integrity": "sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.60.4.tgz", + "integrity": "sha512-F5QXMSiFebS9hKZj02XhWLLnRpJ3B3AROP0tWbFBSj+6kCbg5m9j5JoHKd4mmSVy5mS/IMQloYgYxCuJC0fxEQ==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.60.4.tgz", + "integrity": "sha512-GxxTKApUpzRhof7poWvCJHRF51C67u1R7D6DiluBE8wKU1u5GWE8t+v81JvJYtbawoBFX1hLv5Ei4eVjkWokaw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.60.4.tgz", + "integrity": "sha512-tua0TaJxMOB1R0V0RS1jFZ/RpURFDJIOR2A6jWwQeawuFyS4gBW+rntLRaQd0EQ4bd6Vp44Z2rXW+YYDBsj6IA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.60.4.tgz", + "integrity": "sha512-CSKq7MsP+5PFIcydhAiR1K0UhEI1A2jWXVKHPCBZ151yOutENwvnPocgVHkivu2kviURtCEB6zUQw0vs8RrhMg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.60.4.tgz", + "integrity": "sha512-+O8OkVdyvXMtJEciu2wS/pzm1IxntEEQx3z5TAVy4l32G0etZn+RsA48ARRrFm6Ri8fvqPQfgrvNxSjKAbnd3g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.60.4.tgz", + "integrity": "sha512-Iw3oMskH3AfNuhU0MSN7vNbdi4me/NiYo2azqPz/Le16zHSa+3RRmliCMWWQmh4lcndccU40xcJuTYJZxNo/lw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.60.4.tgz", + "integrity": "sha512-EIPRXTVQpHyF8WOo219AD2yEltPehLTcTMz2fn6JsatLYSzQf00hj3rulF+yauOlF9/FtM2WpkT/hJh/KJFGhA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.60.4.tgz", + "integrity": "sha512-J3Yh9PzzF1Ovah2At+lHiGQdsYgArxBbXv/zHfSyaiFQEqvNv7DcW98pCrmdjCZBrqBiKrKKe2V+aaSGWuBe/w==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.60.4.tgz", + "integrity": "sha512-BFDEZMYfUvLn37ONE1yMBojPxnMlTFsdyNoqncT0qFq1mAfllL+ATMMJd8TeuVMiX84s1KbcxcZbXInmcO2mRg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.60.4.tgz", + "integrity": "sha512-pc9EYOSlOgdQ2uPl1o9PF6/kLSgaUosia7gOuS8mB69IxJvlclko1MECXysjs5ryez1/5zjYqx3+xYU0TU6R1A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.60.4.tgz", + "integrity": "sha512-NxnomyxYerDh5n4iLrNa+sH+Z+U4BMEE46V2PgQ/hoB909i8gV1M5wPojWg9fk1jWpO3IQnOs20K4wyZuFLEFQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-musl": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.60.4.tgz", + "integrity": "sha512-nbJnQ8a3z1mtmrwImCYhc6BGpThAyYVRQxw9uKSKG4wR6aAYno9sVjJ0zaZcW9BPJX1GbrDPf+SvdWjgTuDmnw==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.60.4.tgz", + "integrity": "sha512-2EU6acNrQLd8tYvo/LXW535wupT3m6fo7HKo6lr7ktQoItxTyOL1ZCR/GfGCuXl2vR+zmfI6eRXkSemafv+iVg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-musl": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.60.4.tgz", + "integrity": "sha512-WeBtoMuaMxiiIrO2IYP3xs6GMWkJP2C0EoT8beTLkUPmzV1i/UcOSVw1d5r9KBODtHKilG5yFxsGRnBbK3wJ4A==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.60.4.tgz", + "integrity": "sha512-FJHFfqpKUI3A10WrWKiFbBZ7yVbGT4q4B5o1qKFFojqpaYoh9LrQgqWCmmcxQzVSXYtyB5bzkXrYzlHTs21MYA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.60.4.tgz", + "integrity": "sha512-mcEl6CUT5IAUmQf1m9FYSmVqCJlpQ8r8eyftFUHG8i9OhY7BkBXSUdnLH5DOf0wCOjcP9v/QO93zpmF1SptCCw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.60.4.tgz", + "integrity": "sha512-ynt3JxVd2w2buzoKDWIyiV1pJW93xlQic1THVLXilz429oijRpSHivZAgp65KBu+cMcgf1eVVjdnTLvPxgCuoQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.60.4.tgz", + "integrity": "sha512-Boiz5+MsaROEWDf+GGEwF8VMHGhlUoQMtIPjOgA5fv4osupqTVnJteQNKJwUcnUog2G55jYXH7KZFFiJe0TEzQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.60.4.tgz", + "integrity": "sha512-+qfSY27qIrFfI/Hom04KYFw3GKZSGU4lXus51wsb5EuySfFlWRwjkKWoE9emgRw/ukoT4Udsj4W/+xxG8VbPKg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openbsd-x64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.60.4.tgz", + "integrity": "sha512-VpTfOPHgVXEBeeR8hZ2O0F3aSso+JDWqTWmTmzcQKted54IAdUVbxE+j/MVxUsKa8L20HJhv3vUezVPoquqWjA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.60.4.tgz", + "integrity": "sha512-IPOsh5aRYuLv/nkU51X10Bf75Bsf6+gZdx1X+QP5QM6lIJFHHqbHLG0uJn/hWthzo13UAc2umiUorqZy3axoZg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.60.4.tgz", + "integrity": "sha512-4QzE9E81OohJ/HKzHhsqU+zcYYojVOXlFMs1DdyMT6qXl/niOH7AVElmmEdUNHHS/oRkc++d5k6Vy85zFs0DEw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.60.4.tgz", + "integrity": "sha512-zTPgT1YuHHcd+Tmx7h8aml0FWFVelV5N54oHow9SLj+GfoDy/huQ+UV396N/C7KpMDMiPspRktzM1/0r1usYEA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.60.4.tgz", + "integrity": "sha512-DRS4G7mi9lJxqEDezIkKCaUIKCrLUUDCUaCsTPCi/rtqaC6D/jjwslMQyiDU50Ka0JKpeXeRBFBAXwArY52vBw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.60.4.tgz", + "integrity": "sha512-QVTUovf40zgTqlFVrKA1uXMVvU2QWEFWfAH8Wdc48IxLvrJMQVMBRjuQyUpzZCDkakImib9eVazbWlC6ksWtJw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@tanstack/query-core": { + "version": "5.100.14", + "resolved": "https://registry.npmjs.org/@tanstack/query-core/-/query-core-5.100.14.tgz", + "integrity": "sha512-5X41dGpxgeaHISCRW2oYwcSycZeULZzAunaudXT9ov1KOTj9xwt0CH6hbwqP1/z74ZWF7rYFnDpyYH07XFcZew==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/tannerlinsley" + } + }, + "node_modules/@tanstack/react-query": { + "version": "5.100.14", + "resolved": "https://registry.npmjs.org/@tanstack/react-query/-/react-query-5.100.14.tgz", + "integrity": "sha512-oOr6aRdSFEwWhzxEkD/9ZcItM3+LjBSkeVmadWKwUssAHTsqd/7bOjWrX4AbvEkoEhgAxzN0Xk6H/aYzXiYBAw==", + "license": "MIT", + "dependencies": { + "@tanstack/query-core": "5.100.14" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/tannerlinsley" + }, + "peerDependencies": { + "react": "^18 || ^19" + } + }, + "node_modules/@testing-library/dom": { + "version": "10.4.1", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.1.tgz", + "integrity": "sha512-o4PXJQidqJl82ckFaXUeoAW+XysPLauYI43Abki5hABd853iMhitooc6znOnczgbTYmEP6U6/y1ZyKAIsvMKGg==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "picocolors": "1.1.1", + "pretty-format": "^27.0.2" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@testing-library/jest-dom": { + "version": "6.9.1", + "resolved": "https://registry.npmjs.org/@testing-library/jest-dom/-/jest-dom-6.9.1.tgz", + "integrity": "sha512-zIcONa+hVtVSSep9UT3jZ5rizo2BsxgyDYU7WFD5eICBE7no3881HGeb/QkGfsJs6JTkY1aQhT7rIPC7e+0nnA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@adobe/css-tools": "^4.4.0", + "aria-query": "^5.0.0", + "css.escape": "^1.5.1", + "dom-accessibility-api": "^0.6.3", + "picocolors": "^1.1.1", + "redent": "^3.0.0" + }, + "engines": { + "node": ">=14", + "npm": ">=6", + "yarn": ">=1" + } + }, + "node_modules/@testing-library/jest-dom/node_modules/dom-accessibility-api": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.6.3.tgz", + "integrity": "sha512-7ZgogeTnjuHbo+ct10G9Ffp0mif17idi0IyWNVA/wcwcm7NPOD/WEHVP3n7n3MhXqxoIYm8d6MuZohYWIZ4T3w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@testing-library/react": { + "version": "16.3.2", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-16.3.2.tgz", + "integrity": "sha512-XU5/SytQM+ykqMnAnvB2umaJNIOsLF3PVv//1Ew4CTcpz0/BRyy/af40qqrt7SjKpDdT1saBMc42CUok5gaw+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.12.5" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "@testing-library/dom": "^10.0.0", + "@types/react": "^18.0.0 || ^19.0.0", + "@types/react-dom": "^18.0.0 || ^19.0.0", + "react": "^18.0.0 || ^19.0.0", + "react-dom": "^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@testing-library/user-event": { + "version": "14.6.1", + "resolved": "https://registry.npmjs.org/@testing-library/user-event/-/user-event-14.6.1.tgz", + "integrity": "sha512-vq7fv0rnt+QTXgPxr5Hjc210p6YKq2kmdziLgnsZGgLJ9e6VAShx1pACLuRjd/AS/sr7phAR58OIIpf0LlmQNw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12", + "npm": ">=6" + }, + "peerDependencies": { + "@testing-library/dom": ">=7.21.4" + } + }, + "node_modules/@tiptap/core": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/core/-/core-3.23.6.tgz", + "integrity": "sha512-MRB3pHz4Oxqmcawh0cQ5iOGdY5xtNYp/1CoK7hdTLzw5K0C6/gTC2VvanB1R4INaB6EpBkxG/GiWkVirDRnuXw==", + "license": "MIT", + "peer": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-blockquote": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-blockquote/-/extension-blockquote-3.23.6.tgz", + "integrity": "sha512-2RmnqNqTltZ2k1F7IfjoDNs935Uq4rRDR7d98mqkg3OlDktcQIyBpv0t9dTay6H5bkQeZUuS8ogK2S1E8Edjug==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-bold": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-bold/-/extension-bold-3.23.6.tgz", + "integrity": "sha512-1LMhjnytdbbhWHSoOwnLxZAOQZWPkKyXVCNmaIk0Mhi4tLPUXptG4qKS5sVYTCveE5H6IBPFrbgBFi5dMI6krA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-bubble-menu": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-bubble-menu/-/extension-bubble-menu-3.23.6.tgz", + "integrity": "sha512-Mwkyp9LkDHFbqmWRIkp63FinRxFu3ajC4qSb9t4mnHsb4kAdbNLLsGtbFg+le0SWk4CxGwAOwM7SzeJ+6UGqCA==", + "license": "MIT", + "optional": true, + "dependencies": { + "@floating-ui/dom": "^1.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-bullet-list": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-bullet-list/-/extension-bullet-list-3.23.6.tgz", + "integrity": "sha512-RMRgfXZykr/13X8UBOwvpgysVOo9KchwqMoEbvqQSj4YFfU56iIn59C8sbxiQ1sKfeltUf0wH4fPc0I4iwKqAA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extension-list": "3.23.6" + } + }, + "node_modules/@tiptap/extension-code": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-code/-/extension-code-3.23.6.tgz", + "integrity": "sha512-KG8KXFYyLrtYvT7AZ1WGV61ofx8pDe5g9pH658MERxqQGii+Pyfc6xkz04l7XeBts/7+571UQp/0O7i/z560TA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-code-block": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-code-block/-/extension-code-block-3.23.6.tgz", + "integrity": "sha512-4kccgcn5yHThxrzsIhJny3EwfEZYIk+BjUCL4uIuzOyWvExtGhZ6JMHVCZeMhI8D1/bX1LNkkAKN5DXPzH4lXQ==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-document": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-document/-/extension-document-3.23.6.tgz", + "integrity": "sha512-XDAIgG9KcKumFM9KJWUEUhXPbFIhhl47bfy5GknareWTRKke85rcoj/oxKKO9ihLZr8JfpbXjqnS4SCm5yhYPw==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-dropcursor": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-dropcursor/-/extension-dropcursor-3.23.6.tgz", + "integrity": "sha512-+XWEoRKf3lXxi7Le1aOM2xU1XHwxICGpXjT3m4QaYqUgIpsq8gQEuso6kVg8DnTD7biKQs6+oIQ0o2b/gTW9WA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extensions": "3.23.6" + } + }, + "node_modules/@tiptap/extension-floating-menu": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-floating-menu/-/extension-floating-menu-3.23.6.tgz", + "integrity": "sha512-2kjuDcEq69lEcECl75xqY5MyzUSh2zcC5aLrpwP1WwhJz5bxsIFHiaps5AP6h9R4A+ZBj5b2haay2Y1wDUU3VA==", + "license": "MIT", + "optional": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@floating-ui/dom": "^1.0.0", + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-gapcursor": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-gapcursor/-/extension-gapcursor-3.23.6.tgz", + "integrity": "sha512-wbKmxXsszxWacEkrHucRpSQbiKjz4fmOebD6OVyL9AcrmlbxNk8vcM3iyh/8cVeRy09XY+morM165t/u7/z4IQ==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extensions": "3.23.6" + } + }, + "node_modules/@tiptap/extension-hard-break": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-hard-break/-/extension-hard-break-3.23.6.tgz", + "integrity": "sha512-KeUm+tkUfIVSX9QM9XOIhaay0Fn36sLKUo5NVYjN3uJaxFvaZXZmTlxdO85OTdgF2P5sqh9LomrIgliaFRGk4w==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-heading": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-heading/-/extension-heading-3.23.6.tgz", + "integrity": "sha512-A/0jPhxnUh9THSZymlu0OGPZe1wdFdwHAXnRCmqvYUCwJjrG7LCC/ahzmcj1tcNzI9hgHyuYPSfev8RXYrNu/w==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-horizontal-rule": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-horizontal-rule/-/extension-horizontal-rule-3.23.6.tgz", + "integrity": "sha512-hEUlz4H+I64r+TH6LCuNCRgO7JTHncXGmx9+WbU69EOfY8O0ZurcgeJc8HeiAKL+r9YuC1e5YHfFxgCaaC0jlg==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-italic": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-italic/-/extension-italic-3.23.6.tgz", + "integrity": "sha512-wol5KdwCPAvpiYhH9PLlvO8ZnJHwZtIboVevrfOGgBcKlXRA3dedR4OAMXHnUtkkzu9KtliLg1+TYzEx4JZG9Q==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-link": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-link/-/extension-link-3.23.6.tgz", + "integrity": "sha512-KNZz7z7P2/qbQsx5bPAbSPjrKDg1VHsedGlLHJCr8U2VRD5VgmDLkMpkouP1CsDg15qgyUKv/nDib5KgPpLNWA==", + "license": "MIT", + "dependencies": { + "linkifyjs": "^4.3.3" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-list": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-list/-/extension-list-3.23.6.tgz", + "integrity": "sha512-z6vj9+Qht2sjdQkyyHcUpsC/yCIZqTrQiyHDhs/HGKrfvoANyAZGpqdNeKf1wSyjIso+27tQuIH5NDfk8ygyNw==", + "license": "MIT", + "peer": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/extension-list-item": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-list-item/-/extension-list-item-3.23.6.tgz", + "integrity": "sha512-3zzyhdkUWcHVpXuvy6KiIwjh29rbH6gEDEqPQqHLrl1XGnO9pnShC7pSHctlCDjmcx3O4n9cd4QMtVBlUerbiA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extension-list": "3.23.6" + } + }, + "node_modules/@tiptap/extension-list-keymap": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-list-keymap/-/extension-list-keymap-3.23.6.tgz", + "integrity": "sha512-x8bPcLViGzg/RAmQM/XtmfqIwQ/Pv9Q8mkd+OgfUiTqjeJqKwVQmiqbLFNa7zw81+H61M+HDU+qGAaQ3vRIMjw==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extension-list": "3.23.6" + } + }, + "node_modules/@tiptap/extension-ordered-list": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-ordered-list/-/extension-ordered-list-3.23.6.tgz", + "integrity": "sha512-1m/wWB/ZtXcmG2vNdiUkCqsOgqv5vBjCv/mVaHhF9OvV+zQS8YDjoWE7zEuT/GgELdT77Xq8lHrn4nCDudB3/A==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extension-list": "3.23.6" + } + }, + "node_modules/@tiptap/extension-paragraph": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-paragraph/-/extension-paragraph-3.23.6.tgz", + "integrity": "sha512-+7m58LUSncodjrIyXks4RZ3tLNYrvgT77wRR4l3HnM5OABY3GDsDTqi7c1t1yI29NVOSk/DUacqy6UwYAj1DGg==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-placeholder": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-placeholder/-/extension-placeholder-3.23.6.tgz", + "integrity": "sha512-8I6b2aevF74aLgymKMxbDxSLxWA2y+2dh0zZDeI8sRZ2m6WHHes+Kyuuwkq1HIPcR+ZLpbec74cmf6lcL/yvqQ==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/extensions": "3.23.6" + } + }, + "node_modules/@tiptap/extension-strike": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-strike/-/extension-strike-3.23.6.tgz", + "integrity": "sha512-oF7FEZ37f15aCe5kPgzGDYf/m+hr7VdQ/Ko/Hds/UM9pX7AG1fdtmRrl6wqkRqDM/incZaC/AQR2/Dpo2VCNGQ==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-text": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-text/-/extension-text-3.23.6.tgz", + "integrity": "sha512-ipoC2TkIAIOTiF5ByiGgvQB1DqDyfP90wrUB3mohBcgvp7lQnwHszCDGv8dNnmcUek8uXV/uoLu2VXeVQlxjPA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extension-underline": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extension-underline/-/extension-underline-3.23.6.tgz", + "integrity": "sha512-P55wGIZGYTVH92Fq0cgI4/O9AhLCaJC3hhxg15RSERP5/YegM9eJHDK/GQ1EE/DvYA+xpYGOV6agKwAUqfA/Iw==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6" + } + }, + "node_modules/@tiptap/extensions": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/extensions/-/extensions-3.23.6.tgz", + "integrity": "sha512-X09/Db1teB+ifXzDGVVFmOeQRx7wTAayE9/280spxpsHkHZvJ5bHRvWIzUzviMIjbBz+NPDIKYPK7gMfh9iaig==", + "license": "MIT", + "peer": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6" + } + }, + "node_modules/@tiptap/pm": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/pm/-/pm-3.23.6.tgz", + "integrity": "sha512-in5CaMaWlJcH2A1q6GJKFtrodE8WLS3M9tIi/f89jPmIVHJShpodC0KZDNyJkrVBQomYk0DEh86Utm6ASXzQww==", + "license": "MIT", + "peer": true, + "dependencies": { + "prosemirror-changeset": "^2.3.0", + "prosemirror-commands": "^1.6.2", + "prosemirror-dropcursor": "^1.8.1", + "prosemirror-gapcursor": "^1.3.2", + "prosemirror-history": "^1.4.1", + "prosemirror-keymap": "^1.2.2", + "prosemirror-model": "^1.24.1", + "prosemirror-schema-list": "^1.5.0", + "prosemirror-state": "^1.4.3", + "prosemirror-tables": "^1.6.4", + "prosemirror-transform": "^1.10.2", + "prosemirror-view": "^1.38.1" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + } + }, + "node_modules/@tiptap/react": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/react/-/react-3.23.6.tgz", + "integrity": "sha512-Tw9KZkYqFMk3vaJAEQKqEYIO/iq3cSJe7OUEGBul4k4GaMQeLItLf5EYhUd0GIPXci1WVVPNntKJsHfX25M37w==", + "license": "MIT", + "dependencies": { + "@types/use-sync-external-store": "^0.0.6", + "fast-equals": "^5.3.3", + "use-sync-external-store": "^1.4.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + }, + "optionalDependencies": { + "@tiptap/extension-bubble-menu": "^3.23.6", + "@tiptap/extension-floating-menu": "^3.23.6" + }, + "peerDependencies": { + "@tiptap/core": "3.23.6", + "@tiptap/pm": "3.23.6", + "@types/react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "@types/react-dom": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react": "^17.0.0 || ^18.0.0 || ^19.0.0", + "react-dom": "^17.0.0 || ^18.0.0 || ^19.0.0" + } + }, + "node_modules/@tiptap/starter-kit": { + "version": "3.23.6", + "resolved": "https://registry.npmjs.org/@tiptap/starter-kit/-/starter-kit-3.23.6.tgz", + "integrity": "sha512-gykwtGWrnWCmtql1hid3opac/KV8zQvOAnu3bTqIqcHrn1FusbUwKmNzavSbfGvcktHM3hFjb35W48JyVLyu/A==", + "license": "MIT", + "dependencies": { + "@tiptap/core": "^3.23.6", + "@tiptap/extension-blockquote": "^3.23.6", + "@tiptap/extension-bold": "^3.23.6", + "@tiptap/extension-bullet-list": "^3.23.6", + "@tiptap/extension-code": "^3.23.6", + "@tiptap/extension-code-block": "^3.23.6", + "@tiptap/extension-document": "^3.23.6", + "@tiptap/extension-dropcursor": "^3.23.6", + "@tiptap/extension-gapcursor": "^3.23.6", + "@tiptap/extension-hard-break": "^3.23.6", + "@tiptap/extension-heading": "^3.23.6", + "@tiptap/extension-horizontal-rule": "^3.23.6", + "@tiptap/extension-italic": "^3.23.6", + "@tiptap/extension-link": "^3.23.6", + "@tiptap/extension-list": "^3.23.6", + "@tiptap/extension-list-item": "^3.23.6", + "@tiptap/extension-list-keymap": "^3.23.6", + "@tiptap/extension-ordered-list": "^3.23.6", + "@tiptap/extension-paragraph": "^3.23.6", + "@tiptap/extension-strike": "^3.23.6", + "@tiptap/extension-text": "^3.23.6", + "@tiptap/extension-underline": "^3.23.6", + "@tiptap/extensions": "^3.23.6", + "@tiptap/pm": "^3.23.6" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/ueberdosis" + } + }, + "node_modules/@tybys/wasm-util": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz", + "integrity": "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@types/aria-query": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/chai": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", + "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*", + "assertion-error": "^2.0.1" + } + }, + "node_modules/@types/deep-eql": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", + "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/estree": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.9.tgz", + "integrity": "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/geojson": { + "version": "7946.0.16", + "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.16.tgz", + "integrity": "sha512-6C8nqWur3j98U6+lXDfTUWIfgvZU+EumvpHKcYjujKH7woYyLj2sUmff0tRhrqM7BohUw7Pz3ZB1jj2gW9Fvmg==", + "license": "MIT" + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "24.12.4", + "resolved": "https://registry.npmjs.org/@types/node/-/node-24.12.4.tgz", + "integrity": "sha512-GUUEShf+PBCGW2KaXwcIt3Yk+e3pkKwWKb9GSyM9WQVE+ep2jzmHdGsHzu4wgcZy5fN9FBdVzjpBQsYlpfpgLA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "undici-types": "~7.16.0" + } + }, + "node_modules/@types/parse-json": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.2.tgz", + "integrity": "sha512-dISoDXWWQwUquiKsyZ4Ng+HX2KsPL7LyHKHQwgGFEA3IaKac4Obd+h2a/a6waisAoepJlBcx9paWqjA8/HVjCw==", + "license": "MIT" + }, + "node_modules/@types/prop-types": { + "version": "15.7.15", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", + "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==", + "license": "MIT" + }, + "node_modules/@types/react": { + "version": "19.2.15", + "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.15.tgz", + "integrity": "sha512-eRwcGNHve+E8qtEQSSRl6urh+rFop4v8gm6O8rGv25CodbvFdLjA1vVQ1KkiFE0w0UPOnb8tDiFKL5lp0rtY5Q==", + "license": "MIT", + "peer": true, + "dependencies": { + "csstype": "^3.2.2" + } + }, + "node_modules/@types/react-dom": { + "version": "19.2.3", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz", + "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==", + "license": "MIT", + "peer": true, + "peerDependencies": { + "@types/react": "^19.2.0" + } + }, + "node_modules/@types/react-transition-group": { + "version": "4.4.12", + "resolved": "https://registry.npmjs.org/@types/react-transition-group/-/react-transition-group-4.4.12.tgz", + "integrity": "sha512-8TV6R3h2j7a91c+1DXdJi3Syo69zzIZbz7Lg5tORM5LEJG7X/E6a1V3drRyBRZq7/utz7A+c4OgYLiLcYGHG6w==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*" + } + }, + "node_modules/@types/supercluster": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@types/supercluster/-/supercluster-7.1.3.tgz", + "integrity": "sha512-Z0pOY34GDFl3Q6hUFYf3HkTwKEE02e7QgtJppBt+beEAxnyOpJua+voGFvxINBHa06GwLFFym7gRPY2SiKIfIA==", + "license": "MIT", + "dependencies": { + "@types/geojson": "*" + } + }, + "node_modules/@types/use-sync-external-store": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.6.tgz", + "integrity": "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg==", + "license": "MIT" + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.60.0.tgz", + "integrity": "sha512-QYb/sa74/s7OKMbACMjrYnGspj9Hs5YI5aaffSL65UfeBUzVzBJfVo3oWSpbzPurvm7yaCCo2Lk7lVj610HqKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.12.2", + "@typescript-eslint/scope-manager": "8.60.0", + "@typescript-eslint/type-utils": "8.60.0", + "@typescript-eslint/utils": "8.60.0", + "@typescript-eslint/visitor-keys": "8.60.0", + "ignore": "^7.0.5", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.60.0", + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", + "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.60.0.tgz", + "integrity": "sha512-fcqpj/MyK4sxDPcbe7STNPbpQL4RLZOPWuaTmwZYuc+hJKzRf58yRxfhqGpc6PIq9ZyfSBpfHgmUHmHs0KwHwg==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@typescript-eslint/scope-manager": "8.60.0", + "@typescript-eslint/types": "8.60.0", + "@typescript-eslint/typescript-estree": "8.60.0", + "@typescript-eslint/visitor-keys": "8.60.0", + "debug": "^4.4.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/project-service": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.60.0.tgz", + "integrity": "sha512-aZu74NNKJeUWqCjDddzdiKaS82dgYgV/vmf+Ui3ZdZejmgfXR/q+pRumgobnQ2cCJTgGTWp4ypiwsuofFubavg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.60.0", + "@typescript-eslint/types": "^8.60.0", + "debug": "^4.4.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.60.0.tgz", + "integrity": "sha512-pFzqhllJMs+jghLQWzV00ds39xLzuyqPSev5pd8f4Ir0rtKR3ZLUB4/4dhjOFighWb9larvtfJvqL+4yKDI3Xw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.0", + "@typescript-eslint/visitor-keys": "8.60.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.60.0.tgz", + "integrity": "sha512-BZPR3RGYlAXnly6ymAxfkVn5rCbZzQNou0rxv3GfWZ8cTQp+hhVd73khbGLAd8k1TlAPLISH337M+tAgAnaJDQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.60.0.tgz", + "integrity": "sha512-SX46wEUtitCpq7AN38HkUU/+zvUpdKf7ephtWAFgckH8O7PQIyL5gvrhQgBLuEYgLfuKWOVvWVskMbuFHAz5xg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.0", + "@typescript-eslint/typescript-estree": "8.60.0", + "@typescript-eslint/utils": "8.60.0", + "debug": "^4.4.3", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/types": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.60.0.tgz", + "integrity": "sha512-AsE7x2XaAK+CVbeih0Fvbn+r1qHxtpLDJ3XUuFcIinT318T90yHMJC+Zgv+jUuDjQQd06HKwxnDu6sz1IcTilA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.60.0.tgz", + "integrity": "sha512-3AcZNBGMClm6CXDyo8kYvVGT/sx29sS0oBsIb9oZI2gunA4Vm2M3YHzRLPvsUBBsl+yB5FPtltq7gGH0iTlp9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/project-service": "8.60.0", + "@typescript-eslint/tsconfig-utils": "8.60.0", + "@typescript-eslint/types": "8.60.0", + "@typescript-eslint/visitor-keys": "8.60.0", + "debug": "^4.4.3", + "minimatch": "^10.2.2", + "semver": "^7.7.3", + "tinyglobby": "^0.2.15", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/balanced-match": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", + "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^4.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { + "version": "10.2.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", + "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "brace-expansion": "^5.0.5" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.60.0.tgz", + "integrity": "sha512-HtXuPfrHTyBDkameWpl+vJb1Uevu2tznAyahM1Oc4AENidCLTPiZDWIo4GfcxNdC/RcfGcadzzkqbRG87dUrQA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/scope-manager": "8.60.0", + "@typescript-eslint/types": "8.60.0", + "@typescript-eslint/typescript-estree": "8.60.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.60.0.tgz", + "integrity": "sha512-9WI52t8ZGLVGrPMBet25yAftqY/n95+zmoUUtJBBQTKDSKUu7OsPTroT2op7U9JatkoRccL0YkWDNMFfC4Sjxg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.0", + "eslint-visitor-keys": "^5.0.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@unrs/resolver-binding-android-arm-eabi": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm-eabi/-/resolver-binding-android-arm-eabi-1.12.2.tgz", + "integrity": "sha512-g5T90pqg1bo/7mytQx6F4iBNC0Wsh9cu+z9veDbFjc7HjpesJFWD7QMS0NGStXM075+7dJPPVvBbpZlnrdpi/w==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@unrs/resolver-binding-android-arm64": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm64/-/resolver-binding-android-arm64-1.12.2.tgz", + "integrity": "sha512-YGCRZv/9GLhwmz6mYDeTsm/92BAyR28l6c2ReweVW5pWgfsitWLY8upvfRlGdoyD8HjeTHSYJWyZGD4KJA/nFQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@unrs/resolver-binding-darwin-arm64": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-arm64/-/resolver-binding-darwin-arm64-1.12.2.tgz", + "integrity": "sha512-u9DiNT1auQMO20A9SyTuG3wUgQWB9Z7KjAg0uFuCDR1FsAY8A0CG2S6JpHS1xwm/w1G08bjXZDcyOCjv1WAm2w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@unrs/resolver-binding-darwin-x64": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-x64/-/resolver-binding-darwin-x64-1.12.2.tgz", + "integrity": "sha512-f7rPLi/T1HVKZu/u6t87lroib16n8vrSzcyxI7lg4BGO9UF26KhQL44sd9eOUgrTYhvRXtWOIZT5PejdPyJfUA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@unrs/resolver-binding-freebsd-x64": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-freebsd-x64/-/resolver-binding-freebsd-x64-1.12.2.tgz", + "integrity": "sha512-BpcOjWCJub6nRZUS2zA20pmLvjtqAtGejETaIyRLiZiQf++cbrjltLA5NN/xaXfqeOBOSlMFbemIl5/S5tljmg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm-gnueabihf": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-gnueabihf/-/resolver-binding-linux-arm-gnueabihf-1.12.2.tgz", + "integrity": "sha512-vZTDvdSISZjJx66OzJqtsOhzifbqRjbmI1Mnu49fQDwog5GtDI4QidRiEAYbZCRj9C8YZEW+3ZjqsyS9GR4k2A==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm-musleabihf": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-musleabihf/-/resolver-binding-linux-arm-musleabihf-1.12.2.tgz", + "integrity": "sha512-BiPI+IrIlwcW4nLLMM21+B1dFPzd55yAVgVGrdgDjNef+ch03GdxrcyaIz8X9SsQirh/kCQ7mviyWlMxdh2D7g==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm64-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-gnu/-/resolver-binding-linux-arm64-gnu-1.12.2.tgz", + "integrity": "sha512-zJc0H99FEPoFfSrNpa91HYfxzfAJCr502oxNK1cfdC9hlaFI43RT+JFCann9JUgZmLzzntChHyn13Sgn9ljHNg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm64-musl": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-musl/-/resolver-binding-linux-arm64-musl-1.12.2.tgz", + "integrity": "sha512-KQ3Lki6l+Pz1k/eBipN41ES+YUK30beLGb9YqcB1O542cyLCNE6GaxrfcY3T6EezmGGk84wb5XyO9loTM9tkcA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-loong64-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-loong64-gnu/-/resolver-binding-linux-loong64-gnu-1.12.2.tgz", + "integrity": "sha512-3SJGEh1DborhG6pyxvhPzCT4bbSIVihsvgJc13P1bHG7KLdNDaF9T3gsTwFc7Jw/5Y5/iWOjkEx7Zy0NvCGX3Q==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-loong64-musl": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-loong64-musl/-/resolver-binding-linux-loong64-musl-1.12.2.tgz", + "integrity": "sha512-jiuG/Obbel7uw1PwHNFfrkiKhLAF6mnyZ6aWlOAVN9WqKm8v0OFGnciJIHu8+CMvXLQ8AD51LPzAoUfT21D5Ew==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-ppc64-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-ppc64-gnu/-/resolver-binding-linux-ppc64-gnu-1.12.2.tgz", + "integrity": "sha512-q7xRvVpmcfeL+LlZg8Pbbo6QaTZwDU5BaGZbwfhkEsXJn3Was8xYfE0RBH266xZt0rM6B7i8xAYIvjthuUIWHg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-riscv64-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-gnu/-/resolver-binding-linux-riscv64-gnu-1.12.2.tgz", + "integrity": "sha512-0CVdx6lcnT3Q9inOH8tsMIOJ6ImndllMjqJHg8RLVdB7Vq4SfkEXl9mCSsVNuNA4MCYycRicCUxPCabVHJRr6A==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-riscv64-musl": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-musl/-/resolver-binding-linux-riscv64-musl-1.12.2.tgz", + "integrity": "sha512-iOwlRo9vnp6R6ohHQS11n0NnfdXx/omhkocmIfaPRpQhKZ+3BDMkkdRVh53qjkFkpPddf+FETA28NwGN7l5l+w==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-s390x-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-s390x-gnu/-/resolver-binding-linux-s390x-gnu-1.12.2.tgz", + "integrity": "sha512-HYJtLfXq94q8iZNFT1lknx258wlkkWhZeUXJRqzKBBUJ00CvZ+N33zgbCqimLjsyw5Va6uUxhVa12mI+kaveEw==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-x64-gnu": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-gnu/-/resolver-binding-linux-x64-gnu-1.12.2.tgz", + "integrity": "sha512-mPsUhunKKDih5O96Y6enDQyHc1SqBPlY1E/SfMWDM3EdJ95Z9CArPeCVwCCqbP45ljvivdEk8Fxn+SIb1rDAJQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-x64-musl": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-musl/-/resolver-binding-linux-x64-musl-1.12.2.tgz", + "integrity": "sha512-azrt6+5ydLd8Vt210AAFis/lZevSfPw93EJRIJG+xPu4WCJ8K0kppCTpMyLPcKT7H15M4Jnt2tMp5bOvCkRC6A==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-openharmony-arm64": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-openharmony-arm64/-/resolver-binding-openharmony-arm64-1.12.2.tgz", + "integrity": "sha512-YZ9hP4O0X9PQb8eO980qmLNGH4zT3I9+SZTdt0Pr0YyuGQhYKoOZkV02VzrzyOZJ5xIJ3UFIenKkUkGg8GjgWQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@unrs/resolver-binding-wasm32-wasi": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-wasm32-wasi/-/resolver-binding-wasm32-wasi-1.12.2.tgz", + "integrity": "sha512-tYFDIkMxSflfEc/h92ZWNsZlHSwgimbNHSO3PL2JWQHfCuC2q316jMyYU9TIWZsFK2bQwyK5VAdYgn8ygPj69A==", + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "1.10.0", + "@emnapi/runtime": "1.10.0", + "@napi-rs/wasm-runtime": "^1.1.4" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@unrs/resolver-binding-wasm32-wasi/node_modules/@emnapi/core": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", + "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/wasi-threads": "1.2.1", + "tslib": "^2.4.0" + } + }, + "node_modules/@unrs/resolver-binding-wasm32-wasi/node_modules/@emnapi/runtime": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", + "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@unrs/resolver-binding-wasm32-wasi/node_modules/@emnapi/wasi-threads": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", + "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@unrs/resolver-binding-win32-arm64-msvc": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-arm64-msvc/-/resolver-binding-win32-arm64-msvc-1.12.2.tgz", + "integrity": "sha512-qzNyg3xL0VPQmCaUh+N5jSitce6k+uCBfMDesWRnlULOZaqUkaJ0ybdT+UqlAWJoQjuqfIU/0Ptx9bteN4D82g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@unrs/resolver-binding-win32-ia32-msvc": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-ia32-msvc/-/resolver-binding-win32-ia32-msvc-1.12.2.tgz", + "integrity": "sha512-WD9sY00OfpHVGfsnHZoA8jVT+esS/Bg8z8jzxp5BnDCjjwsuKsPQrzswwpFy4J1AUJbXPRfkpcX0mXrzeXW79g==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@unrs/resolver-binding-win32-x64-msvc": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-x64-msvc/-/resolver-binding-win32-x64-msvc-1.12.2.tgz", + "integrity": "sha512-nAB74NfSNKknqQ1RrYj6uz8FcXEomu/MATJZxh/x+BArzN2U3JbOYC0APYzUIGhVY3m5hRxA8VPNdPBoG8txlA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@vis.gl/react-mapbox": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/@vis.gl/react-mapbox/-/react-mapbox-8.1.1.tgz", + "integrity": "sha512-KMDTjtWESXxHS4uqWxjsvgQUHvuL3Z6SdKe68o7Nxma2qUfuyH3x4TCkIqGn3FQTrFvZLWvTnSAbGvtm+Kd13A==", + "license": "MIT", + "peerDependencies": { + "mapbox-gl": ">=3.5.0", + "react": ">=16.3.0", + "react-dom": ">=16.3.0" + }, + "peerDependenciesMeta": { + "mapbox-gl": { + "optional": true + } + } + }, + "node_modules/@vis.gl/react-maplibre": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/@vis.gl/react-maplibre/-/react-maplibre-8.1.1.tgz", + "integrity": "sha512-iUOfzJAhFAJwEZp1644tQb7LOTFgi5/GzdaztkhzNgFVuoF2Ez7guvwZjQAKB9CN2TlHTgNuYH8UW85kO7cVhw==", + "license": "MIT", + "dependencies": { + "@maplibre/maplibre-gl-style-spec": "^19.2.1" + }, + "peerDependencies": { + "maplibre-gl": ">=4.0.0", + "react": ">=16.3.0", + "react-dom": ">=16.3.0" + }, + "peerDependenciesMeta": { + "maplibre-gl": { + "optional": true + } + } + }, + "node_modules/@vis.gl/react-maplibre/node_modules/@maplibre/maplibre-gl-style-spec": { + "version": "19.3.3", + "resolved": "https://registry.npmjs.org/@maplibre/maplibre-gl-style-spec/-/maplibre-gl-style-spec-19.3.3.tgz", + "integrity": "sha512-cOZZOVhDSulgK0meTsTkmNXb1ahVvmTmWmfx9gRBwc6hq98wS9JP35ESIoNq3xqEan+UN+gn8187Z6E4NKhLsw==", + "license": "ISC", + "dependencies": { + "@mapbox/jsonlint-lines-primitives": "~2.0.2", + "@mapbox/unitbezier": "^0.0.1", + "json-stringify-pretty-compact": "^3.0.0", + "minimist": "^1.2.8", + "rw": "^1.3.3", + "sort-object": "^3.0.3" + }, + "bin": { + "gl-style-format": "dist/gl-style-format.mjs", + "gl-style-migrate": "dist/gl-style-migrate.mjs", + "gl-style-validate": "dist/gl-style-validate.mjs" + } + }, + "node_modules/@vis.gl/react-maplibre/node_modules/json-stringify-pretty-compact": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/json-stringify-pretty-compact/-/json-stringify-pretty-compact-3.0.0.tgz", + "integrity": "sha512-Rc2suX5meI0S3bfdZuA7JMFBGkJ875ApfVyq2WHELjBiiG22My/l7/8zPpH/CfFVQHuVLd8NLR0nv6vi0BYYKA==", + "license": "MIT" + }, + "node_modules/@vitejs/plugin-react": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.2.tgz", + "integrity": "sha512-DlSMqo4WhThw4vB8Mpn0Woe9J+Jfq1geJ61AKW0QEgLzGMNwtIMdxbDUzLxcun8W7NbJO0e2Jg/Nxm3cCSVzzg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@rolldown/pluginutils": "^1.0.0" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "peerDependencies": { + "@rolldown/plugin-babel": "^0.1.7 || ^0.2.0", + "babel-plugin-react-compiler": "^1.0.0", + "vite": "^8.0.0" + }, + "peerDependenciesMeta": { + "@rolldown/plugin-babel": { + "optional": true + }, + "babel-plugin-react-compiler": { + "optional": true + } + } + }, + "node_modules/@vitest/expect": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-3.2.4.tgz", + "integrity": "sha512-Io0yyORnB6sikFlt8QW5K7slY4OjqNX9jmJQ02QDda8lyM6B5oNgVWoSoKPac8/kgnCUzuHQKrSLtu/uOqqrig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/chai": "^5.2.2", + "@vitest/spy": "3.2.4", + "@vitest/utils": "3.2.4", + "chai": "^5.2.0", + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/pretty-format": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-3.2.4.tgz", + "integrity": "sha512-IVNZik8IVRJRTr9fxlitMKeJeXFFFN0JaB9PHPGQ8NKQbGpfjlTx9zO4RefN8gp7eqjNy8nyK3NZmBzOPeIxtA==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/runner": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-3.2.4.tgz", + "integrity": "sha512-oukfKT9Mk41LreEW09vt45f8wx7DordoWUZMYdY/cyAk7w5TWkTRCNZYF7sX7n2wB7jyGAl74OxgwhPgKaqDMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/utils": "3.2.4", + "pathe": "^2.0.3", + "strip-literal": "^3.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/snapshot": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-3.2.4.tgz", + "integrity": "sha512-dEYtS7qQP2CjU27QBC5oUOxLE/v5eLkGqPE0ZKEIDGMs4vKWe7IjgLOeauHsR0D5YuuycGRO5oSRXnwnmA78fQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "3.2.4", + "magic-string": "^0.30.17", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/spy": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-3.2.4.tgz", + "integrity": "sha512-vAfasCOe6AIK70iP5UD11Ac4siNUNJ9i/9PZ3NKx07sG6sUxeag1LWdNrMWeKKYBLlzuK+Gn65Yd5nyL6ds+nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyspy": "^4.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/utils": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-3.2.4.tgz", + "integrity": "sha512-fB2V0JFrQSMsCo9HiSq3Ezpdv4iYaXRG1Sx8edX3MwxfyNn83mKiGzOcH+Fkxt4MHxr3y42fQi1oeAInqgX2QA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "3.2.4", + "loupe": "^3.1.4", + "tinyrainbow": "^2.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/acorn": { + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", + "dev": true, + "license": "MIT", + "peer": true, + "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/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "license": "MIT", + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/ajv": { + "version": "6.15.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz", + "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==", + "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/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/aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "dequal": "^2.0.3" + } + }, + "node_modules/arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha512-sKpyeERZ02v1FeCZT8lrfJq5u6goHCtpTAzPwJYe7c8SPFOboNjNg1vz2L4VTn9T4PQxEx13TbXLmYUcS6Ug7Q==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-buffer-byte-length": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", + "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "is-array-buffer": "^3.0.5" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-includes": { + "version": "3.1.9", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.9.tgz", + "integrity": "sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.4", + "define-properties": "^1.2.1", + "es-abstract": "^1.24.0", + "es-object-atoms": "^1.1.1", + "get-intrinsic": "^1.3.0", + "is-string": "^1.1.1", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.findlast": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", + "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flat": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz", + "integrity": "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.5", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flatmap": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz", + "integrity": "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.5", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.tosorted": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.4.tgz", + "integrity": "sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.3", + "es-errors": "^1.3.0", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/arraybuffer.prototype.slice": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", + "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.6", + "is-array-buffer": "^3.0.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/assertion-error": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", + "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha512-Q+JC7Whu8HhmTdBph/Tq59IoRtoy6KAm5zzPv00WdujX82lbAL8K7WVjne7vdCsAmbF4AYaDOPyO3k0kl8qIrw==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ast-types-flow": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.8.tgz", + "integrity": "sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/async-function": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/async-function/-/async-function-1.0.0.tgz", + "integrity": "sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "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/available-typed-arrays": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/axe-core": { + "version": "4.11.4", + "resolved": "https://registry.npmjs.org/axe-core/-/axe-core-4.11.4.tgz", + "integrity": "sha512-KunSNx+TVpkAw/6ULfhnx+HWRecjqZGTOyquAoWHYLRSdK1tB5Ihce1ZW+UY3fj33bYAFWPu7W/GRSmmrCGuxA==", + "dev": true, + "license": "MPL-2.0", + "engines": { + "node": ">=4" + } + }, + "node_modules/axios": { + "version": "1.16.1", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.16.1.tgz", + "integrity": "sha512-caYkukvroVPO8KrzuJEb50Hm07KwfBZPEC3VeFHTsqWHvKTsy54hjJz9BS/cdaypROE2rH6xvm9mHX4fgWkr3A==", + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.16.0", + "form-data": "^4.0.5", + "https-proxy-agent": "^5.0.1", + "proxy-from-env": "^2.1.0" + } + }, + "node_modules/axobject-query": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-4.1.0.tgz", + "integrity": "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/babel-plugin-macros": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/babel-plugin-macros/-/babel-plugin-macros-3.1.0.tgz", + "integrity": "sha512-Cg7TFGpIr01vOQNODXOOaGz2NpCU5gl8x1qJFbb6hbZxR7XrcE2vtbAsTAbJ7/xwJtUuJEw8K8Zr/AE0LHlesg==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.12.5", + "cosmiconfig": "^7.0.0", + "resolve": "^1.19.0" + }, + "engines": { + "node": ">=10", + "npm": ">=6" + } + }, + "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/baseline-browser-mapping": { + "version": "2.10.32", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.32.tgz", + "integrity": "sha512-wbPvpyjJPC0zdfdKXxqEL3Ea+bOMD/87X4lftiJkkaBiuG6ALQy1SLmEd7BSmVCuwCQsBrCamgBoLyfFDD1EPg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.cjs" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.15.tgz", + "integrity": "sha512-EwOCDEex4quD37XhqM3omwtMoJjr//isUZz1JopUNWms+4Z2ViyM/k1YIRePpoVNnQhENnxtFjLaxNHrT7xIUg==", + "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/browserslist": { + "version": "4.28.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz", + "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "peer": true, + "dependencies": { + "baseline-browser-mapping": "^2.10.12", + "caniuse-lite": "^1.0.30001782", + "electron-to-chromium": "^1.5.328", + "node-releases": "^2.0.36", + "update-browserslist-db": "^1.2.3" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bytewise": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/bytewise/-/bytewise-1.1.0.tgz", + "integrity": "sha512-rHuuseJ9iQ0na6UDhnrRVDh8YnWVlU6xM3VH6q/+yHDeUH2zIhUzP+2/h3LIrhLDBtTqzWpE3p3tP/boefskKQ==", + "license": "MIT", + "dependencies": { + "bytewise-core": "^1.2.2", + "typewise": "^1.0.3" + } + }, + "node_modules/bytewise-core": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/bytewise-core/-/bytewise-core-1.2.3.tgz", + "integrity": "sha512-nZD//kc78OOxeYtRlVk8/zXqTB4gf/nlguL1ggWA8FuchMyOxcyHR4QPQZMUmA7czC+YnaBrPUCubqAWe50DaA==", + "license": "MIT", + "dependencies": { + "typewise-core": "^1.2" + } + }, + "node_modules/cac": { + "version": "6.7.14", + "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", + "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/call-bind": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.9.tgz", + "integrity": "sha512-a/hy+pNsFUTR+Iz8TCJvXudKVLAnz/DyeSUo10I5yvFDQJBFU2s9uqQpoSrJlroHUKoKqzg+epxyP9lqFdzfBQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "get-intrinsic": "^1.3.0", + "set-function-length": "^1.2.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001793", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001793.tgz", + "integrity": "sha512-iwSsYWaCOoh26cV8NwNRViHlrfUvYsHDfRVcbtmw0Kg6PJIZZXwMkj1442FYLBGkeUf1juAsU3DTfxW579mrPA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chai": { + "version": "5.3.3", + "resolved": "https://registry.npmjs.org/chai/-/chai-5.3.3.tgz", + "integrity": "sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "assertion-error": "^2.0.1", + "check-error": "^2.1.1", + "deep-eql": "^5.0.1", + "loupe": "^3.1.0", + "pathval": "^2.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "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/check-error": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.3.tgz", + "integrity": "sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 16" + } + }, + "node_modules/clsx": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", + "license": "MIT", + "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/comment-parser": { + "version": "1.4.7", + "resolved": "https://registry.npmjs.org/comment-parser/-/comment-parser-1.4.7.tgz", + "integrity": "sha512-0h+uSNtQGW3D98eQt3jJ8L06Fves8hncB4V/PKdw/Qb8Hnk19VaKuTr55UNRYiSoVa7WwrFls+rh3ux9agmkeQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 12.0.0" + } + }, + "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/convert-source-map": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", + "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", + "license": "MIT" + }, + "node_modules/cookie": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.1.1.tgz", + "integrity": "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/cosmiconfig": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.1.0.tgz", + "integrity": "sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==", + "license": "MIT", + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/cosmiconfig/node_modules/yaml": { + "version": "1.10.3", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.3.tgz", + "integrity": "sha512-vIYeF1u3CjlhAFekPPAk2h/Kv4T3mAkMox5OymRiJQB0spDP10LHvt+K7G9Ny6NuuMAb25/6n1qyUjAcGNf/AA==", + "license": "ISC", + "engines": { + "node": ">= 6" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/css.escape": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz", + "integrity": "sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cssstyle": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-4.6.0.tgz", + "integrity": "sha512-2z+rWdzbbSZv6/rhtvzvqeZQHrBaqgogqt85sqFNbabZOuFbCVFb8kPeEtZjiKkbrm395irpNKiYeFeLiQnFPg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@asamuzakjp/css-color": "^3.2.0", + "rrweb-cssom": "^0.8.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/csstype": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", + "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "license": "MIT" + }, + "node_modules/damerau-levenshtein": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz", + "integrity": "sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==", + "dev": true, + "license": "BSD-2-Clause" + }, + "node_modules/data-urls": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-5.0.0.tgz", + "integrity": "sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==", + "dev": true, + "license": "MIT", + "dependencies": { + "whatwg-mimetype": "^4.0.0", + "whatwg-url": "^14.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/data-view-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", + "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-length": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", + "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/inspect-js" + } + }, + "node_modules/data-view-byte-offset": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", + "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.6.0", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.6.0.tgz", + "integrity": "sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==", + "dev": true, + "license": "MIT" + }, + "node_modules/deep-eql": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz", + "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "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/define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/define-properties": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-data-property": "^1.0.1", + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/detect-libc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", + "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, + "node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/dom-accessibility-api": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", + "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", + "dev": true, + "license": "MIT" + }, + "node_modules/dom-helpers": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/dom-helpers/-/dom-helpers-5.2.1.tgz", + "integrity": "sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==", + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.8.7", + "csstype": "^3.0.2" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/earcut": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/earcut/-/earcut-3.0.2.tgz", + "integrity": "sha512-X7hshQbLyMJ/3RPhyObLARM2sNxxmRALLKx1+NVFFnQ9gKzmCrxm9+uLIAdBcvc8FNLpctqlQ2V6AE92Ol9UDQ==", + "license": "ISC" + }, + "node_modules/effector": { + "version": "23.4.4", + "resolved": "https://registry.npmjs.org/effector/-/effector-23.4.4.tgz", + "integrity": "sha512-QkZboRN28K/iwxigDhlJcI3ux3aNbt8kYGGH/GkqWG0OlGeyuBhb7PdM89Iu+ogV8Lmz16xIlwnXR2UNWI6psg==", + "funding": [ + { + "type": "patreon", + "url": "https://www.patreon.com/zero_bias" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/effector" + } + ], + "license": "MIT", + "peer": true, + "engines": { + "node": ">=11.0.0" + } + }, + "node_modules/effector-react": { + "version": "23.3.0", + "resolved": "https://registry.npmjs.org/effector-react/-/effector-react-23.3.0.tgz", + "integrity": "sha512-QR0+x1EnbiWhO80Yc0GVF+I9xCYoxBm3t+QLB5Wg+1uY1Q1BrSWDmKvJaJJZ/+9BU4RAr25yS5J2EkdWnicu8g==", + "funding": [ + { + "type": "patreon", + "url": "https://www.patreon.com/zero_bias" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/effector" + } + ], + "license": "MIT", + "dependencies": { + "use-sync-external-store": "^1.0.0" + }, + "engines": { + "node": ">=11.0.0" + }, + "peerDependencies": { + "effector": "^23.0.0", + "react": ">=16.8.0 <20.0.0" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.5.363", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.363.tgz", + "integrity": "sha512-VjUKPyWzGnT1fujlkEGC/BvN70Hh70KXtAqcmniXviYlJC/ivcT+BWGPyxWVbJZLfvtKR6dqg1L7T7pgAMBtWA==", + "dev": true, + "license": "ISC" + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", + "dev": true, + "license": "MIT" + }, + "node_modules/entities": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/entities/-/entities-6.0.1.tgz", + "integrity": "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/error-ex": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/es-abstract": { + "version": "1.24.2", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.24.2.tgz", + "integrity": "sha512-2FpH9Q5i2RRwyEP1AylXe6nYLR5OhaJTZwmlcP0dL/+JCbgg7yyEo/sEK6HeGZRf3dFpWwThaRHVApXSkW3xeg==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-buffer-byte-length": "^1.0.2", + "arraybuffer.prototype.slice": "^1.0.4", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.8", + "call-bound": "^1.0.4", + "data-view-buffer": "^1.0.2", + "data-view-byte-length": "^1.0.2", + "data-view-byte-offset": "^1.0.1", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "es-set-tostringtag": "^2.1.0", + "es-to-primitive": "^1.3.0", + "function.prototype.name": "^1.1.8", + "get-intrinsic": "^1.3.0", + "get-proto": "^1.0.1", + "get-symbol-description": "^1.1.0", + "globalthis": "^1.0.4", + "gopd": "^1.2.0", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "internal-slot": "^1.1.0", + "is-array-buffer": "^3.0.5", + "is-callable": "^1.2.7", + "is-data-view": "^1.0.2", + "is-negative-zero": "^2.0.3", + "is-regex": "^1.2.1", + "is-set": "^2.0.3", + "is-shared-array-buffer": "^1.0.4", + "is-string": "^1.1.1", + "is-typed-array": "^1.1.15", + "is-weakref": "^1.1.1", + "math-intrinsics": "^1.1.0", + "object-inspect": "^1.13.4", + "object-keys": "^1.1.1", + "object.assign": "^4.1.7", + "own-keys": "^1.0.1", + "regexp.prototype.flags": "^1.5.4", + "safe-array-concat": "^1.1.3", + "safe-push-apply": "^1.0.0", + "safe-regex-test": "^1.1.0", + "set-proto": "^1.0.0", + "stop-iteration-iterator": "^1.1.0", + "string.prototype.trim": "^1.2.10", + "string.prototype.trimend": "^1.0.9", + "string.prototype.trimstart": "^1.0.8", + "typed-array-buffer": "^1.0.3", + "typed-array-byte-length": "^1.0.3", + "typed-array-byte-offset": "^1.0.4", + "typed-array-length": "^1.0.7", + "unbox-primitive": "^1.1.0", + "which-typed-array": "^1.1.19" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-iterator-helpers": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.3.2.tgz", + "integrity": "sha512-HVLACW1TppGYjJ8H6/jqH/pqOtKRw6wMlrB23xfExmFWxFquAIWCmwoLsOyN96K4a5KbmOf5At9ZUO3GZbetAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.9", + "call-bound": "^1.0.4", + "define-properties": "^1.2.1", + "es-abstract": "^1.24.2", + "es-errors": "^1.3.0", + "es-set-tostringtag": "^2.1.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.3.0", + "globalthis": "^1.0.4", + "gopd": "^1.2.0", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.2.0", + "has-symbols": "^1.1.0", + "internal-slot": "^1.1.0", + "iterator.prototype": "^1.1.5", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-module-lexer": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", + "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", + "dev": true, + "license": "MIT" + }, + "node_modules/es-object-atoms": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.2.tgz", + "integrity": "sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-set-tostringtag": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", + "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.6", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-shim-unscopables": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.1.0.tgz", + "integrity": "sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-to-primitive": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", + "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-callable": "^1.2.7", + "is-date-object": "^1.0.5", + "is-symbol": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/esbuild": { + "version": "0.27.7", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.27.7.tgz", + "integrity": "sha512-IxpibTjyVnmrIQo5aqNpCgoACA/dTKLTlhMHihVHhdkxKyPO1uBBthumT0rdHmcsk9uMonIWS0m4FljWzILh3w==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "peer": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.27.7", + "@esbuild/android-arm": "0.27.7", + "@esbuild/android-arm64": "0.27.7", + "@esbuild/android-x64": "0.27.7", + "@esbuild/darwin-arm64": "0.27.7", + "@esbuild/darwin-x64": "0.27.7", + "@esbuild/freebsd-arm64": "0.27.7", + "@esbuild/freebsd-x64": "0.27.7", + "@esbuild/linux-arm": "0.27.7", + "@esbuild/linux-arm64": "0.27.7", + "@esbuild/linux-ia32": "0.27.7", + "@esbuild/linux-loong64": "0.27.7", + "@esbuild/linux-mips64el": "0.27.7", + "@esbuild/linux-ppc64": "0.27.7", + "@esbuild/linux-riscv64": "0.27.7", + "@esbuild/linux-s390x": "0.27.7", + "@esbuild/linux-x64": "0.27.7", + "@esbuild/netbsd-arm64": "0.27.7", + "@esbuild/netbsd-x64": "0.27.7", + "@esbuild/openbsd-arm64": "0.27.7", + "@esbuild/openbsd-x64": "0.27.7", + "@esbuild/openharmony-arm64": "0.27.7", + "@esbuild/sunos-x64": "0.27.7", + "@esbuild/win32-arm64": "0.27.7", + "@esbuild/win32-ia32": "0.27.7", + "@esbuild/win32-x64": "0.27.7" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "9.39.4", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.4.tgz", + "integrity": "sha512-XoMjdBOwe/esVgEvLmNsD3IRHkm7fbKIUGvrleloJXUZgDHig2IPWNniv+GwjyJXzuNqVjlr5+4yVUZjycJwfQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.21.2", + "@eslint/config-helpers": "^0.4.2", + "@eslint/core": "^0.17.0", + "@eslint/eslintrc": "^3.3.5", + "@eslint/js": "9.39.4", + "@eslint/plugin-kit": "^0.4.1", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.14.0", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^8.4.0", + "eslint-visitor-keys": "^4.2.1", + "espree": "^10.4.0", + "esquery": "^1.5.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.5", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } + } + }, + "node_modules/eslint-config-prettier": { + "version": "10.1.8", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.8.tgz", + "integrity": "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "eslint-config-prettier": "bin/cli.js" + }, + "funding": { + "url": "https://opencollective.com/eslint-config-prettier" + }, + "peerDependencies": { + "eslint": ">=7.0.0" + } + }, + "node_modules/eslint-import-context": { + "version": "0.1.9", + "resolved": "https://registry.npmjs.org/eslint-import-context/-/eslint-import-context-0.1.9.tgz", + "integrity": "sha512-K9Hb+yRaGAGUbwjhFNHvSmmkZs9+zbuoe3kFQ4V1wYjrepUFYM2dZAfNtjbbj3qsPfUfsA68Bx/ICWQMi+C8Eg==", + "dev": true, + "license": "MIT", + "dependencies": { + "get-tsconfig": "^4.10.1", + "stable-hash-x": "^0.2.0" + }, + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint-import-context" + }, + "peerDependencies": { + "unrs-resolver": "^1.0.0" + }, + "peerDependenciesMeta": { + "unrs-resolver": { + "optional": true + } + } + }, + "node_modules/eslint-import-resolver-node": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", + "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", + "dev": true, + "license": "MIT", + "dependencies": { + "debug": "^3.2.7", + "is-core-module": "^2.13.0", + "resolve": "^1.22.4" + } + }, + "node_modules/eslint-import-resolver-node/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-import-resolver-typescript": { + "version": "4.4.4", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-typescript/-/eslint-import-resolver-typescript-4.4.4.tgz", + "integrity": "sha512-1iM2zeBvrYmUNTj2vSC/90JTHDth+dfOfiNKkxApWRsTJYNrc8rOdxxIf5vazX+BiAXTeOT0UvWpGI/7qIWQOw==", + "dev": true, + "license": "ISC", + "dependencies": { + "debug": "^4.4.1", + "eslint-import-context": "^0.1.8", + "get-tsconfig": "^4.10.1", + "is-bun-module": "^2.0.0", + "stable-hash-x": "^0.2.0", + "tinyglobby": "^0.2.14", + "unrs-resolver": "^1.7.11" + }, + "engines": { + "node": "^16.17.0 || >=18.6.0" + }, + "funding": { + "url": "https://opencollective.com/eslint-import-resolver-typescript" + }, + "peerDependencies": { + "eslint": "*", + "eslint-plugin-import": "*", + "eslint-plugin-import-x": "*" + }, + "peerDependenciesMeta": { + "eslint-plugin-import": { + "optional": true + }, + "eslint-plugin-import-x": { + "optional": true + } + } + }, + "node_modules/eslint-module-utils": { + "version": "2.12.1", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.12.1.tgz", + "integrity": "sha512-L8jSWTze7K2mTg0vos/RuLRS5soomksDPoJLXIslC7c8Wmut3bx7CPpJijDcBZtxQ5lrbUdM+s0OlNbz0DCDNw==", + "dev": true, + "license": "MIT", + "dependencies": { + "debug": "^3.2.7" + }, + "engines": { + "node": ">=4" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } + } + }, + "node_modules/eslint-module-utils/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-plugin-boundaries": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-boundaries/-/eslint-plugin-boundaries-6.0.2.tgz", + "integrity": "sha512-wSHgiYeMEbziP91lH0UQ9oslgF2djG1x+LV9z/qO19ggMKZaCB8pKIGePHAY91eLF4EAgpsxQk8MRSFGRPfPzw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@boundaries/elements": "2.0.1", + "chalk": "4.1.2", + "eslint-import-resolver-node": "0.3.9", + "eslint-module-utils": "2.12.1", + "handlebars": "4.7.9", + "micromatch": "4.0.8" + }, + "engines": { + "node": ">=18.18" + }, + "peerDependencies": { + "eslint": ">=6.0.0" + } + }, + "node_modules/eslint-plugin-import-x": { + "version": "4.16.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-import-x/-/eslint-plugin-import-x-4.16.2.tgz", + "integrity": "sha512-rM9K8UBHcWKpzQzStn1YRN2T5NvdeIfSVoKu/lKF41znQXHAUcBbYXe5wd6GNjZjTrP7viQ49n1D83x/2gYgIw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@package-json/types": "^0.0.12", + "@typescript-eslint/types": "^8.56.0", + "comment-parser": "^1.4.1", + "debug": "^4.4.1", + "eslint-import-context": "^0.1.9", + "is-glob": "^4.0.3", + "minimatch": "^9.0.3 || ^10.1.2", + "semver": "^7.7.2", + "stable-hash-x": "^0.2.0", + "unrs-resolver": "^1.9.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint-plugin-import-x" + }, + "peerDependencies": { + "@typescript-eslint/utils": "^8.56.0", + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "eslint-import-resolver-node": "*" + }, + "peerDependenciesMeta": { + "@typescript-eslint/utils": { + "optional": true + }, + "eslint-import-resolver-node": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-import-x/node_modules/balanced-match": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/eslint-plugin-import-x/node_modules/brace-expansion": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", + "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^4.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/eslint-plugin-import-x/node_modules/minimatch": { + "version": "10.2.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", + "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "brace-expansion": "^5.0.5" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/eslint-plugin-jsx-a11y": { + "version": "6.10.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-jsx-a11y/-/eslint-plugin-jsx-a11y-6.10.2.tgz", + "integrity": "sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "aria-query": "^5.3.2", + "array-includes": "^3.1.8", + "array.prototype.flatmap": "^1.3.2", + "ast-types-flow": "^0.0.8", + "axe-core": "^4.10.0", + "axobject-query": "^4.1.0", + "damerau-levenshtein": "^1.0.8", + "emoji-regex": "^9.2.2", + "hasown": "^2.0.2", + "jsx-ast-utils": "^3.3.5", + "language-tags": "^1.0.9", + "minimatch": "^3.1.2", + "object.fromentries": "^2.0.8", + "safe-regex-test": "^1.0.3", + "string.prototype.includes": "^2.0.1" + }, + "engines": { + "node": ">=4.0" + }, + "peerDependencies": { + "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9" + } + }, + "node_modules/eslint-plugin-jsx-a11y/node_modules/aria-query": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.2.tgz", + "integrity": "sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/eslint-plugin-prettier": { + "version": "5.5.5", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.5.5.tgz", + "integrity": "sha512-hscXkbqUZ2sPithAuLm5MXL+Wph+U7wHngPBv9OMWwlP8iaflyxpjTYZkmdgB4/vPIhemRlBEoLrH7UC1n7aUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "prettier-linter-helpers": "^1.0.1", + "synckit": "^0.11.12" + }, + "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": ">= 7.0.0 <10.0.0 || >=10.1.0", + "prettier": ">=3.0.0" + }, + "peerDependenciesMeta": { + "@types/eslint": { + "optional": true + }, + "eslint-config-prettier": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-react": { + "version": "7.37.5", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.37.5.tgz", + "integrity": "sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-includes": "^3.1.8", + "array.prototype.findlast": "^1.2.5", + "array.prototype.flatmap": "^1.3.3", + "array.prototype.tosorted": "^1.1.4", + "doctrine": "^2.1.0", + "es-iterator-helpers": "^1.2.1", + "estraverse": "^5.3.0", + "hasown": "^2.0.2", + "jsx-ast-utils": "^2.4.1 || ^3.0.0", + "minimatch": "^3.1.2", + "object.entries": "^1.1.9", + "object.fromentries": "^2.0.8", + "object.values": "^1.2.1", + "prop-types": "^15.8.1", + "resolve": "^2.0.0-next.5", + "semver": "^6.3.1", + "string.prototype.matchall": "^4.0.12", + "string.prototype.repeat": "^1.0.0" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7" + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.1.1.tgz", + "integrity": "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.24.4", + "@babel/parser": "^7.24.4", + "hermes-parser": "^0.25.1", + "zod": "^3.25.0 || ^4.0.0", + "zod-validation-error": "^3.5.0 || ^4.0.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 || ^10.0.0" + } + }, + "node_modules/eslint-plugin-react-refresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.5.2.tgz", + "integrity": "sha512-hmgTH57GfzoTFjVN0yBwTggnsVUF2tcqi7RJZHqi9lIezSs4eFyAMktA68YD4r5kNw1mxyY4dmkyoFDb3FIqrA==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "eslint": "^9 || ^10" + } + }, + "node_modules/eslint-plugin-react/node_modules/is-core-module": { + "version": "2.16.2", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.2.tgz", + "integrity": "sha512-evOr8xfXKxE6qSR0hSXL2r3sd7ALj8+7jQEUvPYcm5sgZFdJ+AYzT6yNmJenvIYQBgIGwfwz08sL8zoL7yq2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/eslint-plugin-react/node_modules/resolve": { + "version": "2.0.0-next.7", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.7.tgz", + "integrity": "sha512-tqt+NBWwyaMgw3zDsnygx4CByWjQEJHOPMdslYhppaQSJUtL/D4JO9CcBBlhPoI8lz9oJIDXkwXfhF4aWqP8xQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "is-core-module": "^2.16.2", + "node-exports-info": "^1.6.0", + "object-keys": "^1.1.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/eslint-plugin-react/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-scope": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", + "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", + "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", + "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.15.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.2.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", + "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": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "^1.0.0" + } + }, + "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/expect-type": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.3.0.tgz", + "integrity": "sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "license": "MIT", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "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-equals": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/fast-equals/-/fast-equals-5.4.0.tgz", + "integrity": "sha512-jt2DW/aNFNwke7AUd+Z+e6pz39KO5rzdbbFCg2sGafS4mk13MI7Z8O5z9cADNn5lhGODIgLwug6TZO2ctf7kcw==", + "license": "MIT", + "engines": { + "node": ">=6.0.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/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.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-root": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-root/-/find-root-1.1.0.tgz", + "integrity": "sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng==", + "license": "MIT" + }, + "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": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/flatted": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz", + "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==", + "dev": true, + "license": "ISC" + }, + "node_modules/follow-redirects": { + "version": "1.16.0", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.16.0.tgz", + "integrity": "sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/for-each": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", + "integrity": "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-callable": "^1.2.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/form-data": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz", + "integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==", + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "es-set-tostringtag": "^2.1.0", + "hasown": "^2.0.2", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "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/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/function.prototype.name": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", + "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "define-properties": "^1.2.1", + "functions-have-names": "^1.2.3", + "hasown": "^2.0.2", + "is-callable": "^1.2.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/generator-function": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/generator-function/-/generator-function-2.0.1.tgz", + "integrity": "sha512-SFdFmIJi+ybC0vjlHN0ZGVGHc3lgE0DxPAT0djjVg+kjOnSqclqmj0KQ7ykTOLP6YxoqOvuAODGdcHJn+43q3g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-symbol-description": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", + "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-tsconfig": { + "version": "4.14.0", + "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.14.0.tgz", + "integrity": "sha512-yTb+8DXzDREzgvYmh6s9vHsSVCHeC0G3PI5bEXNBHtmshPnO+S5O7qgLEOn0I5QvMy6kpZN8K1NKGyilLb93wA==", + "dev": true, + "license": "MIT", + "dependencies": { + "resolve-pkg-maps": "^1.0.0" + }, + "funding": { + "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" + } + }, + "node_modules/get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha512-Ln0UQDlxH1BapMu3GPtf7CuYNwRZf2gwCuPqbyG6pB8WfmFpzqcy4xtAaAMUhnNqjMKTiCPZG2oMT3YSx8U2NA==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gl-matrix": { + "version": "3.4.4", + "resolved": "https://registry.npmjs.org/gl-matrix/-/gl-matrix-3.4.4.tgz", + "integrity": "sha512-latSnyDNt/8zYUB6VIJ6PCh2jBjJX6gnDsoCZ7LyW7GkqrD51EWwa9qCoGixj8YqBtETQK/xY7OmpTF8xz1DdQ==", + "license": "MIT" + }, + "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": "17.6.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-17.6.0.tgz", + "integrity": "sha512-sepffkT8stwnIYbsMBpoCHJuJM5l98FUF2AnE07hfvE0m/qp3R586hw4jF4uadbhvg1ooIdzuu7CsfD2jzCaNA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/globalthis": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", + "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-properties": "^1.2.1", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/handlebars": { + "version": "4.7.9", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.9.tgz", + "integrity": "sha512-4E71E0rpOaQuJR2A3xDZ+GM1HyWYv1clR58tC8emQNeQe3RH7MAzSbat+V0wG78LQBo6m6bzSG/L4pBuCsgnUQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, + "node_modules/handlebars/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-bigints": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", + "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/has-property-descriptors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-define-property": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-proto": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", + "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "license": "MIT", + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.3.tgz", + "integrity": "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg==", + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/hermes-estree": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz", + "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==", + "dev": true, + "license": "MIT" + }, + "node_modules/hermes-parser": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz", + "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "hermes-estree": "0.25.1" + } + }, + "node_modules/hoist-non-react-statics": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz", + "integrity": "sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==", + "license": "BSD-3-Clause", + "dependencies": { + "react-is": "^16.7.0" + } + }, + "node_modules/hoist-non-react-statics/node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "license": "MIT" + }, + "node_modules/html-encoding-sniffer": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-4.0.0.tgz", + "integrity": "sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "whatwg-encoding": "^3.1.1" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/http-proxy-agent": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", + "integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==", + "dev": true, + "license": "MIT", + "dependencies": { + "agent-base": "^7.1.0", + "debug": "^4.3.4" + }, + "engines": { + "node": ">= 14" + } + }, + "node_modules/http-proxy-agent/node_modules/agent-base": { + "version": "7.1.4", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.4.tgz", + "integrity": "sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14" + } + }, + "node_modules/https-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", + "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", + "license": "MIT", + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "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/indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/internal-slot": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", + "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "hasown": "^2.0.2", + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-array-buffer": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", + "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "license": "MIT" + }, + "node_modules/is-async-function": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.1.1.tgz", + "integrity": "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "async-function": "^1.0.0", + "call-bound": "^1.0.3", + "get-proto": "^1.0.1", + "has-tostringtag": "^1.0.2", + "safe-regex-test": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-bigint": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", + "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-bigints": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-boolean-object": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.2.tgz", + "integrity": "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-bun-module": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-bun-module/-/is-bun-module-2.0.0.tgz", + "integrity": "sha512-gNCGbnnnnFAUGKeZ9PdbyeGYJqewpmc2aKHUEMO5nQPWU9lOmv7jcmQIv+qHD8fXW6W7qfuCwX4rY9LNRjXrkQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.7.1" + } + }, + "node_modules/is-callable": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", + "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-data-view": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", + "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "get-intrinsic": "^1.2.6", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-date-object": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", + "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "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-finalizationregistry": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", + "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-generator-function": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.1.2.tgz", + "integrity": "sha512-upqt1SkGkODW9tsGNG5mtXTXtECizwtS2kA161M+gJPc1xdb/Ax629af6YrTwcOeQHbewrPNlE5Dx7kzvXTizA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.4", + "generator-function": "^2.0.0", + "get-proto": "^1.0.1", + "has-tostringtag": "^1.0.2", + "safe-regex-test": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-map": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", + "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-negative-zero": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-number-object": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", + "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "license": "MIT", + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-potential-custom-element-name": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", + "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/is-regex": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", + "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "gopd": "^1.2.0", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-set": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", + "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", + "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-string": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", + "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-symbol": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", + "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "has-symbols": "^1.1.0", + "safe-regex-test": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-typed-array": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", + "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "which-typed-array": "^1.1.16" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakmap": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", + "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakref": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.1.tgz", + "integrity": "sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakset": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", + "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true, + "license": "MIT" + }, + "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/isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/iterator.prototype": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.5.tgz", + "integrity": "sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.6", + "get-proto": "^1.0.0", + "has-symbols": "^1.1.0", + "set-function-name": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsdom": { + "version": "26.1.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-26.1.0.tgz", + "integrity": "sha512-Cvc9WUhxSMEo4McES3P7oK3QaXldCfNWp7pl2NNeiIFlCoLr3kfq9kb1fxftiwk1FLV7CvpvDfonxtzUDeSOPg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cssstyle": "^4.2.1", + "data-urls": "^5.0.0", + "decimal.js": "^10.5.0", + "html-encoding-sniffer": "^4.0.0", + "http-proxy-agent": "^7.0.2", + "https-proxy-agent": "^7.0.6", + "is-potential-custom-element-name": "^1.0.1", + "nwsapi": "^2.2.16", + "parse5": "^7.2.1", + "rrweb-cssom": "^0.8.0", + "saxes": "^6.0.0", + "symbol-tree": "^3.2.4", + "tough-cookie": "^5.1.1", + "w3c-xmlserializer": "^5.0.0", + "webidl-conversions": "^7.0.0", + "whatwg-encoding": "^3.1.1", + "whatwg-mimetype": "^4.0.0", + "whatwg-url": "^14.1.1", + "ws": "^8.18.0", + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "canvas": "^3.0.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jsdom/node_modules/agent-base": { + "version": "7.1.4", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.4.tgz", + "integrity": "sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14" + } + }, + "node_modules/jsdom/node_modules/https-proxy-agent": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz", + "integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==", + "dev": true, + "license": "MIT", + "dependencies": { + "agent-base": "^7.1.2", + "debug": "4" + }, + "engines": { + "node": ">= 14" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "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-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "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/json-stringify-pretty-compact": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/json-stringify-pretty-compact/-/json-stringify-pretty-compact-4.0.0.tgz", + "integrity": "sha512-3CNZ2DnrpByG9Nqj6Xo8vqbjT4F6N+tb4Gb28ESAZjYZ5yqvmc56J+/kuIwkaAMOyblTQhUW7PxMkUb8Q36N3Q==", + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsx-ast-utils": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz", + "integrity": "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-includes": "^3.1.6", + "array.prototype.flat": "^1.3.1", + "object.assign": "^4.1.4", + "object.values": "^1.1.6" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/kdbush": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/kdbush/-/kdbush-4.1.0.tgz", + "integrity": "sha512-e9vurzrXJQrFX6ckpHP3bvj5l+9CnYzkxDNnNQ1h2QTqdWsUAJgXiKdGNcOa1EY85dU8KbQ+z/FdQdB7P+9yfQ==", + "license": "ISC" + }, + "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/language-subtag-registry": { + "version": "0.3.23", + "resolved": "https://registry.npmjs.org/language-subtag-registry/-/language-subtag-registry-0.3.23.tgz", + "integrity": "sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ==", + "dev": true, + "license": "CC0-1.0" + }, + "node_modules/language-tags": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/language-tags/-/language-tags-1.0.9.tgz", + "integrity": "sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA==", + "dev": true, + "license": "MIT", + "dependencies": { + "language-subtag-registry": "^0.3.20" + }, + "engines": { + "node": ">=0.10" + } + }, + "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/lightningcss": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", + "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", + "dev": true, + "license": "MPL-2.0", + "dependencies": { + "detect-libc": "^2.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-android-arm64": "1.32.0", + "lightningcss-darwin-arm64": "1.32.0", + "lightningcss-darwin-x64": "1.32.0", + "lightningcss-freebsd-x64": "1.32.0", + "lightningcss-linux-arm-gnueabihf": "1.32.0", + "lightningcss-linux-arm64-gnu": "1.32.0", + "lightningcss-linux-arm64-musl": "1.32.0", + "lightningcss-linux-x64-gnu": "1.32.0", + "lightningcss-linux-x64-musl": "1.32.0", + "lightningcss-win32-arm64-msvc": "1.32.0", + "lightningcss-win32-x64-msvc": "1.32.0" + } + }, + "node_modules/lightningcss-android-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", + "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", + "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", + "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", + "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", + "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", + "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", + "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", + "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", + "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", + "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", + "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "license": "MIT" + }, + "node_modules/linkifyjs": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/linkifyjs/-/linkifyjs-4.3.3.tgz", + "integrity": "sha512-P8aEP5U/D1/IlTY2OeYsErdwh9bGuLE30NcXtKEjgdHcahveQoQwM2yZNsioQHsWFz0P7KKudisbrzCgR0sDHg==", + "license": "MIT" + }, + "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.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/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "license": "MIT", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/loupe": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.2.1.tgz", + "integrity": "sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/lucide-react": { + "version": "1.16.0", + "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-1.16.0.tgz", + "integrity": "sha512-dYwyPzb4MEKpGUmNYk3WKWPnMrHs3FKM+q94kAnJrcDIqqn1hq2xY8scaS2ovsOCM5D51ey2gaRG3PBb1vgoYQ==", + "license": "ISC", + "peerDependencies": { + "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" + } + }, + "node_modules/lz-string": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", + "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", + "dev": true, + "license": "MIT", + "bin": { + "lz-string": "bin/bin.js" + } + }, + "node_modules/magic-string": { + "version": "0.30.21", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", + "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.5" + } + }, + "node_modules/maplibre-gl": { + "version": "5.24.0", + "resolved": "https://registry.npmjs.org/maplibre-gl/-/maplibre-gl-5.24.0.tgz", + "integrity": "sha512-ALyFxgtd5R+65UqZ/++lOqwWcC0SNho9c27fYSyLmG7AfnAul2o46F05aDJGPbFU57wos9dgcIySHs0Xe6ia3A==", + "license": "BSD-3-Clause", + "peer": true, + "dependencies": { + "@mapbox/jsonlint-lines-primitives": "^2.0.2", + "@mapbox/point-geometry": "^1.1.0", + "@mapbox/tiny-sdf": "^2.1.0", + "@mapbox/unitbezier": "^0.0.1", + "@mapbox/vector-tile": "^2.0.4", + "@mapbox/whoots-js": "^3.1.0", + "@maplibre/geojson-vt": "^6.1.0", + "@maplibre/maplibre-gl-style-spec": "^24.8.1", + "@maplibre/mlt": "^1.1.8", + "@maplibre/vt-pbf": "^4.3.0", + "@types/geojson": "^7946.0.16", + "earcut": "^3.0.2", + "gl-matrix": "^3.4.4", + "kdbush": "^4.0.2", + "murmurhash-js": "^1.0.0", + "pbf": "^4.0.1", + "potpack": "^2.1.0", + "quickselect": "^3.0.0", + "tinyqueue": "^3.0.0" + }, + "engines": { + "node": ">=16.14.0", + "npm": ">=8.1.0" + }, + "funding": { + "url": "https://github.com/maplibre/maplibre-gl-js?sponsor=1" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "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/min-indent": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/min-indent/-/min-indent-1.0.1.tgz", + "integrity": "sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/minimatch": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "license": "MIT" + }, + "node_modules/murmurhash-js": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/murmurhash-js/-/murmurhash-js-1.0.0.tgz", + "integrity": "sha512-TvmkNhkv8yct0SVBSy+o8wYzXjE4Zz3PCesbfs8HiCXXdcTuocApFv11UWlNFWKYsP2okqrhb7JNlSm9InBhIw==", + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.12", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz", + "integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==", + "dev": true, + "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/napi-postinstall": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/napi-postinstall/-/napi-postinstall-0.3.4.tgz", + "integrity": "sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ==", + "dev": true, + "license": "MIT", + "bin": { + "napi-postinstall": "lib/cli.js" + }, + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/napi-postinstall" + } + }, + "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/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-exports-info": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/node-exports-info/-/node-exports-info-1.6.0.tgz", + "integrity": "sha512-pyFS63ptit/P5WqUkt+UUfe+4oevH+bFeIiPPdfb0pFeYEu/1ELnJu5l+5EcTKYL5M7zaAa7S8ddywgXypqKCw==", + "dev": true, + "license": "MIT", + "dependencies": { + "array.prototype.flatmap": "^1.3.3", + "es-errors": "^1.3.0", + "object.entries": "^1.1.9", + "semver": "^6.3.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/node-exports-info/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/node-releases": { + "version": "2.0.46", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.46.tgz", + "integrity": "sha512-GYVXHE2KnrzAfsAjl4uP++evGFCrAU1jta4ubEjIG7YWt/64Gqv66a30yKwWczVjA6j3bM4nBwH7Pk1JmDHaxQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/nwsapi": { + "version": "2.2.23", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.23.tgz", + "integrity": "sha512-7wfH4sLbt4M0gCDzGE6vzQBo0bfTKjU7Sfpqy/7gs1qBfYz2vEJH6vXcBKpO3+6Yu1telwd0t9HpyOoLEQQbIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.7", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", + "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0", + "has-symbols": "^1.1.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.entries": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.9.tgz", + "integrity": "sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.4", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.fromentries": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", + "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.values": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.1.tgz", + "integrity": "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/orderedmap": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/orderedmap/-/orderedmap-2.1.1.tgz", + "integrity": "sha512-TvAWxi0nDe1j/rtMcWcIj94+Ffe6n7zhow33h40SKxmsmozs6dz/e+EajymfoFcHd7sxNn8yHM8839uixMOV6g==", + "license": "MIT" + }, + "node_modules/own-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", + "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "get-intrinsic": "^1.2.6", + "object-keys": "^1.1.1", + "safe-push-apply": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/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==", + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parse5": { + "version": "7.3.0", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.3.0.tgz", + "integrity": "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "entities": "^6.0.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "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-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/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "license": "MIT" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/pathe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", + "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==", + "dev": true, + "license": "MIT" + }, + "node_modules/pathval": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.1.tgz", + "integrity": "sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 14.16" + } + }, + "node_modules/pbf": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/pbf/-/pbf-4.0.2.tgz", + "integrity": "sha512-J0ajxARhZfpUEebxYs1vhMGMuLSXtBe1e+fFPDrf2uA2hgo+UshKfNUWOz92HJNz6/NFEXseQPddnHkTreWRqg==", + "license": "BSD-3-Clause", + "dependencies": { + "resolve-protobuf-schema": "^2.1.0" + }, + "bin": { + "pbf": "bin/pbf" + } + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.2.tgz", + "integrity": "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/possible-typed-array-names": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz", + "integrity": "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/postcss": { + "version": "8.5.15", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.15.tgz", + "integrity": "sha512-FfR8sjd4em2T6fb3I2MwAJU7HWVMr9zba+enmQeeWFfCbm+UOC/0X4DS8XtpUTMwWMGbjKYP7xjfNekzyGmB3A==", + "dev": true, + "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.12", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/potpack": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/potpack/-/potpack-2.1.0.tgz", + "integrity": "sha512-pcaShQc1Shq0y+E7GqJqvZj8DTthWV1KeHGdi0Z6IAin2Oi3JnLCOfwnCo84qc+HAp52wT9nK9H7FAJp5a44GQ==", + "license": "ISC" + }, + "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.8.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.8.3.tgz", + "integrity": "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw==", + "dev": true, + "license": "MIT", + "peer": true, + "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.1", + "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.1.tgz", + "integrity": "sha512-SxToR7P8Y2lWmv/kTzVLC1t/GDI2WGjMwNhLLE9qtH8Q13C+aEmuRlzDst4Up4s0Wc8sF2M+J57iB3cMLqftfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-diff": "^1.1.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/pretty-format/node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "license": "MIT" + }, + "node_modules/prop-types": { + "version": "15.8.1", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", + "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "license": "MIT", + "dependencies": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.13.1" + } + }, + "node_modules/prop-types/node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", + "license": "MIT" + }, + "node_modules/prosemirror-changeset": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/prosemirror-changeset/-/prosemirror-changeset-2.4.1.tgz", + "integrity": "sha512-96WBLhOaYhJ+kPhLg3uW359Tz6I/MfcrQfL4EGv4SrcqKEMC1gmoGrXHecPE8eOwTVCJ4IwgfzM8fFad25wNfw==", + "license": "MIT", + "dependencies": { + "prosemirror-transform": "^1.0.0" + } + }, + "node_modules/prosemirror-commands": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/prosemirror-commands/-/prosemirror-commands-1.7.1.tgz", + "integrity": "sha512-rT7qZnQtx5c0/y/KlYaGvtG411S97UaL6gdp6RIZ23DLHanMYLyfGBV5DtSnZdthQql7W+lEVbpSfwtO8T+L2w==", + "license": "MIT", + "dependencies": { + "prosemirror-model": "^1.0.0", + "prosemirror-state": "^1.0.0", + "prosemirror-transform": "^1.10.2" + } + }, + "node_modules/prosemirror-dropcursor": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/prosemirror-dropcursor/-/prosemirror-dropcursor-1.8.2.tgz", + "integrity": "sha512-CCk6Gyx9+Tt2sbYk5NK0nB1ukHi2ryaRgadV/LvyNuO3ena1payM2z6Cg0vO1ebK8cxbzo41ku2DE5Axj1Zuiw==", + "license": "MIT", + "dependencies": { + "prosemirror-state": "^1.0.0", + "prosemirror-transform": "^1.1.0", + "prosemirror-view": "^1.1.0" + } + }, + "node_modules/prosemirror-gapcursor": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/prosemirror-gapcursor/-/prosemirror-gapcursor-1.4.1.tgz", + "integrity": "sha512-pMdYaEnjNMSwl11yjEGtgTmLkR08m/Vl+Jj443167p9eB3HVQKhYCc4gmHVDsLPODfZfjr/MmirsdyZziXbQKw==", + "license": "MIT", + "dependencies": { + "prosemirror-keymap": "^1.0.0", + "prosemirror-model": "^1.0.0", + "prosemirror-state": "^1.0.0", + "prosemirror-view": "^1.0.0" + } + }, + "node_modules/prosemirror-history": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/prosemirror-history/-/prosemirror-history-1.5.0.tgz", + "integrity": "sha512-zlzTiH01eKA55UAf1MEjtssJeHnGxO0j4K4Dpx+gnmX9n+SHNlDqI2oO1Kv1iPN5B1dm5fsljCfqKF9nFL6HRg==", + "license": "MIT", + "dependencies": { + "prosemirror-state": "^1.2.2", + "prosemirror-transform": "^1.0.0", + "prosemirror-view": "^1.31.0", + "rope-sequence": "^1.3.0" + } + }, + "node_modules/prosemirror-keymap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/prosemirror-keymap/-/prosemirror-keymap-1.2.3.tgz", + "integrity": "sha512-4HucRlpiLd1IPQQXNqeo81BGtkY8Ai5smHhKW9jjPKRc2wQIxksg7Hl1tTI2IfT2B/LgX6bfYvXxEpJl7aKYKw==", + "license": "MIT", + "dependencies": { + "prosemirror-state": "^1.0.0", + "w3c-keyname": "^2.2.0" + } + }, + "node_modules/prosemirror-model": { + "version": "1.25.7", + "resolved": "https://registry.npmjs.org/prosemirror-model/-/prosemirror-model-1.25.7.tgz", + "integrity": "sha512-A79aN8QEFUwI6cax8Yq4Rpcx1TJZ3Kagn+ii7qLo4/V8H3mMiHrhFyhTyHHvpSnOgMPpWiDGSwM3etwrxE50ug==", + "license": "MIT", + "dependencies": { + "orderedmap": "^2.0.0" + } + }, + "node_modules/prosemirror-schema-list": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/prosemirror-schema-list/-/prosemirror-schema-list-1.5.1.tgz", + "integrity": "sha512-927lFx/uwyQaGwJxLWCZRkjXG0p48KpMj6ueoYiu4JX05GGuGcgzAy62dfiV8eFZftgyBUvLx76RsMe20fJl+Q==", + "license": "MIT", + "dependencies": { + "prosemirror-model": "^1.0.0", + "prosemirror-state": "^1.0.0", + "prosemirror-transform": "^1.7.3" + } + }, + "node_modules/prosemirror-state": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/prosemirror-state/-/prosemirror-state-1.4.4.tgz", + "integrity": "sha512-6jiYHH2CIGbCfnxdHbXZ12gySFY/fz/ulZE333G6bPqIZ4F+TXo9ifiR86nAHpWnfoNjOb3o5ESi7J8Uz1jXHw==", + "license": "MIT", + "dependencies": { + "prosemirror-model": "^1.0.0", + "prosemirror-transform": "^1.0.0", + "prosemirror-view": "^1.27.0" + } + }, + "node_modules/prosemirror-tables": { + "version": "1.8.5", + "resolved": "https://registry.npmjs.org/prosemirror-tables/-/prosemirror-tables-1.8.5.tgz", + "integrity": "sha512-V/0cDCsHKHe/tfWkeCmthNUcEp1IVO3p6vwN8XtwE9PZQLAZJigbw3QoraAdfJPir4NKJtNvOB8oYGKRl+t0Dw==", + "license": "MIT", + "dependencies": { + "prosemirror-keymap": "^1.2.3", + "prosemirror-model": "^1.25.4", + "prosemirror-state": "^1.4.4", + "prosemirror-transform": "^1.10.5", + "prosemirror-view": "^1.41.4" + } + }, + "node_modules/prosemirror-transform": { + "version": "1.12.0", + "resolved": "https://registry.npmjs.org/prosemirror-transform/-/prosemirror-transform-1.12.0.tgz", + "integrity": "sha512-GxboyN4AMIsoHNtz5uf2r2Ru551i5hWeCMD6E2Ib4Eogqoub0NflniaBPVQ4MrGE5yZ8JV9tUHg9qcZTTrcN4w==", + "license": "MIT", + "dependencies": { + "prosemirror-model": "^1.21.0" + } + }, + "node_modules/prosemirror-view": { + "version": "1.41.8", + "resolved": "https://registry.npmjs.org/prosemirror-view/-/prosemirror-view-1.41.8.tgz", + "integrity": "sha512-TnKDdohEatgyZNGCDWIdccOHXhYloJwbwU+phw/a23KBvJIR9lWQWW7WHHK3vBdOLDNuF7TaX98GObUZOWkOnA==", + "license": "MIT", + "dependencies": { + "prosemirror-model": "^1.20.0", + "prosemirror-state": "^1.0.0", + "prosemirror-transform": "^1.1.0" + } + }, + "node_modules/protocol-buffers-schema": { + "version": "3.6.1", + "resolved": "https://registry.npmjs.org/protocol-buffers-schema/-/protocol-buffers-schema-3.6.1.tgz", + "integrity": "sha512-VG2K63Igkiv9p76tk1lilczEK1cT+kCjKtkdhw1dQZV3k3IXJbd3o6Ho8b9zJZaHSnT2hKe4I+ObmX9w6m5SmQ==", + "license": "MIT" + }, + "node_modules/proxy-from-env": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-2.1.0.tgz", + "integrity": "sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==", + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "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/quickselect": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/quickselect/-/quickselect-3.0.0.tgz", + "integrity": "sha512-XdjUArbK4Bm5fLLvlm5KpTFOiOThgfWWI4axAZDWg4E/0mKdZyI9tNEfds27qCi1ze/vwTR16kvmmGhRra3c2g==", + "license": "ISC" + }, + "node_modules/react": { + "version": "19.2.6", + "resolved": "https://registry.npmjs.org/react/-/react-19.2.6.tgz", + "integrity": "sha512-sfWGGfavi0xr8Pg0sVsyHMAOziVYKgPLNrS7ig+ivMNb3wbCBw3KxtflsGBAwD3gYQlE/AEZsTLgToRrSCjb0Q==", + "license": "MIT", + "peer": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "19.2.6", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.6.tgz", + "integrity": "sha512-0prMI+hvBbPjsWnxDLxlCGyM8PN6UuWjEUCYmZhO67xIV9Xasa/r/vDnq+Xyq4Lo27g8QSbO5YzARu0D1Sps3g==", + "license": "MIT", + "peer": true, + "dependencies": { + "scheduler": "^0.27.0" + }, + "peerDependencies": { + "react": "^19.2.6" + } + }, + "node_modules/react-hook-form": { + "version": "7.76.1", + "resolved": "https://registry.npmjs.org/react-hook-form/-/react-hook-form-7.76.1.tgz", + "integrity": "sha512-rYM7tPiWlu3nZchkR/ex7piyzui2vFPyaLnXnI/RnblB/L4qfMmyses8llJVtF1NpE9WBBsJlGtcSZzPCXW1qQ==", + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/react-hook-form" + }, + "peerDependencies": { + "react": "^16.8.0 || ^17 || ^18 || ^19" + } + }, + "node_modules/react-is": { + "version": "19.2.6", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-19.2.6.tgz", + "integrity": "sha512-XjBR15BhXuylgWGuslhDKqlSayuqvqBX91BP8pauG8kd1zY8kotkNWbXksTCNRarse4kuGbe2kIY05ARtwNIvw==", + "license": "MIT" + }, + "node_modules/react-map-gl": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/react-map-gl/-/react-map-gl-8.1.1.tgz", + "integrity": "sha512-aSqFAFoxvY7wxbGI93Dz0E41171mkAb3GcNbnkFIotmu88OFw495os6mIDZSi7irYNT/PZEIOEHUxhun4ToGuQ==", + "license": "MIT", + "dependencies": { + "@vis.gl/react-mapbox": "8.1.1", + "@vis.gl/react-maplibre": "8.1.1" + }, + "peerDependencies": { + "mapbox-gl": ">=1.13.0", + "maplibre-gl": ">=1.13.0", + "react": ">=16.3.0", + "react-dom": ">=16.3.0" + }, + "peerDependenciesMeta": { + "mapbox-gl": { + "optional": true + }, + "maplibre-gl": { + "optional": true + } + } + }, + "node_modules/react-router": { + "version": "7.15.1", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.15.1.tgz", + "integrity": "sha512-R8rl9HhgikFYoPJymnUtPXWbnDb3oget6lQnfIoupbt61aT9aOhRkDsY2XRhZRyX1Z/8a5sL74fXmFNm3NRK5A==", + "license": "MIT", + "dependencies": { + "cookie": "^1.0.1", + "set-cookie-parser": "^2.6.0" + }, + "engines": { + "node": ">=20.0.0" + }, + "peerDependencies": { + "react": ">=18", + "react-dom": ">=18" + }, + "peerDependenciesMeta": { + "react-dom": { + "optional": true + } + } + }, + "node_modules/react-router-dom": { + "version": "7.15.1", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-7.15.1.tgz", + "integrity": "sha512-AzF62gjY6U9rkMq4RfP/r2EVtQ7DMfNMjyOp/flLTCrtRylLiK4wT4pSq6O8rOXZ2eXdZYJPEYe+ifomiv+Igg==", + "license": "MIT", + "dependencies": { + "react-router": "7.15.1" + }, + "engines": { + "node": ">=20.0.0" + }, + "peerDependencies": { + "react": ">=18", + "react-dom": ">=18" + } + }, + "node_modules/react-transition-group": { + "version": "4.4.5", + "resolved": "https://registry.npmjs.org/react-transition-group/-/react-transition-group-4.4.5.tgz", + "integrity": "sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==", + "license": "BSD-3-Clause", + "dependencies": { + "@babel/runtime": "^7.5.5", + "dom-helpers": "^5.0.1", + "loose-envify": "^1.4.0", + "prop-types": "^15.6.2" + }, + "peerDependencies": { + "react": ">=16.6.0", + "react-dom": ">=16.6.0" + } + }, + "node_modules/redent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/redent/-/redent-3.0.0.tgz", + "integrity": "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==", + "dev": true, + "license": "MIT", + "dependencies": { + "indent-string": "^4.0.0", + "strip-indent": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/reflect.getprototypeof": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", + "integrity": "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.9", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.7", + "get-proto": "^1.0.1", + "which-builtin-type": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regexp.prototype.flags": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", + "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "set-function-name": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve": { + "version": "1.22.12", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.12.tgz", + "integrity": "sha512-TyeJ1zif53BPfHootBGwPRYT1RUt6oGWsaQr8UyZW/eAm9bKoijtvruSDEmZHm92CwS9nj7/fWttqPCgzep8CA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-pkg-maps": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", + "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" + } + }, + "node_modules/resolve-protobuf-schema": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/resolve-protobuf-schema/-/resolve-protobuf-schema-2.1.0.tgz", + "integrity": "sha512-kI5ffTiZWmJaS/huM8wZfEMer1eRd7oJQhDuxeCLe3t7N7mX3z94CN0xPxBQxFYQTSNz9T0i+v6inKqSdK8xrQ==", + "license": "MIT", + "dependencies": { + "protocol-buffers-schema": "^3.3.1" + } + }, + "node_modules/rolldown": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.2.tgz", + "integrity": "sha512-oZx5zVDtVB44AW3eaifgDml1gWRDZGvjcfdxonE4swNPG98PrrXjaO/KrnUjzlMnztCCRVlUueA1kCXhARGk6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@oxc-project/types": "=0.132.0", + "@rolldown/pluginutils": "^1.0.0" + }, + "bin": { + "rolldown": "bin/cli.mjs" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "optionalDependencies": { + "@rolldown/binding-android-arm64": "1.0.2", + "@rolldown/binding-darwin-arm64": "1.0.2", + "@rolldown/binding-darwin-x64": "1.0.2", + "@rolldown/binding-freebsd-x64": "1.0.2", + "@rolldown/binding-linux-arm-gnueabihf": "1.0.2", + "@rolldown/binding-linux-arm64-gnu": "1.0.2", + "@rolldown/binding-linux-arm64-musl": "1.0.2", + "@rolldown/binding-linux-ppc64-gnu": "1.0.2", + "@rolldown/binding-linux-s390x-gnu": "1.0.2", + "@rolldown/binding-linux-x64-gnu": "1.0.2", + "@rolldown/binding-linux-x64-musl": "1.0.2", + "@rolldown/binding-openharmony-arm64": "1.0.2", + "@rolldown/binding-wasm32-wasi": "1.0.2", + "@rolldown/binding-win32-arm64-msvc": "1.0.2", + "@rolldown/binding-win32-x64-msvc": "1.0.2" + } + }, + "node_modules/rolldown/node_modules/@rolldown/binding-win32-x64-msvc": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.2.tgz", + "integrity": "sha512-v7qRI7gXLRINcOGXt+7YmAZ6iFuyZVMIoXAxhd8oP+DR9dLfL9GfNIx7PLMxmhZdvq8waUJBQiWN9EKNy+TRBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/rollup": { + "version": "4.60.4", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.4.tgz", + "integrity": "sha512-WHeFSbZYsPu3+bLoNRUuAO+wavNlocOPf3wSHTP7hcFKVnJeWsYlCDbr3mTS14FCizf9ccIxXA8sGL8zKeQN3g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.8" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.60.4", + "@rollup/rollup-android-arm64": "4.60.4", + "@rollup/rollup-darwin-arm64": "4.60.4", + "@rollup/rollup-darwin-x64": "4.60.4", + "@rollup/rollup-freebsd-arm64": "4.60.4", + "@rollup/rollup-freebsd-x64": "4.60.4", + "@rollup/rollup-linux-arm-gnueabihf": "4.60.4", + "@rollup/rollup-linux-arm-musleabihf": "4.60.4", + "@rollup/rollup-linux-arm64-gnu": "4.60.4", + "@rollup/rollup-linux-arm64-musl": "4.60.4", + "@rollup/rollup-linux-loong64-gnu": "4.60.4", + "@rollup/rollup-linux-loong64-musl": "4.60.4", + "@rollup/rollup-linux-ppc64-gnu": "4.60.4", + "@rollup/rollup-linux-ppc64-musl": "4.60.4", + "@rollup/rollup-linux-riscv64-gnu": "4.60.4", + "@rollup/rollup-linux-riscv64-musl": "4.60.4", + "@rollup/rollup-linux-s390x-gnu": "4.60.4", + "@rollup/rollup-linux-x64-gnu": "4.60.4", + "@rollup/rollup-linux-x64-musl": "4.60.4", + "@rollup/rollup-openbsd-x64": "4.60.4", + "@rollup/rollup-openharmony-arm64": "4.60.4", + "@rollup/rollup-win32-arm64-msvc": "4.60.4", + "@rollup/rollup-win32-ia32-msvc": "4.60.4", + "@rollup/rollup-win32-x64-gnu": "4.60.4", + "@rollup/rollup-win32-x64-msvc": "4.60.4", + "fsevents": "~2.3.2" + } + }, + "node_modules/rollup/node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" + }, + "node_modules/rope-sequence": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/rope-sequence/-/rope-sequence-1.3.4.tgz", + "integrity": "sha512-UT5EDe2cu2E/6O4igUr5PSFs23nvvukicWHx6GnOPlHAiiYbzNuCRQCuiUdHJQcqKalLKlrYJnjY0ySGsXNQXQ==", + "license": "MIT" + }, + "node_modules/rrweb-cssom": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/rrweb-cssom/-/rrweb-cssom-0.8.0.tgz", + "integrity": "sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw==", + "dev": true, + "license": "MIT" + }, + "node_modules/rw": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz", + "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==", + "license": "BSD-3-Clause" + }, + "node_modules/safe-array-concat": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.4.tgz", + "integrity": "sha512-wtZlHyOje6OZTGqAoaDKxFkgRtkF9CnHAVnCHKfuj200wAgL+bSJhdsCD2l0Qx/2ekEXjPWcyKkfGb5CPboslg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.9", + "call-bound": "^1.0.4", + "get-intrinsic": "^1.3.0", + "has-symbols": "^1.1.0", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">=0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-push-apply": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", + "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-regex-test": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", + "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "is-regex": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true, + "license": "MIT" + }, + "node_modules/saxes": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-6.0.0.tgz", + "integrity": "sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==", + "dev": true, + "license": "ISC", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=v12.22.7" + } + }, + "node_modules/scheduler": { + "version": "0.27.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz", + "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==", + "license": "MIT" + }, + "node_modules/semver": { + "version": "7.8.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.1.tgz", + "integrity": "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/set-cookie-parser": { + "version": "2.7.2", + "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.2.tgz", + "integrity": "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw==", + "license": "MIT" + }, + "node_modules/set-function-length": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-proto": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/set-proto/-/set-proto-1.0.0.tgz", + "integrity": "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==", + "dev": true, + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-value": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", + "license": "MIT", + "dependencies": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.3", + "split-string": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.1.tgz", + "integrity": "sha512-mjn/0bi/oUURjc5Xl7IaWi/OJJJumuoJFQJfDDyO46+hBWsfaVM65TBHq2eoZBhzl9EchxOijpkbRC8SVBQU0w==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/siginfo": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", + "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", + "dev": true, + "license": "ISC" + }, + "node_modules/sort-asc": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/sort-asc/-/sort-asc-0.2.0.tgz", + "integrity": "sha512-umMGhjPeHAI6YjABoSTrFp2zaBtXBej1a0yKkuMUyjjqu6FJsTF+JYwCswWDg+zJfk/5npWUUbd33HH/WLzpaA==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sort-desc": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/sort-desc/-/sort-desc-0.2.0.tgz", + "integrity": "sha512-NqZqyvL4VPW+RAxxXnB8gvE1kyikh8+pR+T+CXLksVRN9eiQqkQlPwqWYU0mF9Jm7UnctShlxLyAt1CaBOTL1w==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sort-object": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/sort-object/-/sort-object-3.0.3.tgz", + "integrity": "sha512-nK7WOY8jik6zaG9CRwZTaD5O7ETWDLZYMM12pqY8htll+7dYeqGfEUPcUBHOpSJg2vJOrvFIY2Dl5cX2ih1hAQ==", + "license": "MIT", + "dependencies": { + "bytewise": "^1.1.0", + "get-value": "^2.0.2", + "is-extendable": "^0.1.1", + "sort-asc": "^0.2.0", + "sort-desc": "^0.2.0", + "union-value": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==", + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "license": "MIT", + "dependencies": { + "extend-shallow": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/split-string/node_modules/extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha512-BwY5b5Ql4+qZoefgMj2NUmx+tehVTH/Kf4k1ZEtOHNFcm2wSxMRo992l6X3TIgni2eZVTZ85xMOjF31fwZAj6Q==", + "license": "MIT", + "dependencies": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/split-string/node_modules/is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "license": "MIT", + "dependencies": { + "is-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/stable-hash-x": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/stable-hash-x/-/stable-hash-x-0.2.0.tgz", + "integrity": "sha512-o3yWv49B/o4QZk5ZcsALc6t0+eCelPc44zZsLtCQnZPDwFpDYSWcDnrv2TtMmMbQ7uKo3J0HTURCqckw23czNQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/stackback": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", + "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", + "dev": true, + "license": "MIT" + }, + "node_modules/std-env": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.10.0.tgz", + "integrity": "sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==", + "dev": true, + "license": "MIT" + }, + "node_modules/stop-iteration-iterator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.1.0.tgz", + "integrity": "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "internal-slot": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/string.prototype.includes": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/string.prototype.includes/-/string.prototype.includes-2.0.1.tgz", + "integrity": "sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.3" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/string.prototype.matchall": { + "version": "4.0.12", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.12.tgz", + "integrity": "sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.3", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.6", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.6", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "internal-slot": "^1.1.0", + "regexp.prototype.flags": "^1.5.3", + "set-function-name": "^2.0.2", + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.repeat": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/string.prototype.repeat/-/string.prototype.repeat-1.0.0.tgz", + "integrity": "sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==", + "dev": true, + "license": "MIT", + "dependencies": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.5" + } + }, + "node_modules/string.prototype.trim": { + "version": "1.2.10", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", + "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.2", + "define-data-property": "^1.1.4", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.5", + "es-object-atoms": "^1.0.0", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimend": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", + "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "call-bound": "^1.0.2", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimstart": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", + "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/strip-indent": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-3.0.0.tgz", + "integrity": "sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "min-indent": "^1.0.0" + }, + "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/strip-literal": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/strip-literal/-/strip-literal-3.1.0.tgz", + "integrity": "sha512-8r3mkIM/2+PpjHoOtiAW8Rg3jJLHaV7xPwG+YRGrv6FP0wwk/toTpATxWYOW0BKdWwl82VT2tFYi5DlROa0Mxg==", + "dev": true, + "license": "MIT", + "dependencies": { + "js-tokens": "^9.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/antfu" + } + }, + "node_modules/strip-literal/node_modules/js-tokens": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", + "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/stylis": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.2.0.tgz", + "integrity": "sha512-Orov6g6BB1sDfYgzWfTHDOxamtX1bE/zo104Dh9e6fqJ3PooipYyfJ0pUmrZO2wAvO8YbEyeFrkV91XTsGMSrw==", + "license": "MIT" + }, + "node_modules/supercluster": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/supercluster/-/supercluster-8.0.1.tgz", + "integrity": "sha512-IiOea5kJ9iqzD2t7QJq/cREyLHTtSmUT6gQsweojg9WH2sYJqZK9SswTu6jrscO6D1G5v5vYZ9ru/eq85lXeZQ==", + "license": "ISC", + "dependencies": { + "kdbush": "^4.0.2" + } + }, + "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/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/swiper": { + "version": "12.2.0", + "resolved": "https://registry.npmjs.org/swiper/-/swiper-12.2.0.tgz", + "integrity": "sha512-K8uXsBZU6ME97Ia3xbBge8IRCnR1lOmIILzvY/jGVic7dSTQ530s3uO8RvXbPUtkkXLWIwmZLRPbtDxRWVAFdg==", + "funding": [ + { + "type": "custom", + "url": "https://sponsors.nolimits4web.com" + }, + { + "type": "github", + "url": "https://github.com/sponsors/nolimits4web" + } + ], + "license": "MIT", + "engines": { + "node": ">= 4.7.0" + } + }, + "node_modules/symbol-tree": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", + "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", + "dev": true, + "license": "MIT" + }, + "node_modules/synckit": { + "version": "0.11.12", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.12.tgz", + "integrity": "sha512-Bh7QjT8/SuKUIfObSXNHNSK6WHo6J1tHCqJsuaFDP7gP0fkzSfTxI8y85JrppZ0h8l0maIgc2tfuZQ6/t3GtnQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@pkgr/core": "^0.2.9" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/synckit" + } + }, + "node_modules/tinybench": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", + "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyexec": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", + "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyglobby": { + "version": "0.2.16", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz", + "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/tinyglobby/node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/tinyglobby/node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/tinypool": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.1.1.tgz", + "integrity": "sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.0.0 || >=20.0.0" + } + }, + "node_modules/tinyqueue": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/tinyqueue/-/tinyqueue-3.0.0.tgz", + "integrity": "sha512-gRa9gwYU3ECmQYv3lslts5hxuIa90veaEcxDYuu3QGOIAEM2mOZkVHp48ANJuu1CURtRdHKUBY5Lm1tHV+sD4g==", + "license": "ISC" + }, + "node_modules/tinyrainbow": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-2.0.0.tgz", + "integrity": "sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tinyspy": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-4.0.4.tgz", + "integrity": "sha512-azl+t0z7pw/z958Gy9svOTuzqIk6xq+NSheJzn5MMWtWTFywIacg2wUlzKFGtt3cthx0r2SxMK0yzJOR0IES7Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tldts": { + "version": "6.1.86", + "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.86.tgz", + "integrity": "sha512-WMi/OQ2axVTf/ykqCQgXiIct+mSQDFdH2fkwhPwgEwvJ1kSzZRiinb0zF2Xb8u4+OqPChmyI6MEu4EezNJz+FQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "tldts-core": "^6.1.86" + }, + "bin": { + "tldts": "bin/cli.js" + } + }, + "node_modules/tldts-core": { + "version": "6.1.86", + "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-6.1.86.tgz", + "integrity": "sha512-Je6p7pkk+KMzMv2XXKmAE3McmolOQFdxkKw0R8EYNr7sELW46JqnNeTX8ybPiQgvg1ymCoF8LXs5fzFaZvJPTA==", + "dev": true, + "license": "MIT" + }, + "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/tough-cookie": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-5.1.2.tgz", + "integrity": "sha512-FVDYdxtnj0G6Qm/DhNPSb8Ju59ULcup3tuJxkFb5K8Bv2pUXILbf0xZWU8PX8Ov19OXljbUyveOFwRMwkXzO+A==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tldts": "^6.1.32" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/tr46": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-5.1.1.tgz", + "integrity": "sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw==", + "dev": true, + "license": "MIT", + "dependencies": { + "punycode": "^2.3.1" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/ts-api-utils": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", + "integrity": "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" + } + }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, + "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/typed-array-buffer": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", + "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.14" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/typed-array-byte-length": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", + "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8", + "for-each": "^0.3.3", + "gopd": "^1.2.0", + "has-proto": "^1.2.0", + "is-typed-array": "^1.1.14" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-byte-offset": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", + "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.8", + "for-each": "^0.3.3", + "gopd": "^1.2.0", + "has-proto": "^1.2.0", + "is-typed-array": "^1.1.15", + "reflect.getprototypeof": "^1.0.9" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-length": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", + "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0", + "reflect.getprototypeof": "^1.0.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typescript": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-6.0.3.tgz", + "integrity": "sha512-y2TvuxSZPDyQakkFRPZHKFm+KKVqIisdg9/CZwm9ftvKXLP8NRWj38/ODjNbr43SsoXqNuAisEf1GdCxqWcdBw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/typescript-eslint": { + "version": "8.60.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.60.0.tgz", + "integrity": "sha512-9f65qWLZdAW9m1JaxBDUHcqRUfL8bkxxXL7XxEfI+F09q56PkBvIfCjLF3yInsDM/BBmwkqmCQdCZe/RYlIWEw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.60.0", + "@typescript-eslint/parser": "8.60.0", + "@typescript-eslint/typescript-estree": "8.60.0", + "@typescript-eslint/utils": "8.60.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/typewise": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/typewise/-/typewise-1.0.3.tgz", + "integrity": "sha512-aXofE06xGhaQSPzt8hlTY+/YWQhm9P0jYUp1f2XtmW/3Bk0qzXcyFWAtPoo2uTGQj1ZwbDuSyuxicq+aDo8lCQ==", + "license": "MIT", + "dependencies": { + "typewise-core": "^1.2.0" + } + }, + "node_modules/typewise-core": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/typewise-core/-/typewise-core-1.2.0.tgz", + "integrity": "sha512-2SCC/WLzj2SbUwzFOzqMCkz5amXLlxtJqDKTICqg30x+2DZxcfZN2MvQZmGfXWKNWaKK9pBPsvkcwv8bF/gxKg==", + "license": "MIT" + }, + "node_modules/uglify-js": { + "version": "3.19.3", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.19.3.tgz", + "integrity": "sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==", + "dev": true, + "license": "BSD-2-Clause", + "optional": true, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/unbox-primitive": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", + "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.3", + "has-bigints": "^1.0.2", + "has-symbols": "^1.1.0", + "which-boxed-primitive": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/union-value": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", + "license": "MIT", + "dependencies": { + "arr-union": "^3.1.0", + "get-value": "^2.0.6", + "is-extendable": "^0.1.1", + "set-value": "^2.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unrs-resolver": { + "version": "1.12.2", + "resolved": "https://registry.npmjs.org/unrs-resolver/-/unrs-resolver-1.12.2.tgz", + "integrity": "sha512-dmlRxBJJayXjqTwC+JtF1HhJmgf3ftQ3YejFcZrf4+KKtJv0qDsK1pjqaaVjG7wJ5NJ6UVP1OqRMQ71Z4C3rxQ==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "dependencies": { + "napi-postinstall": "^0.3.4" + }, + "funding": { + "url": "https://opencollective.com/unrs-resolver" + }, + "optionalDependencies": { + "@unrs/resolver-binding-android-arm-eabi": "1.12.2", + "@unrs/resolver-binding-android-arm64": "1.12.2", + "@unrs/resolver-binding-darwin-arm64": "1.12.2", + "@unrs/resolver-binding-darwin-x64": "1.12.2", + "@unrs/resolver-binding-freebsd-x64": "1.12.2", + "@unrs/resolver-binding-linux-arm-gnueabihf": "1.12.2", + "@unrs/resolver-binding-linux-arm-musleabihf": "1.12.2", + "@unrs/resolver-binding-linux-arm64-gnu": "1.12.2", + "@unrs/resolver-binding-linux-arm64-musl": "1.12.2", + "@unrs/resolver-binding-linux-loong64-gnu": "1.12.2", + "@unrs/resolver-binding-linux-loong64-musl": "1.12.2", + "@unrs/resolver-binding-linux-ppc64-gnu": "1.12.2", + "@unrs/resolver-binding-linux-riscv64-gnu": "1.12.2", + "@unrs/resolver-binding-linux-riscv64-musl": "1.12.2", + "@unrs/resolver-binding-linux-s390x-gnu": "1.12.2", + "@unrs/resolver-binding-linux-x64-gnu": "1.12.2", + "@unrs/resolver-binding-linux-x64-musl": "1.12.2", + "@unrs/resolver-binding-openharmony-arm64": "1.12.2", + "@unrs/resolver-binding-wasm32-wasi": "1.12.2", + "@unrs/resolver-binding-win32-arm64-msvc": "1.12.2", + "@unrs/resolver-binding-win32-ia32-msvc": "1.12.2", + "@unrs/resolver-binding-win32-x64-msvc": "1.12.2" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "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/use-sync-external-store": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz", + "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==", + "license": "MIT", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" + } + }, + "node_modules/vite": { + "version": "8.0.14", + "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.14.tgz", + "integrity": "sha512-s4BJJ+5y1pYL6Otw51FHhVJQhPnuRinKig64g/1+EUNaJsd3gCKdD31IPFvswUgW9/60QT9oFHbZHbQK5imcxw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "lightningcss": "^1.32.0", + "picomatch": "^4.0.4", + "postcss": "^8.5.15", + "rolldown": "1.0.2", + "tinyglobby": "^0.2.16" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^20.19.0 || >=22.12.0", + "@vitejs/devtools": "^0.1.18", + "esbuild": "^0.27.0 || ^0.28.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "@vitejs/devtools": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/vite-node": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-3.2.4.tgz", + "integrity": "sha512-EbKSKh+bh1E1IFxeO0pg1n4dvoOTt0UDiXMd/qn++r98+jPO1xtJilvXldeuQ8giIB5IkpjCgMleHMNEsGH6pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cac": "^6.7.14", + "debug": "^4.4.1", + "es-module-lexer": "^1.7.0", + "pathe": "^2.0.3", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" + }, + "bin": { + "vite-node": "vite-node.mjs" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/vite-node/node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/vite-node/node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/vite-node/node_modules/vite": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/vite/-/vite-7.3.3.tgz", + "integrity": "sha512-/4XH147Ui7OGTjg3HbdWe5arnZQSbfuRzdr9Ec7TQi5I7R+ir0Rlc9GIvD4v0XZurELqA035KVXJXpR61xhiTA==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.27.0", + "fdir": "^6.5.0", + "picomatch": "^4.0.3", + "postcss": "^8.5.6", + "rollup": "^4.43.0", + "tinyglobby": "^0.2.15" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^20.19.0 || >=22.12.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "lightningcss": "^1.21.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/vite/node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/vitest": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-3.2.4.tgz", + "integrity": "sha512-LUCP5ev3GURDysTWiP47wRRUpLKMOfPh+yKTx3kVIEiu5KOMeqzpnYNsKyOoVrULivR8tLcks4+lga33Whn90A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/chai": "^5.2.2", + "@vitest/expect": "3.2.4", + "@vitest/mocker": "3.2.4", + "@vitest/pretty-format": "^3.2.4", + "@vitest/runner": "3.2.4", + "@vitest/snapshot": "3.2.4", + "@vitest/spy": "3.2.4", + "@vitest/utils": "3.2.4", + "chai": "^5.2.0", + "debug": "^4.4.1", + "expect-type": "^1.2.1", + "magic-string": "^0.30.17", + "pathe": "^2.0.3", + "picomatch": "^4.0.2", + "std-env": "^3.9.0", + "tinybench": "^2.9.0", + "tinyexec": "^0.3.2", + "tinyglobby": "^0.2.14", + "tinypool": "^1.1.1", + "tinyrainbow": "^2.0.0", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0", + "vite-node": "3.2.4", + "why-is-node-running": "^2.3.0" + }, + "bin": { + "vitest": "vitest.mjs" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@edge-runtime/vm": "*", + "@types/debug": "^4.1.12", + "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", + "@vitest/browser": "3.2.4", + "@vitest/ui": "3.2.4", + "happy-dom": "*", + "jsdom": "*" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@types/debug": { + "optional": true + }, + "@types/node": { + "optional": true + }, + "@vitest/browser": { + "optional": true + }, + "@vitest/ui": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + } + } + }, + "node_modules/vitest/node_modules/@vitest/mocker": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-3.2.4.tgz", + "integrity": "sha512-46ryTE9RZO/rfDd7pEqFl7etuyzekzEhUbTW3BvmeO/BcCMEgq59BKhek3dXDWgAj4oMK6OZi+vRr1wPW6qjEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "3.2.4", + "estree-walker": "^3.0.3", + "magic-string": "^0.30.17" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "msw": "^2.4.9", + "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true + } + } + }, + "node_modules/vitest/node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/vitest/node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/vitest/node_modules/vite": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/vite/-/vite-7.3.3.tgz", + "integrity": "sha512-/4XH147Ui7OGTjg3HbdWe5arnZQSbfuRzdr9Ec7TQi5I7R+ir0Rlc9GIvD4v0XZurELqA035KVXJXpR61xhiTA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "esbuild": "^0.27.0", + "fdir": "^6.5.0", + "picomatch": "^4.0.3", + "postcss": "^8.5.6", + "rollup": "^4.43.0", + "tinyglobby": "^0.2.15" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^20.19.0 || >=22.12.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "lightningcss": "^1.21.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/w3c-keyname": { + "version": "2.2.8", + "resolved": "https://registry.npmjs.org/w3c-keyname/-/w3c-keyname-2.2.8.tgz", + "integrity": "sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==", + "license": "MIT" + }, + "node_modules/w3c-xmlserializer": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-5.0.0.tgz", + "integrity": "sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/webidl-conversions": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", + "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + } + }, + "node_modules/whatwg-encoding": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz", + "integrity": "sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==", + "deprecated": "Use @exodus/bytes instead for a more spec-conformant and faster implementation", + "dev": true, + "license": "MIT", + "dependencies": { + "iconv-lite": "0.6.3" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/whatwg-mimetype": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", + "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/whatwg-url": { + "version": "14.2.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.2.0.tgz", + "integrity": "sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tr46": "^5.1.0", + "webidl-conversions": "^7.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "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-boxed-primitive": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", + "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-bigint": "^1.1.0", + "is-boolean-object": "^1.2.1", + "is-number-object": "^1.1.1", + "is-string": "^1.1.1", + "is-symbol": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-builtin-type": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", + "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "function.prototype.name": "^1.1.6", + "has-tostringtag": "^1.0.2", + "is-async-function": "^2.0.0", + "is-date-object": "^1.1.0", + "is-finalizationregistry": "^1.1.0", + "is-generator-function": "^1.0.10", + "is-regex": "^1.2.1", + "is-weakref": "^1.0.2", + "isarray": "^2.0.5", + "which-boxed-primitive": "^1.1.0", + "which-collection": "^1.0.2", + "which-typed-array": "^1.1.16" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-collection": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", + "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-map": "^2.0.3", + "is-set": "^2.0.3", + "is-weakmap": "^2.0.2", + "is-weakset": "^2.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.21", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.21.tgz", + "integrity": "sha512-zbRA8cVm6io/d5W8uIe2hblzN76/Wm3v/yiythQvr+dpBWeqhPSWIDNj4zOyHi4zKbMK6DN34Xsr9jPHJERAEw==", + "dev": true, + "license": "MIT", + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.9", + "call-bound": "^1.0.4", + "for-each": "^0.3.5", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/why-is-node-running": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", + "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", + "dev": true, + "license": "MIT", + "dependencies": { + "siginfo": "^2.0.0", + "stackback": "0.0.2" + }, + "bin": { + "why-is-node-running": "cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "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/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/ws": { + "version": "8.21.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.21.0.tgz", + "integrity": "sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xml-name-validator": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-5.0.0.tgz", + "integrity": "sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18" + } + }, + "node_modules/xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", + "dev": true, + "license": "MIT" + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "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": "4.4.3", + "resolved": "https://registry.npmjs.org/zod/-/zod-4.4.3.tgz", + "integrity": "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ==", + "dev": true, + "license": "MIT", + "peer": true, + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + }, + "node_modules/zod-validation-error": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz", + "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "zod": "^3.25.0 || ^4.0.0" + } + } + } +} diff --git a/client/package.json b/client/package.json new file mode 100755 index 0000000..d031569 --- /dev/null +++ b/client/package.json @@ -0,0 +1,82 @@ +{ + "name": "client", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "tsc -b && vite build", + "lint": "eslint .", + "preview": "vite preview", + "lint:fix": "eslint . --fix", + "format": "prettier . --write --ignore-unknown", + "format:check": "prettier . --check --ignore-unknown", + "test": "vitest run", + "test:watch": "vitest" + }, + "dependencies": { + "@dicebear/adventurer": "^9.4.2", + "@dicebear/avataaars": "^9.4.2", + "@dicebear/big-ears": "^9.4.2", + "@dicebear/big-smile": "^9.4.2", + "@dicebear/bottts": "^9.4.2", + "@dicebear/core": "^9.4.2", + "@dicebear/croodles": "^9.4.2", + "@dicebear/fun-emoji": "^9.4.2", + "@dicebear/identicon": "^9.4.2", + "@dicebear/initials": "^9.4.2", + "@dicebear/lorelei": "^9.4.2", + "@dicebear/micah": "^9.4.2", + "@dicebear/notionists": "^9.4.2", + "@dicebear/pixel-art": "^9.4.2", + "@dicebear/rings": "^9.4.2", + "@dicebear/shapes": "^9.4.2", + "@dicebear/thumbs": "^9.4.2", + "@emotion/react": "^11.14.0", + "@emotion/styled": "^11.14.1", + "@mui/icons-material": "^9.0.0", + "@mui/material": "^9.0.0", + "@tanstack/react-query": "^5.100.5", + "@tiptap/extension-placeholder": "^3.22.5", + "@tiptap/react": "^3.22.5", + "@tiptap/starter-kit": "^3.22.5", + "axios": "^1.15.2", + "effector": "^23.4.4", + "effector-react": "^23.3.0", + "lucide-react": "^1.14.0", + "maplibre-gl": "^5.24.0", + "react": "^19.2.5", + "react-dom": "^19.2.5", + "react-hook-form": "^7.74.0", + "react-map-gl": "^8.1.1", + "react-router-dom": "^7.14.2", + "swiper": "^12.1.3" + }, + "devDependencies": { + "@eslint/js": "^9.39.4", + "@testing-library/jest-dom": "^6.9.1", + "@testing-library/react": "^16.3.2", + "@testing-library/user-event": "^14.6.1", + "@types/node": "^24.12.2", + "@types/react": "^19.2.14", + "@types/react-dom": "^19.2.3", + "@vitejs/plugin-react": "^6.0.1", + "eslint": "^9.39.4", + "eslint-config-prettier": "^10.1.8", + "eslint-import-resolver-typescript": "^4.4.4", + "eslint-plugin-boundaries": "^6.0.2", + "eslint-plugin-import-x": "^4.16.2", + "eslint-plugin-jsx-a11y": "^6.10.2", + "eslint-plugin-prettier": "^5.5.5", + "eslint-plugin-react": "^7.37.5", + "eslint-plugin-react-hooks": "^7.1.1", + "eslint-plugin-react-refresh": "^0.5.2", + "globals": "^17.5.0", + "jsdom": "^26.1.0", + "prettier": "^3.8.3", + "typescript": "~6.0.2", + "typescript-eslint": "^8.58.2", + "vite": "^8.0.10", + "vitest": "^3.2.4" + } +} diff --git a/client/public/apple-touch-icon.png b/client/public/apple-touch-icon.png new file mode 100755 index 0000000000000000000000000000000000000000..6eff9ad5e7a3ed3dfc138350583ae8693870c0d5 GIT binary patch literal 4629 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 LNkvXXu0mjf{@LxQ literal 0 HcmV?d00001 diff --git a/client/public/favicon-128.png b/client/public/favicon-128.png new file mode 100755 index 0000000000000000000000000000000000000000..b50150b914f2625f1316eb2a0de3eed86ae59acf GIT binary patch literal 2796 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 Eg72?t#sB~S literal 0 HcmV?d00001 diff --git a/client/public/favicon.svg b/client/public/favicon.svg new file mode 100755 index 0000000..5839c46 --- /dev/null +++ b/client/public/favicon.svg @@ -0,0 +1 @@ +К \ No newline at end of file diff --git a/client/public/fonts/Outfit-Bold.woff2 b/client/public/fonts/Outfit-Bold.woff2 new file mode 100755 index 0000000000000000000000000000000000000000..8674e73594a26b7670d3ba93e42063f39e11c3ee GIT binary patch literal 14060 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 literal 0 HcmV?d00001 diff --git a/client/public/fonts/Outfit-Medium.woff2 b/client/public/fonts/Outfit-Medium.woff2 new file mode 100755 index 0000000000000000000000000000000000000000..264ce74280afa9bce3b16dbdd77a42754ba52986 GIT binary patch literal 13528 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 OeG&HNd{LtS0sap##2Lu| literal 0 HcmV?d00001 diff --git a/client/public/icons.svg b/client/public/icons.svg new file mode 100755 index 0000000..e952219 --- /dev/null +++ b/client/public/icons.svg @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/client/public/logo.webp b/client/public/logo.webp new file mode 100755 index 0000000000000000000000000000000000000000..e13fb532e766e4cbc6fb056630aaf479511e4b6e GIT binary patch literal 23522 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 qQ9R-Km0|w*AMpNM + + + https://любимыйкреатив.рф/ + 1.0 + daily + + + https://любимыйкреатив.рф/info + 0.8 + monthly + + + https://любимыйкреатив.рф/about + 0.7 + monthly + + + https://любимыйкреатив.рф/privacy + 0.5 + yearly + + + https://любимыйкреатив.рф/terms + 0.5 + yearly + + diff --git a/client/src/app/App.tsx b/client/src/app/App.tsx new file mode 100755 index 0000000..65634bb --- /dev/null +++ b/client/src/app/App.tsx @@ -0,0 +1,22 @@ +import { BrowserRouter } from 'react-router-dom' +import { AppProviders } from '@/app/providers/AppProviders' +import { AppRoutes } from '@/app/routes' +import { NotificationStack } from '@/shared/ui/NotificationStack' +import { ErrorBoundary } from '@/shared/ui/ErrorBoundary' +import { NoiseOverlay } from '@/shared/ui/NoiseOverlay' +import { DemoOverlay } from '@/shared/ui/DemoOverlay' + +export function App() { + return ( + + + + + + + + + + + ) +} diff --git a/client/src/app/layout/AppHeader.tsx b/client/src/app/layout/AppHeader.tsx new file mode 100755 index 0000000..1c2011a --- /dev/null +++ b/client/src/app/layout/AppHeader.tsx @@ -0,0 +1,202 @@ +import * as React from 'react' +import { useEffect, useState } from 'react' +import AppBar from '@mui/material/AppBar' +import Badge from '@mui/material/Badge' +import Box from '@mui/material/Box' +import Button from '@mui/material/Button' +import IconButton from '@mui/material/IconButton' +import { alpha, useTheme } from '@mui/material/styles' +import Toolbar from '@mui/material/Toolbar' +import Tooltip from '@mui/material/Tooltip' +import Typography from '@mui/material/Typography' +import useMediaQuery from '@mui/material/useMediaQuery' +import { useQuery } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { Menu, Package } from 'lucide-react' +import { Link as RouterLink, useNavigate } from 'react-router-dom' +import { useThemeController } from '@/app/providers/theme-controller' +import { fetchMyCart } from '@/entities/cart/api/cart-api' +import { fetchMyOrders } from '@/entities/order/api/order-api' +import { CartBadge } from '@/features/cart/cart-badge' +import { UserMenu } from '@/features/user/user-menu' +import { STORE_NAME } from '@/shared/config' +import { $user, logout, tokenSet } from '@/shared/model/auth' +import type { ColorScheme } from '@/shared/model/theme' +import { BearLogo } from '@/shared/ui/BearLogo' +import { ModeSwitcher } from '@/shared/ui/ModeSwitcher' +import { SchemeSwitcher } from '@/shared/ui/SchemeSwitcher' +import { NavigationDrawer } from '@/widgets/navigation-drawer' + +type NavItem = { label: string; to: string } + +const navItems: NavItem[] = [{ label: 'Каталог', to: '/' }] + +export const AppHeader = React.memo(function AppHeader() { + const { mode, resolvedMode, scheme, setScheme, cycleMode } = useThemeController() + const user = useUnit($user) + const navigate = useNavigate() + const isAdmin = Boolean(user?.isAdmin) + const headerNavItems = isAdmin ? [...navItems, { label: 'Админка', to: '/admin' }] : navItems + const theme = useTheme() + const isMobile = useMediaQuery(theme.breakpoints.down('md')) + + const cartQuery = useQuery({ + queryKey: ['me', 'cart'], + queryFn: fetchMyCart, + enabled: Boolean(user) && !isAdmin, + }) + const cartCount = cartQuery.data?.items?.length ?? 0 + + const ordersQuery = useQuery({ + queryKey: ['me', 'orders'], + queryFn: fetchMyOrders, + enabled: Boolean(user) && !isAdmin, + }) + const activeOrdersCount = (ordersQuery.data?.items ?? []).filter( + (o) => o.status !== 'DONE' && o.status !== 'CANCELLED', + ).length + + const [mobileOpen, setMobileOpen] = useState(false) + const [scrolled, setScrolled] = useState(false) + + useEffect(() => { + const handler = () => setScrolled(window.scrollY > 0) + handler() + window.addEventListener('scroll', handler, { passive: true }) + return () => window.removeEventListener('scroll', handler) + }, []) + + const go = (to: string) => { + setMobileOpen(false) + navigate(to) + } + + const onLogout = () => { + tokenSet(null) + logout() + setMobileOpen(false) + navigate('/') + } + + return ( + <> + + + {isMobile && ( + setMobileOpen(true)} + aria-label="Открыть меню" + edge="start" + sx={{ mr: 1 }} + > + + + )} + + + + + {STORE_NAME} + + + + {!isMobile && + headerNavItems.map((i) => ( + + ))} + + {!isAdmin && ( + <> + {user && ( + + navigate('/me/orders')} + aria-label={activeOrdersCount > 0 ? `Заказы (${activeOrdersCount})` : 'Заказы'} + > + + + + + + )} + + + + )} + + {!isAdmin && } + + {isAdmin && user && !isMobile && ( + + )} + + {!isMobile && ( + + + setScheme(s)} /> + + + + )} + + + + setMobileOpen(false)} + user={user} + isAdmin={isAdmin} + navItems={headerNavItems} + scheme={scheme} + mode={mode} + resolvedMode={resolvedMode} + onSchemeChange={(s: ColorScheme) => setScheme(s)} + onCycleMode={cycleMode} + onNavigate={go} + onLogout={onLogout} + /> + + ) +}) diff --git a/client/src/app/layout/MainLayout.tsx b/client/src/app/layout/MainLayout.tsx new file mode 100755 index 0000000..063307b --- /dev/null +++ b/client/src/app/layout/MainLayout.tsx @@ -0,0 +1,144 @@ +import { type PropsWithChildren } from 'react' +import Box from '@mui/material/Box' +import Container from '@mui/material/Container' +import Divider from '@mui/material/Divider' +import Grid from '@mui/material/Grid' +import Link from '@mui/material/Link' +import Stack from '@mui/material/Stack' +import Typography from '@mui/material/Typography' +import { Link as RouterLink } from 'react-router-dom' +import { AppHeader } from '@/app/layout/AppHeader' +import vkLogoSrc from '@/shared/assets/vk-logo.svg' +import { STORE_EMAIL, STORE_NAME, STORE_PHONE, VK_URL } from '@/shared/config' +import { CookieConsentBanner } from '@/shared/ui/CookieConsentBanner' +import { DemoBanner } from '@/shared/ui/DemoBanner' +import { ScrollOnNavigate } from '@/shared/ui/ScrollOnNavigate' +import { ScrollToTop } from '@/shared/ui/ScrollToTop' + +export function MainLayout({ children }: PropsWithChildren) { + const year = new Date().getFullYear() + + return ( + + + + + + + + + {children} + + + + + + + + + {STORE_NAME} + + + Изделия ручной работы: вещи с характером и вниманием к деталям. + + + + + Покупателям + + + + Личный кабинет + + + О покупке + + + О нас + + + + + + Контакты + + + + Email:{' '} + + {STORE_EMAIL} + + + + Телефон:{' '} + + {STORE_PHONE} + + + + + VK + + + + + + Юридическая информация + + + + Политика конфиденциальности + + + Пользовательское соглашение + + + + + + + + © {year} {STORE_NAME} + + + + + + + ) +} diff --git a/client/src/app/layout/__tests__/MainLayout.test.tsx b/client/src/app/layout/__tests__/MainLayout.test.tsx new file mode 100755 index 0000000..2ede55f --- /dev/null +++ b/client/src/app/layout/__tests__/MainLayout.test.tsx @@ -0,0 +1,36 @@ +import { render } from '@testing-library/react' +import { MemoryRouter } from 'react-router-dom' +import { describe, expect, it, vi } from 'vitest' +import { MainLayout } from '../MainLayout' + +vi.mock('@/app/layout/AppHeader', () => ({ + AppHeader: () =>
Шапка
, +})) + +vi.mock('@/shared/ui/CookieConsentBanner', () => ({ + CookieConsentBanner: () => null, +})) + +vi.mock('@/shared/ui/DemoBanner', () => ({ + DemoBanner: () => null, +})) + +vi.mock('@/shared/ui/ScrollOnNavigate', () => ({ + ScrollOnNavigate: () => null, +})) + +vi.mock('@/shared/ui/ScrollToTop', () => ({ + ScrollToTop: () => null, +})) + +describe('MainLayout', () => { + it('не задает фиксированную минимальную ширину, которая ломает мобильный экран', () => { + const { container } = render( + + Контент + , + ) + + expect(container.firstElementChild).not.toHaveStyle({ minWidth: '500px' }) + }) +}) diff --git a/client/src/app/providers/AppProviders.tsx b/client/src/app/providers/AppProviders.tsx new file mode 100755 index 0000000..293f057 --- /dev/null +++ b/client/src/app/providers/AppProviders.tsx @@ -0,0 +1,361 @@ +import { type PropsWithChildren, useMemo } from 'react' +import CssBaseline from '@mui/material/CssBaseline' +import { alpha, ThemeProvider, createTheme } from '@mui/material/styles' +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { ThemeControllerProvider, useThemeController } from '@/app/providers/theme-controller' +import { SseProvider } from './SseProvider' + +function AppThemeInner({ children }: PropsWithChildren) { + const controller = useThemeController() + const isDark = controller.resolvedMode === 'dark' + + const theme = useMemo( + () => + createTheme({ + palette: (() => { + const common = { mode: controller.resolvedMode } + + const text = isDark + ? { primary: '#F2F2F2', secondary: 'rgba(242,242,242,0.72)', disabled: 'rgba(242,242,242,0.48)' } + : { primary: '#1F1B16', secondary: 'rgba(31,27,22,0.72)', disabled: 'rgba(31,27,22,0.48)' } + + const chip = isDark ? { default: '#0E1510', paper: '#121B14' } : { default: '#F6FAF6', paper: '#FFFFFF' } + + switch (controller.scheme) { + case 'forest': + return { + ...common, + primary: { main: isDark ? '#8FBC8F' : '#2E8B57' }, + secondary: { main: isDark ? '#CD853F' : '#8B4513' }, + info: { main: isDark ? '#4682B4' : '#1E90FF' }, + success: { main: isDark ? '#90EE90' : '#32CD32' }, + warning: { main: isDark ? '#FFD700' : '#FFA500' }, + error: { main: isDark ? '#F08080' : '#CD5C5C' }, + divider: isDark ? 'rgba(255,255,255,0.12)' : 'rgba(0,0,0,0.08)', + text, + chip, + background: isDark + ? { default: '#0F1720', paper: '#1A242E' } + : { default: '#F8F6F3', paper: '#FFFFFF' }, + } + case 'ocean': + return { + ...common, + primary: { main: isDark ? '#5F9EA0' : '#20B2AA' }, + secondary: { main: isDark ? '#7B68EE' : '#6A5ACD' }, + info: { main: isDark ? '#87CEEB' : '#00BFFF' }, + success: { main: isDark ? '#98FB98' : '#00FA9A' }, + warning: { main: isDark ? '#FFE4B5' : '#FFDAB9' }, + error: { main: isDark ? '#FF6347' : '#FF4500' }, + divider: isDark ? 'rgba(255,255,255,0.12)' : 'rgba(0,0,0,0.08)', + text, + chip, + background: isDark + ? { default: '#0A1A2A', paper: '#0F1D35' } + : { default: '#F0F8FF', paper: '#FFFFFF' }, + } + case 'berry': + return { + ...common, + primary: { main: isDark ? '#9370DB' : '#8A2BE2' }, + secondary: { main: isDark ? '#FF69B4' : '#FF1493' }, + info: { main: isDark ? '#00CED1' : '#00BFFF' }, + success: { main: isDark ? '#00FF7F' : '#7CFC00' }, + warning: { main: isDark ? '#FFD700' : '#FFA500' }, + error: { main: isDark ? '#FF4500' : '#FF6347' }, + divider: isDark ? 'rgba(255,255,255,0.12)' : 'rgba(0,0,0,0.08)', + text, + chip, + background: isDark + ? { default: '#1A0A1A', paper: '#250E25' } + : { default: '#FFF0F5', paper: '#FFFFFF' }, + } + case 'craft': + default: + return { + ...common, + primary: { main: isDark ? '#90A4AE' : '#546E7A' }, + secondary: { main: isDark ? '#78909C' : '#78909C' }, + info: { main: isDark ? '#7986CB' : '#3F51B5' }, + success: { main: isDark ? '#66BB6A' : '#43A047' }, + warning: { main: isDark ? '#FFB74D' : '#F57C00' }, + error: { main: isDark ? '#EF5350' : '#D32F2F' }, + divider: isDark ? 'rgba(255,255,255,0.12)' : 'rgba(0,0,0,0.08)', + text, + chip, + background: isDark + ? { default: '#121212', paper: '#1E1E1E' } + : { default: '#F5F5F5', paper: '#FFFFFF' }, + } + } + })(), + shape: { borderRadius: 12 }, + typography: { + fontFamily: '"Outfit", "Segoe UI", system-ui, sans-serif', + h1: { fontWeight: 700, letterSpacing: '-1px', lineHeight: 1.1, textWrap: 'balance' }, + h2: { fontWeight: 700, letterSpacing: '-0.75px', lineHeight: 1.15, textWrap: 'balance' }, + h3: { fontWeight: 700, letterSpacing: '-0.5px', lineHeight: 1.2, textWrap: 'balance' }, + h4: { fontWeight: 700, letterSpacing: '-0.5px', textWrap: 'balance' }, + h5: { fontWeight: 600, letterSpacing: '-0.25px', textWrap: 'balance' }, + h6: { fontWeight: 600, textWrap: 'balance' }, + subtitle1: { fontWeight: 600 }, + subtitle2: { fontWeight: 500 }, + body1: { fontSize: '0.875rem', lineHeight: 1.6 }, + body2: { fontSize: '0.75rem', lineHeight: 1.5 }, + button: { textTransform: 'none', fontWeight: 600 }, + }, + components: { + MuiButton: { + styleOverrides: { + root: { + textTransform: 'none', + borderRadius: 12, + fontWeight: 600, + transition: 'all 0.2s ease-in-out', + '&:focus-visible': { + outline: '2px solid currentColor', + outlineOffset: 2, + }, + }, + contained: { + boxShadow: '0 4px 14px 0 rgba(0,0,0,0.12)', + '&:hover': { + boxShadow: '0 6px 20px 0 rgba(0,0,0,0.18)', + transform: 'translateY(-2px)', + }, + '&:active': { + boxShadow: '0 2px 8px 0 rgba(0,0,0,0.12)', + transform: 'translateY(0) scale(0.98)', + }, + }, + outlined: { + border: '1px solid', + '&:hover': { + boxShadow: '0 2px 8px 0 rgba(0,0,0,0.08)', + }, + '&:active': { + boxShadow: 'none', + transform: 'scale(0.98)', + }, + }, + text: { + '&:hover': { + backgroundColor: 'action.hover', + }, + '&:active': { + backgroundColor: 'action.selected', + }, + }, + }, + }, + MuiIconButton: { + styleOverrides: { + root: { + transition: 'all 0.2s ease-in-out', + '&:hover': { + backgroundColor: 'action.hover', + transform: 'scale(1.08)', + }, + '&:active': { + backgroundColor: 'action.selected', + transform: 'scale(0.95)', + }, + '&:focus-visible': { + outline: '2px solid currentColor', + outlineOffset: 2, + }, + }, + }, + }, + MuiCard: { + styleOverrides: { + root: { + '&:focus-visible': { + outline: '2px solid currentColor', + outlineOffset: 2, + }, + }, + }, + }, + MuiLink: { + styleOverrides: { + root: { + '&:focus-visible': { + outline: '2px solid currentColor', + outlineOffset: 2, + borderRadius: 2, + }, + }, + }, + }, + MuiInputBase: { + styleOverrides: { + root: { + '&.Mui-focused': { + '& .MuiOutlinedInput-notchedOutline': { + borderWidth: 2, + }, + }, + }, + }, + }, + MuiAlert: { + styleOverrides: { + root: { + borderRadius: 12, + border: '1px solid', + boxShadow: 'none', + fontWeight: 500, + alignItems: 'center', + padding: '8px 12px', + '& .MuiAlert-icon': { + padding: 0, + marginRight: 12, + display: 'flex', + alignItems: 'center', + }, + '& .MuiAlert-message': { + padding: 0, + }, + '& .MuiAlert-action': { + padding: 0, + marginRight: 0, + marginLeft: 8, + }, + }, + colorSuccess: ({ theme }) => { + const isDark = theme.palette.mode === 'dark' + const p = theme.palette.success + return { + bgcolor: isDark ? alpha(p.light, 0.08) : alpha(p.main, 0.08), + borderColor: isDark ? alpha(p.light, 0.2) : alpha(p.main, 0.2), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + '&.MuiAlert-outlined': { + bgcolor: 'transparent', + borderColor: isDark ? alpha(p.light, 0.3) : alpha(p.main, 0.3), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + }, + '&.MuiAlert-filled': { + bgcolor: isDark ? alpha(p.light, 0.15) : p.dark, + borderColor: 'transparent', + color: isDark ? alpha(p.light, 0.9) : '#FFFFFF', + }, + } + }, + colorError: ({ theme }) => { + const isDark = theme.palette.mode === 'dark' + const p = theme.palette.error + return { + bgcolor: isDark ? alpha(p.light, 0.08) : alpha(p.main, 0.08), + borderColor: isDark ? alpha(p.light, 0.2) : alpha(p.main, 0.2), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + '&.MuiAlert-outlined': { + bgcolor: 'transparent', + borderColor: isDark ? alpha(p.light, 0.3) : alpha(p.main, 0.3), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + }, + '&.MuiAlert-filled': { + bgcolor: isDark ? alpha(p.light, 0.15) : p.dark, + borderColor: 'transparent', + color: isDark ? alpha(p.light, 0.9) : '#FFFFFF', + }, + } + }, + colorWarning: ({ theme }) => { + const isDark = theme.palette.mode === 'dark' + const p = theme.palette.warning + return { + bgcolor: isDark ? alpha(p.light, 0.08) : alpha(p.main, 0.08), + borderColor: isDark ? alpha(p.light, 0.2) : alpha(p.main, 0.2), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + '&.MuiAlert-outlined': { + bgcolor: 'transparent', + borderColor: isDark ? alpha(p.light, 0.3) : alpha(p.main, 0.3), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + }, + '&.MuiAlert-filled': { + bgcolor: isDark ? alpha(p.light, 0.15) : p.dark, + borderColor: 'transparent', + color: isDark ? alpha(p.light, 0.9) : '#FFFFFF', + }, + } + }, + colorInfo: ({ theme }) => { + const isDark = theme.palette.mode === 'dark' + const p = theme.palette.info + return { + bgcolor: isDark ? alpha(p.light, 0.08) : alpha(p.main, 0.08), + borderColor: isDark ? alpha(p.light, 0.2) : alpha(p.main, 0.2), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + '&.MuiAlert-outlined': { + bgcolor: 'transparent', + borderColor: isDark ? alpha(p.light, 0.3) : alpha(p.main, 0.3), + color: isDark ? p.light : p.dark, + '& .MuiAlert-icon': { color: isDark ? p.light : p.dark }, + }, + '&.MuiAlert-filled': { + bgcolor: isDark ? alpha(p.light, 0.15) : p.dark, + borderColor: 'transparent', + color: isDark ? alpha(p.light, 0.9) : '#FFFFFF', + }, + } + }, + }, + }, + MuiSnackbarContent: { + styleOverrides: { + root: { + borderRadius: 12, + border: '1px solid', + borderColor: isDark ? 'rgba(255,255,255,0.1)' : 'rgba(0,0,0,0.06)', + bgcolor: isDark ? '#1E1E1E' : '#FFFFFF', + boxShadow: '0 2px 12px rgba(0,0,0,0.06)', + color: isDark ? '#F2F2F2' : '#1F1B16', + fontWeight: 500, + }, + }, + }, + }, + }), + [controller.resolvedMode, controller.scheme], + ) + + return ( + + + {children} + + ) +} + +export function AppProviders({ children }: PropsWithChildren) { + const queryClient = useMemo( + () => + new QueryClient({ + defaultOptions: { + queries: { + staleTime: 30_000, + retry: 1, + refetchOnWindowFocus: false, + }, + }, + }), + [], + ) + + return ( + + + + {children} + + + ) +} diff --git a/client/src/app/providers/SseProvider.tsx b/client/src/app/providers/SseProvider.tsx new file mode 100755 index 0000000..5b24b46 --- /dev/null +++ b/client/src/app/providers/SseProvider.tsx @@ -0,0 +1,83 @@ +import { useEffect, useRef } from 'react' +import { useQueryClient } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { createEventStream } from '@/shared/lib/sse' +import { $token } from '@/shared/model/auth' + +export function SseProvider() { + const token = useUnit($token) + const queryClient = useQueryClient() + const sourceRef = useRef(null) + + useEffect(() => { + if (!token) { + if (sourceRef.current) { + sourceRef.current.close() + sourceRef.current = null + } + return + } + + const es = createEventStream(token) + sourceRef.current = es + + function invalidateOrderQueries(orderId: unknown) { + if (!orderId) return + queryClient.invalidateQueries({ queryKey: ['me', 'orders'] }) + queryClient.invalidateQueries({ queryKey: ['me', 'orders', orderId] }) + queryClient.invalidateQueries({ queryKey: ['admin', 'orders', 'detail', orderId] }) + queryClient.invalidateQueries({ queryKey: ['admin', 'orders'] }) + queryClient.invalidateQueries({ queryKey: ['admin', 'orders', 'summary'] }) + } + + function handleEvent(eventName: string) { + return function (event: MessageEvent) { + try { + const data = JSON.parse(event.data) + const orderId = data.orderId + + switch (eventName) { + case 'message:new': + queryClient.invalidateQueries({ queryKey: ['me', 'messages', 'unread-count'] }) + queryClient.invalidateQueries({ queryKey: ['me', 'conversations'] }) + invalidateOrderQueries(orderId) + break + case 'order:statusChanged': + invalidateOrderQueries(orderId) + break + case 'order:updated': + invalidateOrderQueries(orderId) + break + case 'order:new': + queryClient.invalidateQueries({ queryKey: ['admin', 'orders', 'summary'] }) + queryClient.invalidateQueries({ queryKey: ['admin', 'orders'] }) + break + } + } catch (err) { + console.warn('[sse] Failed to parse event data', err) + } + } + } + + const messageNewHandler = handleEvent('message:new') + const orderStatusHandler = handleEvent('order:statusChanged') + const orderUpdatedHandler = handleEvent('order:updated') + const orderNewHandler = handleEvent('order:new') + + es.addEventListener('message:new', messageNewHandler) + es.addEventListener('order:statusChanged', orderStatusHandler) + es.addEventListener('order:updated', orderUpdatedHandler) + es.addEventListener('order:new', orderNewHandler) + + return () => { + es.removeEventListener('message:new', messageNewHandler) + es.removeEventListener('order:statusChanged', orderStatusHandler) + es.removeEventListener('order:updated', orderUpdatedHandler) + es.removeEventListener('order:new', orderNewHandler) + es.close() + sourceRef.current = null + } + }, [token, queryClient]) + + return null +} diff --git a/client/src/app/providers/__tests__/SseProvider.test.tsx b/client/src/app/providers/__tests__/SseProvider.test.tsx new file mode 100755 index 0000000..7bc3ba8 --- /dev/null +++ b/client/src/app/providers/__tests__/SseProvider.test.tsx @@ -0,0 +1,159 @@ +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render } from '@testing-library/react' +import { afterEach, describe, expect, it, vi } from 'vitest' +import { SseProvider } from '../SseProvider' + +const mockInvalidateQueries = vi.fn() + +vi.mock('@tanstack/react-query', async () => { + const actual = await vi.importActual('@tanstack/react-query') + return { ...actual, useQueryClient: () => ({ invalidateQueries: mockInvalidateQueries }) } +}) + +vi.mock('@/shared/model/auth', () => ({ + $token: { + defaultState: null, + subscribe: () => () => {}, + getState: () => null, + watch: () => () => {}, + on: () => {}, + reset: () => {}, + }, +})) + +let mockToken: string | null = null +let mockEventHandlers: Record void> = {} +let mockCloseCalls = 0 + +class MockEventSource { + url: string + constructor(url: string) { + this.url = url + mockCloseCalls = 0 + mockEventHandlers = {} + } + addEventListener(type: string, handler: (event: MessageEvent) => void) { + mockEventHandlers[type] = handler + } + removeEventListener(type: string, _handler: (event: MessageEvent) => void) { + delete mockEventHandlers[type] + } + close() { + mockCloseCalls++ + } +} + +vi.mock('@/shared/lib/sse', () => ({ + createEventStream: (token: string) => { + mockToken = token + return new MockEventSource(`/api/sse/stream?token=${token}`) as unknown as EventSource + }, +})) + +vi.mock('effector-react', async () => { + const actual = await vi.importActual('effector-react') + return { ...actual, useUnit: () => mockToken } +}) + +function renderSse() { + const qc = new QueryClient({ defaultOptions: { queries: { retry: false } } }) + return render( + + + , + ) +} + +describe('SseProvider', () => { + afterEach(() => { + mockToken = null + mockInvalidateQueries.mockReset() + mockCloseCalls = 0 + mockEventHandlers = {} + }) + + it('renders nothing (returns null)', () => { + mockToken = null + const { container } = renderSse() + expect(container.innerHTML).toBe('') + }) + + it('does not create EventSource when token is null', () => { + mockToken = null + renderSse() + expect(mockToken).toBeNull() + }) + + it('creates EventSource when token is set', () => { + mockToken = 'test-jwt' + renderSse() + expect(mockToken).toBe('test-jwt') + }) + + it('closes EventSource on unmount', () => { + mockToken = 'test-jwt' + const { unmount } = renderSse() + expect(mockCloseCalls).toBe(0) + unmount() + expect(mockCloseCalls).toBe(1) + }) + + it('invalidates unread-count and conversations on message:new', () => { + mockToken = 'test-jwt' + renderSse() + const handler = mockEventHandlers['message:new'] + expect(handler).toBeDefined() + handler(new MessageEvent('message:new', { data: JSON.stringify({ orderId: 'o1' }) })) + + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'messages', 'unread-count'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'conversations'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders', 'o1'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'detail', 'o1'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'summary'] }) + }) + + it('invalidates order queries on order:statusChanged', () => { + mockToken = 'test-jwt' + renderSse() + const handler = mockEventHandlers['order:statusChanged'] + handler(new MessageEvent('order:statusChanged', { data: JSON.stringify({ orderId: 'o2' }) })) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders', 'o2'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'detail', 'o2'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'summary'] }) + }) + + it('invalidates order queries on order:updated', () => { + mockToken = 'test-jwt' + renderSse() + const handler = mockEventHandlers['order:updated'] + handler(new MessageEvent('order:updated', { data: JSON.stringify({ orderId: 'o3' }) })) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['me', 'orders', 'o3'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'detail', 'o3'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'summary'] }) + }) + + it('invalidates admin queries on order:new', () => { + mockToken = 'test-jwt' + renderSse() + const handler = mockEventHandlers['order:new'] + handler(new MessageEvent('order:new', { data: JSON.stringify({ orderId: 'o4' }) })) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders', 'summary'] }) + expect(mockInvalidateQueries).toHaveBeenCalledWith({ queryKey: ['admin', 'orders'] }) + }) + + it('handles invalid JSON gracefully', () => { + mockToken = 'test-jwt' + renderSse() + const handler = mockEventHandlers['message:new'] + expect(() => { + handler(new MessageEvent('message:new', { data: ':heartbit' })) + }).not.toThrow() + expect(mockInvalidateQueries).not.toHaveBeenCalled() + }) +}) diff --git a/client/src/app/providers/theme-controller.tsx b/client/src/app/providers/theme-controller.tsx new file mode 100755 index 0000000..7dd2fe2 --- /dev/null +++ b/client/src/app/providers/theme-controller.tsx @@ -0,0 +1,113 @@ +import { createContext, type PropsWithChildren, useContext, useEffect, useMemo, useState } from 'react' +import type { PaletteMode } from '@mui/material' +import type { ColorScheme, ThemeModePreference } from '@/shared/model/theme' + +export type ThemeSettings = { + mode: ThemeModePreference + scheme: ColorScheme +} + +export type ThemeController = ThemeSettings & { + /** Итоговый режим, учитывая system. */ + resolvedMode: PaletteMode + setMode: (mode: ThemeModePreference) => void + toggleMode: () => void + cycleMode: () => void + setScheme: (scheme: ColorScheme) => void +} + +const THEME_STORAGE_KEY = 'craftshop_theme' + +function readStoredTheme(): ThemeSettings | null { + try { + const raw = localStorage.getItem(THEME_STORAGE_KEY) + if (!raw) return null + const parsed = JSON.parse(raw) + const mode: unknown = parsed?.mode + const scheme: unknown = parsed?.scheme + const modeOk = mode === 'light' || mode === 'dark' || mode === 'system' + const schemeOk = scheme === 'craft' || scheme === 'forest' || scheme === 'ocean' || scheme === 'berry' + if (!modeOk || !schemeOk) return null + return { mode, scheme } + } catch (err) { + console.warn('[theme] Failed to read stored theme', err) + return null + } +} + +function getSystemMode(): PaletteMode { + if (typeof window === 'undefined') return 'light' + return window.matchMedia?.('(prefers-color-scheme: dark)').matches ? 'dark' : 'light' +} + +function resolveMode(pref: ThemeModePreference): PaletteMode { + return pref === 'system' ? getSystemMode() : pref +} + +const ThemeControllerContext = createContext(null) + +export function useThemeController(): ThemeController { + const ctx = useContext(ThemeControllerContext) + if (!ctx) throw new Error('useThemeController must be used within ThemeControllerProvider') + return ctx +} + +export function ThemeControllerProvider({ children }: PropsWithChildren) { + const [settings, setSettings] = useState( + () => readStoredTheme() ?? { mode: 'system', scheme: 'craft' }, + ) + + const [systemMode, setSystemMode] = useState(() => getSystemMode()) + + useEffect(() => { + const mql = window.matchMedia?.('(prefers-color-scheme: dark)') + if (!mql) return + + const handler = () => setSystemMode(mql.matches ? 'dark' : 'light') + + // начальное значение + handler() + + if (typeof mql.addEventListener === 'function') { + mql.addEventListener('change', handler) + return () => mql.removeEventListener('change', handler) + } + + // Safari старых версий + mql.addListener(handler) + return () => mql.removeListener(handler) + }, []) + + useEffect(() => { + try { + localStorage.setItem(THEME_STORAGE_KEY, JSON.stringify(settings)) + } catch (err) { + console.warn('[theme] Failed to persist theme setting', err) + } + }, [settings]) + + const resolvedMode = settings.mode === 'system' ? systemMode : settings.mode + + const controller = useMemo( + () => ({ + mode: settings.mode, + resolvedMode, + scheme: settings.scheme, + setMode: (mode) => setSettings((s) => ({ ...s, mode })), + toggleMode: () => + setSettings((s) => ({ + ...s, + mode: resolveMode(s.mode) === 'light' ? 'dark' : 'light', + })), + cycleMode: () => + setSettings((s) => ({ + ...s, + mode: s.mode === 'system' ? 'light' : s.mode === 'light' ? 'dark' : 'system', + })), + setScheme: (scheme) => setSettings((s) => ({ ...s, scheme })), + }), + [resolvedMode, settings.mode, settings.scheme], + ) + + return {children} +} diff --git a/client/src/app/routes/index.tsx b/client/src/app/routes/index.tsx new file mode 100755 index 0000000..2cbe2e5 --- /dev/null +++ b/client/src/app/routes/index.tsx @@ -0,0 +1,135 @@ +import { lazy, Suspense } from 'react' +import { Route, Routes } from 'react-router-dom' +import { MainLayout } from '@/app/layout/MainLayout' +import { usePageTitleReset } from '@/shared/lib/use-page-title' +import { SkeletonPage } from '@/shared/ui/SkeletonPage' + +const AdminLayoutPage = lazy(() => import('@/pages/admin-layout').then((m) => ({ default: m.AdminLayoutPage }))) +const MeLayoutPage = lazy(() => import('@/pages/me').then((m) => ({ default: m.MeLayoutPage }))) + +const HomePage = lazy(() => import('@/pages/home').then((m) => ({ default: m.HomePage }))) +const AuthPage = lazy(() => import('@/pages/auth').then((m) => ({ default: m.AuthPage }))) +const AuthCallbackPage = lazy(() => import('@/pages/auth').then((m) => ({ default: m.AuthCallbackPage }))) +const CartPage = lazy(() => import('@/pages/cart').then((m) => ({ default: m.CartPage }))) +const CheckoutPage = lazy(() => import('@/pages/checkout').then((m) => ({ default: m.CheckoutPage }))) +const AboutPage = lazy(() => import('@/pages/about').then((m) => ({ default: m.AboutPage }))) +const InfoPage = lazy(() => import('@/pages/info').then((m) => ({ default: m.InfoPage }))) +const PrivacyPolicyPage = lazy(() => import('@/pages/privacy-policy').then((m) => ({ default: m.PrivacyPolicyPage }))) +const TermsPage = lazy(() => import('@/pages/terms').then((m) => ({ default: m.TermsPage }))) +const ProductPage = lazy(() => import('@/pages/product').then((m) => ({ default: m.ProductPage }))) +const NotFoundPage = lazy(() => import('@/pages/not-found').then((m) => ({ default: m.NotFoundPage }))) + +export function AppRoutes() { + usePageTitleReset() + + return ( + + + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + }> + + + } + /> + + + ) +} diff --git a/client/src/app/styles/global.css b/client/src/app/styles/global.css new file mode 100755 index 0000000..fa3723d --- /dev/null +++ b/client/src/app/styles/global.css @@ -0,0 +1,45 @@ +@font-face { + font-family: 'Outfit'; + font-style: normal; + font-weight: 400; + src: url('/fonts/Outfit-Regular.woff2') format('woff2'); + font-display: swap; +} +@font-face { + font-family: 'Outfit'; + font-style: normal; + font-weight: 500; + src: url('/fonts/Outfit-Medium.woff2') format('woff2'); + font-display: swap; +} +@font-face { + font-family: 'Outfit'; + font-style: normal; + font-weight: 600; + src: url('/fonts/Outfit-SemiBold.woff2') format('woff2'); + font-display: swap; +} +@font-face { + font-family: 'Outfit'; + font-style: normal; + font-weight: 700; + src: url('/fonts/Outfit-Bold.woff2') format('woff2'); + font-display: swap; +} + +:root { + color-scheme: light; +} +html { + scroll-behavior: smooth; +} +html, +body, +#root { + min-height: 100%; +} +body { + margin: 0; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} diff --git a/client/src/assets/hero.png b/client/src/assets/hero.png new file mode 100755 index 0000000000000000000000000000000000000000..02251f4b956c55af2d76fd0788124d7eee2b45eb GIT binary patch literal 13057 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 P00000NkvXXu0mjft=yBf literal 0 HcmV?d00001 diff --git a/client/src/assets/react.svg b/client/src/assets/react.svg new file mode 100755 index 0000000..6c87de9 --- /dev/null +++ b/client/src/assets/react.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/client/src/assets/vite.svg b/client/src/assets/vite.svg new file mode 100755 index 0000000..5101b67 --- /dev/null +++ b/client/src/assets/vite.svg @@ -0,0 +1 @@ +Vite diff --git a/client/src/entities/cart/api/cart-api.ts b/client/src/entities/cart/api/cart-api.ts new file mode 100755 index 0000000..d509ca8 --- /dev/null +++ b/client/src/entities/cart/api/cart-api.ts @@ -0,0 +1,21 @@ +import type { CartItem } from '@/entities/cart/model/types' +import { apiClient } from '@/shared/api/client' + +export type CartResponse = { items: CartItem[] } + +export async function fetchMyCart(): Promise { + const { data } = await apiClient.get('me/cart') + return data +} + +export async function addToCart(body: { productId: string; qty?: number }): Promise { + await apiClient.post('me/cart/items', body) +} + +export async function setCartQty(id: string, qty: number): Promise { + await apiClient.patch(`me/cart/items/${id}`, { qty }) +} + +export async function removeCartItem(id: string): Promise { + await apiClient.delete(`me/cart/items/${id}`) +} diff --git a/client/src/entities/cart/index.ts b/client/src/entities/cart/index.ts new file mode 100755 index 0000000..6870839 --- /dev/null +++ b/client/src/entities/cart/index.ts @@ -0,0 +1,3 @@ +export type { CartItem } from './model/types' +export { fetchMyCart, addToCart, setCartQty, removeCartItem } from './api/cart-api' +export type { CartResponse } from './api/cart-api' diff --git a/client/src/entities/cart/lib/use-cart-query.ts b/client/src/entities/cart/lib/use-cart-query.ts new file mode 100755 index 0000000..614d1e1 --- /dev/null +++ b/client/src/entities/cart/lib/use-cart-query.ts @@ -0,0 +1,14 @@ +import { useQuery } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { fetchMyCart } from '../api/cart-api' +import { $user } from '@/shared/model/auth' + +export function useCartQuery() { + const user = useUnit($user) + + return useQuery({ + queryKey: ['me', 'cart'], + queryFn: fetchMyCart, + enabled: Boolean(user), + }) +} diff --git a/client/src/entities/cart/model/types.ts b/client/src/entities/cart/model/types.ts new file mode 100755 index 0000000..a64b3bc --- /dev/null +++ b/client/src/entities/cart/model/types.ts @@ -0,0 +1,7 @@ +import type { Product } from '@/entities/product/model/types' + +export type CartItem = { + id: string + qty: number + product: Product +} diff --git a/client/src/entities/catalog-slider/api/catalog-slider-api.ts b/client/src/entities/catalog-slider/api/catalog-slider-api.ts new file mode 100755 index 0000000..f28ced4 --- /dev/null +++ b/client/src/entities/catalog-slider/api/catalog-slider-api.ts @@ -0,0 +1,19 @@ +import { apiClient } from '@/shared/api/client' +import type { CatalogSliderSlide, AdminCatalogSliderSlide } from '../model/types' + +export async function fetchCatalogSlider(): Promise<{ slides: CatalogSliderSlide[] }> { + const { data } = await apiClient.get<{ slides: CatalogSliderSlide[] }>('catalog-slider') + return data +} + +export async function fetchAdminCatalogSlider(): Promise<{ slides: AdminCatalogSliderSlide[] }> { + const { data } = await apiClient.get<{ slides: AdminCatalogSliderSlide[] }>('admin/catalog-slider') + return data +} + +export async function putAdminCatalogSlider(body: { + slides: Array<{ galleryImageId: string; caption: string; textColor?: string }> +}): Promise<{ slides: AdminCatalogSliderSlide[] }> { + const { data } = await apiClient.put<{ slides: AdminCatalogSliderSlide[] }>('admin/catalog-slider', body) + return data +} diff --git a/client/src/entities/catalog-slider/index.ts b/client/src/entities/catalog-slider/index.ts new file mode 100755 index 0000000..1c38c32 --- /dev/null +++ b/client/src/entities/catalog-slider/index.ts @@ -0,0 +1,2 @@ +export { fetchCatalogSlider, fetchAdminCatalogSlider, putAdminCatalogSlider } from './api/catalog-slider-api' +export type { CatalogSliderSlide, AdminCatalogSliderSlide } from './model/types' diff --git a/client/src/entities/catalog-slider/model/types.ts b/client/src/entities/catalog-slider/model/types.ts new file mode 100755 index 0000000..597369c --- /dev/null +++ b/client/src/entities/catalog-slider/model/types.ts @@ -0,0 +1,10 @@ +export type CatalogSliderSlide = { + id: string + url: string + caption: string + textColor?: string +} + +export type AdminCatalogSliderSlide = CatalogSliderSlide & { + galleryImageId: string +} diff --git a/client/src/entities/gallery/api/gallery-api.ts b/client/src/entities/gallery/api/gallery-api.ts new file mode 100755 index 0000000..4d4b15d --- /dev/null +++ b/client/src/entities/gallery/api/gallery-api.ts @@ -0,0 +1,52 @@ +import type { GalleryImageItem } from '@/entities/gallery/model/types' +import { apiClient } from '@/shared/api/client' +import { apiBaseURL } from '@/shared/config' +import { ADMIN_UPLOAD_IMAGE_MAX_BYTES, formatAdminImageMaxSizeHint } from '@/shared/constants/upload-limits' + +export async function fetchAdminGallery(): Promise<{ items: GalleryImageItem[] }> { + const { data } = await apiClient.get<{ items: GalleryImageItem[] }>('admin/gallery') + return data +} + +export async function deleteGalleryImage(id: string): Promise { + await apiClient.delete(`admin/gallery/${id}`) +} + +export async function uploadGalleryImages(files: File[]): Promise { + for (const f of files) { + if (f.size > ADMIN_UPLOAD_IMAGE_MAX_BYTES) { + throw new Error( + `Файл «${f.name}» слишком большой (максимум ${formatAdminImageMaxSizeHint()} на одно изображение).`, + ) + } + } + const fd = new FormData() + for (const f of files) { + fd.append('files', f, f.name) + } + const token = localStorage.getItem('craftshop_auth_token') + const base = apiBaseURL.replace(/\/$/, '') + const res = await fetch(`${base}/admin/gallery/upload`, { + method: 'POST', + headers: token ? { Authorization: `Bearer ${token}` } : {}, + body: fd, + }) + const payload = (await res.json().catch(() => ({}))) as { urls?: string[]; error?: string } + if (!res.ok) { + if (res.status === 413) { + throw new Error( + 'Сервер отклонил файл как слишком большой (413). На проде часто лимит nginx: добавьте client_max_body_size для /api/ (см. docs/nginx-upload-limit.md). Проверьте также MAX_UPLOAD_BODY_BYTES в .env на сервере.', + ) + } + throw new Error(typeof payload.error === 'string' ? payload.error : `Ошибка загрузки (${res.status})`) + } + if (!Array.isArray(payload.urls)) { + throw new Error('Некорректный ответ сервера') + } + return payload.urls +} + +export async function resizeGalleryImage(id: string): Promise<{ url: string }> { + const { data } = await apiClient.post<{ url: string }>(`admin/gallery/${id}/resize`) + return data +} diff --git a/client/src/entities/gallery/index.ts b/client/src/entities/gallery/index.ts new file mode 100755 index 0000000..9f9114c --- /dev/null +++ b/client/src/entities/gallery/index.ts @@ -0,0 +1,3 @@ +export { fetchAdminGallery, deleteGalleryImage, uploadGalleryImages, resizeGalleryImage } from './api/gallery-api' +export type { GalleryImageItem } from './model/types' +export { GalleryGrid } from './ui/GalleryGrid' diff --git a/client/src/entities/gallery/model/types.ts b/client/src/entities/gallery/model/types.ts new file mode 100755 index 0000000..01109b1 --- /dev/null +++ b/client/src/entities/gallery/model/types.ts @@ -0,0 +1,7 @@ +export type GalleryImageItem = { + id: string + url: string + isResized: boolean + createdAt: string + inUse?: boolean +} diff --git a/client/src/entities/gallery/ui/GalleryGrid.tsx b/client/src/entities/gallery/ui/GalleryGrid.tsx new file mode 100755 index 0000000..ad9bb83 --- /dev/null +++ b/client/src/entities/gallery/ui/GalleryGrid.tsx @@ -0,0 +1,100 @@ +import AutoFixHighOutlinedIcon from '@mui/icons-material/AutoFixHighOutlined' +import CheckCircleOutlineOutlinedIcon from '@mui/icons-material/CheckCircleOutlineOutlined' +import DeleteOutlineOutlinedIcon from '@mui/icons-material/DeleteOutlineOutlined' +import Box from '@mui/material/Box' +import Chip from '@mui/material/Chip' +import IconButton from '@mui/material/IconButton' +import Tooltip from '@mui/material/Tooltip' +import { OptimizedImage } from '@/shared/ui/OptimizedImage' +import type { GalleryImageItem } from '../model/types' + +type Props = { + items: GalleryImageItem[] + deleting?: boolean + resizing?: string | null + onDelete: (id: string) => void + onResize: (id: string) => void +} + +export function GalleryGrid({ items, deleting, resizing, onDelete, onResize }: Props) { + return ( + + {items.map((item) => ( + + + + {item.isResized ? ( + } + sx={{ height: 24, '& .MuiChip-label': { px: 0.75 }, '& .MuiChip-icon': { fontSize: 14, ml: 0.5 } }} + /> + ) : ( + + )} + + + {!item.isResized && ( + + onResize(item.id)} + > + + + + )} + + onDelete(item.id)} + > + + + + + + ))} + + ) +} diff --git a/client/src/entities/notification/api/notifications-api.ts b/client/src/entities/notification/api/notifications-api.ts new file mode 100755 index 0000000..c75126f --- /dev/null +++ b/client/src/entities/notification/api/notifications-api.ts @@ -0,0 +1,54 @@ +import { apiClient } from '@/shared/api/client' + +export interface UserNotificationSettings { + id: string + userId: string + globalEnabled: boolean + orderCreated: boolean + orderStatusChanged: boolean + orderMessageReceived: boolean + paymentStatusChanged: boolean + deliveryFeeAdjusted: boolean + createdAt: string + updatedAt: string +} + +export interface AdminNotificationSettings { + id: string + emailEnabled: boolean + telegramEnabled: boolean + telegramChatId: string | null + newOrder: boolean + newOrderMessage: boolean + newReview: boolean + authCodeDuplicate: boolean + createdAt: string + updatedAt: string +} + +export async function fetchUserNotificationSettings(): Promise<{ settings: UserNotificationSettings }> { + const { data } = await apiClient.get<{ settings: UserNotificationSettings }>('me/notifications/settings') + return data +} + +export async function updateUserNotificationSettings( + settings: Partial, +): Promise<{ settings: UserNotificationSettings }> { + const { data } = await apiClient.put<{ settings: UserNotificationSettings }>('me/notifications/settings', settings) + return data +} + +export async function fetchAdminNotificationSettings(): Promise<{ settings: AdminNotificationSettings }> { + const { data } = await apiClient.get<{ settings: AdminNotificationSettings }>('admin/notifications/settings') + return data +} + +export async function updateAdminNotificationSettings( + settings: Partial, +): Promise<{ settings: AdminNotificationSettings }> { + const { data } = await apiClient.put<{ settings: AdminNotificationSettings }>( + 'admin/notifications/settings', + settings, + ) + return data +} diff --git a/client/src/entities/notification/index.ts b/client/src/entities/notification/index.ts new file mode 100755 index 0000000..2394f9c --- /dev/null +++ b/client/src/entities/notification/index.ts @@ -0,0 +1,7 @@ +export { + fetchUserNotificationSettings, + updateUserNotificationSettings, + fetchAdminNotificationSettings, + updateAdminNotificationSettings, +} from './api/notifications-api' +export type { UserNotificationSettings, AdminNotificationSettings } from './api/notifications-api' diff --git a/client/src/entities/order/api/admin-order-api.ts b/client/src/entities/order/api/admin-order-api.ts new file mode 100755 index 0000000..98f6439 --- /dev/null +++ b/client/src/entities/order/api/admin-order-api.ts @@ -0,0 +1,96 @@ +import { apiClient } from '@/shared/api/client' + +export type AdminOrderListItem = { + id: string + status: string + deliveryType: 'delivery' | 'pickup' + deliveryFeeLocked: boolean + deliveryCarrier?: string | null + paymentMethod?: 'online' | 'on_pickup' + totalCents: number + currency: string + createdAt: string + updatedAt: string + user: { id: string; email: string } + itemsCount: number +} + +export type AdminOrdersListResponse = { + items: AdminOrderListItem[] + total: number + page: number + pageSize: number +} + +export type AdminOrderDetailResponse = { + item: { + id: string + status: string + deliveryType: 'delivery' | 'pickup' + deliveryCarrier?: string | null + paymentMethod?: 'online' | 'on_pickup' + itemsSubtotalCents: number + deliveryFeeCents: number + deliveryFeeLocked: boolean + totalCents: number + currency: string + addressSnapshotJson: string | null + comment: string | null + createdAt: string + updatedAt: string + user: { + id: string + email: string + displayName: string | null + avatar?: string | null + avatarStyle?: string | null + } + items: Array<{ + id: string + productId: string + qty: number + titleSnapshot: string + priceCentsSnapshot: number + }> + messages: Array<{ + id: string + authorType: string + text: string + attachmentUrl?: string | null + createdAt: string + }> + } +} + +export async function fetchAdminOrdersSummary(): Promise<{ attentionCount: number }> { + const { data } = await apiClient.get<{ attentionCount: number }>('admin/orders/summary') + return data +} + +export async function fetchAdminOrders(params?: { + status?: string + deliveryType?: 'delivery' | 'pickup' + q?: string + page?: number + pageSize?: number +}): Promise { + const { data } = await apiClient.get('admin/orders', { params }) + return data +} + +export async function fetchAdminOrder(id: string): Promise { + const { data } = await apiClient.get(`admin/orders/${id}`) + return data +} + +export async function setAdminOrderStatus(id: string, status: string): Promise { + await apiClient.patch(`admin/orders/${id}/status`, { status }) +} + +export async function patchAdminOrderDeliveryFee(id: string, deliveryFeeCents: number): Promise { + await apiClient.patch(`admin/orders/${id}/delivery-fee`, { deliveryFeeCents }) +} + +export async function postAdminOrderMessage(id: string, text: string): Promise { + await apiClient.post(`admin/orders/${id}/messages`, { text }) +} diff --git a/client/src/entities/order/api/order-api.ts b/client/src/entities/order/api/order-api.ts new file mode 100755 index 0000000..a57bc29 --- /dev/null +++ b/client/src/entities/order/api/order-api.ts @@ -0,0 +1,103 @@ +import { apiClient } from '@/shared/api/client' +import type { DeliveryCarrierCode } from '@/shared/constants/delivery-carrier' + +export type OrderListItem = { + id: string + status: string + totalCents: number + currency: string + createdAt: string + updatedAt: string + itemsCount: number +} + +export type OrderListResponse = { items: OrderListItem[] } + +export type OrderPaymentMethod = 'online' | 'on_pickup' + +export type OrderDetailResponse = { + item: { + id: string + status: string + deliveryType: 'delivery' | 'pickup' + deliveryCarrier?: DeliveryCarrierCode | null + paymentMethod?: OrderPaymentMethod + itemsSubtotalCents: number + deliveryFeeCents: number + deliveryFeeLocked: boolean + totalCents: number + currency: string + addressSnapshotJson: string | null + comment: string | null + createdAt: string + updatedAt: string + items: Array<{ + id: string + productId: string + qty: number + titleSnapshot: string + priceCentsSnapshot: number + }> + messages: Array<{ + id: string + authorType: 'user' | 'admin' + text: string + attachmentUrl?: string | null + createdAt: string + }> + } +} + +export async function createOrder(body: { + deliveryType: 'delivery' | 'pickup' + deliveryCarrier?: DeliveryCarrierCode | null + paymentMethod?: OrderPaymentMethod + addressId?: string | null + comment?: string | null +}): Promise<{ orderId: string }> { + const { data } = await apiClient.post<{ orderId: string }>('me/orders', body) + return data +} + +export async function fetchMyOrders(): Promise { + const { data } = await apiClient.get('me/orders') + return data +} + +export async function fetchMyOrder(id: string): Promise { + const { data } = await apiClient.get(`me/orders/${id}`) + return data +} + +/** Создать платёж в ЮKassa и получить URL для редиректа на форму оплаты. */ +export async function createOrderPayment(orderId: string): Promise<{ confirmationUrl: string }> { + const { data } = await apiClient.post<{ confirmationUrl: string }>(`me/orders/${orderId}/pay`) + return data +} + +/** Получить статус платежа для заказа. */ +export async function getOrderPaymentStatus(orderId: string): Promise<{ status: string | null; paid: boolean }> { + const { data } = await apiClient.get<{ status: string | null; paid: boolean }>(`me/orders/${orderId}/payment`) + return data +} + +export async function postOrderMessage(id: string, text: string): Promise { + await apiClient.post(`me/orders/${id}/messages`, { text }) +} + +export async function confirmOrderReceived(id: string): Promise<{ ok: boolean; status: string }> { + const { data } = await apiClient.post<{ ok: boolean; status: string }>(`me/orders/${id}/confirm-received`) + return data +} + +export type ReviewEligibilityItem = { productId: string; title: string; hasReview: boolean } + +export async function fetchOrderReviewEligibility(orderId: string): Promise<{ + canReview: boolean + items: ReviewEligibilityItem[] +}> { + const { data } = await apiClient.get<{ canReview: boolean; items: ReviewEligibilityItem[] }>( + `me/orders/${orderId}/review-eligibility`, + ) + return data +} diff --git a/client/src/entities/order/index.ts b/client/src/entities/order/index.ts new file mode 100755 index 0000000..50d153d --- /dev/null +++ b/client/src/entities/order/index.ts @@ -0,0 +1,9 @@ +export { + fetchMyOrders, + createOrder, + confirmOrderReceived, + fetchMyOrder, + fetchOrderReviewEligibility, +} from './api/order-api' +export { createOrderPayment, getOrderPaymentStatus, postOrderMessage } from './api/order-api' +export type { OrderListResponse, OrderDetailResponse } from './api/order-api' diff --git a/client/src/entities/product/api/admin-product-api.ts b/client/src/entities/product/api/admin-product-api.ts new file mode 100755 index 0000000..7e3d1ec --- /dev/null +++ b/client/src/entities/product/api/admin-product-api.ts @@ -0,0 +1,108 @@ +import type { Category, Product } from '@/entities/product/model/types' +import { apiClient } from '@/shared/api/client' +import { apiBaseURL } from '@/shared/config' +import { ADMIN_UPLOAD_IMAGE_MAX_BYTES, formatAdminImageMaxSizeHint } from '@/shared/constants/upload-limits' + +export async function fetchAdminProducts(): Promise { + const { data } = await apiClient.get('admin/products') + return data +} + +export async function createProduct(body: { + title: string + slug?: string + shortDescription?: string | null + description?: string | null + quantity: number + materials?: string[] + priceCents: number + imageUrl?: string | null + imageUrls?: string[] + published: boolean + categoryId: string +}): Promise { + const { data } = await apiClient.post('admin/products', body) + return data +} + +export async function updateProduct( + id: string, + body: Partial<{ + title: string + slug: string + shortDescription: string | null + description: string | null + quantity: number + materials: string[] + priceCents: number + imageUrl: string | null + imageUrls: string[] + published: boolean + categoryId: string + }>, +): Promise { + const { data } = await apiClient.patch(`admin/products/${id}`, body) + return data +} + +export async function deleteProduct(id: string): Promise { + await apiClient.delete(`admin/products/${id}`) +} + +export async function createCategory(body: { name: string; slug?: string; sort?: number }): Promise { + const { data } = await apiClient.post('admin/categories', body) + return data +} + +export async function fetchAdminCategories(): Promise { + const { data } = await apiClient.get<{ items: Category[] }>('admin/categories') + return data.items +} + +export async function updateAdminCategory( + id: string, + body: Partial<{ name: string; slug: string; sort: number }>, +): Promise { + const { data } = await apiClient.patch(`admin/categories/${id}`, body) + return data +} + +export async function deleteAdminCategory(id: string): Promise { + await apiClient.delete(`admin/categories/${id}`) +} + +/** FormData: не задавать Content-Type вручную (boundary задаёт браузер). */ +export async function uploadAdminProductImages(files: FileList | readonly File[]): Promise { + const list = Array.from(files) + for (const f of list) { + if (f.size > ADMIN_UPLOAD_IMAGE_MAX_BYTES) { + throw new Error( + `Файл «${f.name}» слишком большой (максимум ${formatAdminImageMaxSizeHint()} на одно изображение).`, + ) + } + } + const fd = new FormData() + for (const f of list) { + fd.append('files', f, f.name) + } + const token = localStorage.getItem('craftshop_auth_token') + const base = apiBaseURL.replace(/\/$/, '') + const res = await fetch(`${base}/admin/uploads`, { + method: 'POST', + headers: token ? { Authorization: `Bearer ${token}` } : {}, + body: fd, + }) + const payload = (await res.json().catch(() => ({}))) as { urls?: string[]; error?: string } + if (!res.ok) { + if (res.status === 413) { + throw new Error( + 'Сервер отклонил файл как слишком большой (413). На проде часто лимит nginx: добавьте client_max_body_size для /api/ (см. docs/nginx-upload-limit.md). Проверьте также MAX_UPLOAD_BODY_BYTES в .env на сервере.', + ) + } + throw new Error(typeof payload.error === 'string' ? payload.error : `Ошибка загрузки (${res.status})`) + } + if (!Array.isArray(payload.urls)) { + throw new Error('Некорректный ответ сервера') + } + return payload.urls +} diff --git a/client/src/entities/product/api/product-api.ts b/client/src/entities/product/api/product-api.ts new file mode 100755 index 0000000..b456645 --- /dev/null +++ b/client/src/entities/product/api/product-api.ts @@ -0,0 +1,42 @@ +import type { Category, Product } from '@/entities/product/model/types' +import { apiClient } from '@/shared/api/client' + +export type PublicProductsResponse = { + items: Product[] + total: number + page: number + pageSize: number +} + +export async function fetchPublicProducts(params?: { + categorySlug?: string + q?: string + sort?: 'price_asc' | 'price_desc' | '' + page?: number + pageSize?: number + priceMinCents?: number + priceMaxCents?: number +}): Promise { + const { data } = await apiClient.get('products', { + params: { + categorySlug: params?.categorySlug || undefined, + q: params?.q || undefined, + sort: params?.sort || undefined, + page: params?.page || undefined, + pageSize: params?.pageSize || undefined, + priceMin: params?.priceMinCents ?? undefined, + priceMax: params?.priceMaxCents ?? undefined, + }, + }) + return data +} + +export async function fetchPublicProduct(id: string): Promise { + const { data } = await apiClient.get(`products/${id}`) + return data +} + +export async function fetchCategories(): Promise { + const { data } = await apiClient.get('categories') + return data +} diff --git a/client/src/entities/product/index.ts b/client/src/entities/product/index.ts new file mode 100755 index 0000000..98bfb2e --- /dev/null +++ b/client/src/entities/product/index.ts @@ -0,0 +1,2 @@ +export { fetchPublicProducts, fetchPublicProduct, fetchCategories } from './api/product-api' +export type { PublicProductsResponse } from './api/product-api' diff --git a/client/src/entities/product/model/types.ts b/client/src/entities/product/model/types.ts new file mode 100755 index 0000000..a0cf2d4 --- /dev/null +++ b/client/src/entities/product/model/types.ts @@ -0,0 +1,33 @@ +export type Category = { + id: string + name: string + slug: string + sort: number +} + +export type ProductReviewsSummary = { + approvedReviewCount: number + avgRating: number | null + latestApprovedText: string | null +} + +export type Product = { + id: string + title: string + slug: string + shortDescription: string | null + description: string | null + quantity: number + materials?: string[] + priceCents: number + imageUrl: string | null + imageUrls?: string[] // legacy-friendly (used only in admin payloads) + published: boolean + categoryId: string + createdAt: string + updatedAt: string + category?: Category + images?: { id: string; url: string; sort: number }[] + /** Для опубликованных товаров с публичного API. */ + reviewsSummary?: ProductReviewsSummary | null +} diff --git a/client/src/entities/product/ui/ProductCard.tsx b/client/src/entities/product/ui/ProductCard.tsx new file mode 100755 index 0000000..265397b --- /dev/null +++ b/client/src/entities/product/ui/ProductCard.tsx @@ -0,0 +1,264 @@ +import type { ReactNode } from 'react' +import * as React from 'react' +import { useCallback, useMemo, useRef } from 'react' +import { useMediaQuery } from '@mui/material' +import Box from '@mui/material/Box' +import Card from '@mui/material/Card' +import CardMedia from '@mui/material/CardMedia' +import Chip from '@mui/material/Chip' +import Stack from '@mui/material/Stack' +import Typography from '@mui/material/Typography' +import { useNavigate } from 'react-router-dom' +import { Swiper, SwiperSlide } from 'swiper/react' +import 'swiper/css' +import type { Product } from '@/entities/product/model/types' +import { formatPriceRub } from '@/shared/lib/format-price' +import { OptimizedImage } from '@/shared/ui/OptimizedImage' +import type { Swiper as SwiperType } from 'swiper/types' + +type Props = { product: Product; mediaHeight?: number; actions?: ReactNode } + +const ProductCardInner = ({ product, mediaHeight = 390, actions }: Props) => { + const navigate = useNavigate() + const isMobile = useMediaQuery('(max-width:600px)') + const swiperRef = useRef(null) + const imageUrls = useMemo(() => { + const fromImages = (product.images ?? []) + .slice() + .sort((a, b) => a.sort - b.sort) + .map((x) => x.url) + const urls = fromImages.length ? fromImages : product.imageUrl ? [product.imageUrl] : [] + return urls + }, [product.images, product.imageUrl]) + + const materials = (product.materials ?? []).slice(0, 3) + const moreMaterials = Math.max(0, (product.materials?.length ?? 0) - materials.length) + + const onMouseMove = (e: React.MouseEvent) => { + if (!swiperRef.current) return + if (imageUrls.length <= 1) return + const rect = (e.currentTarget as HTMLElement).getBoundingClientRect() + const rel = Math.min(1, Math.max(0, (e.clientX - rect.left) / rect.width)) + const idx = Math.min(imageUrls.length - 1, Math.floor(rel * imageUrls.length)) + swiperRef.current.slideTo(idx, 0) + } + + const goToProduct = useCallback(() => { + navigate(`/products/${product.id}`) + }, [navigate, product.id]) + + const stockLabel = product.quantity > 0 ? null : { label: 'Нет в наличии', color: 'default' as const } + + return ( + + + {imageUrls.length ? ( + + { + swiperRef.current = s + }} + style={{ width: '100%', height: '100%', overflow: 'hidden' }} + > + {imageUrls.map((url) => ( + + + + + + ))} + + + ) : ( + + + Нет фото + + + )} + + {stockLabel && ( + + )} + + + + + {product.category && ( + + )} + + + {product.title} + + + {(product.materials?.length ?? 0) > 0 && ( + + {materials.map((m) => ( + + ))} + {moreMaterials > 0 && ( + + )} + + )} + + + {product.shortDescription ?? 'Описание появится позже.'} + + + + + + {formatPriceRub(product.priceCents)} + + {actions} + + + + ) +} + +export const ProductCard = React.memo(ProductCardInner, (prev, next) => { + return prev.product.id === next.product.id && prev.mediaHeight === next.mediaHeight && prev.actions === next.actions +}) diff --git a/client/src/entities/review/api/admin-review-api.ts b/client/src/entities/review/api/admin-review-api.ts new file mode 100755 index 0000000..b41c1ed --- /dev/null +++ b/client/src/entities/review/api/admin-review-api.ts @@ -0,0 +1,32 @@ +import { apiClient } from '@/shared/api/client' + +export type AdminReview = { + id: string + rating: number + text: string | null + status: string + createdAt: string + moderatedAt: string | null + user: { id: string; email: string; displayName: string | null } + product: { id: string; title: string } +} + +export type AdminReviewsListResponse = { + items: AdminReview[] + total: number + page: number + pageSize: number +} + +export async function fetchAdminReviews(params?: { + status?: string + page?: number + pageSize?: number +}): Promise { + const { data } = await apiClient.get('admin/reviews', { params }) + return data +} + +export async function moderateReview(id: string, action: 'approve' | 'reject'): Promise { + await apiClient.patch(`admin/reviews/${id}`, { action }) +} diff --git a/client/src/entities/review/api/reviews-api.ts b/client/src/entities/review/api/reviews-api.ts new file mode 100755 index 0000000..b7aefc5 --- /dev/null +++ b/client/src/entities/review/api/reviews-api.ts @@ -0,0 +1,78 @@ +import { apiClient } from '@/shared/api/client' +import { OTHER_UPLOAD_MAX_FILE_BYTES, formatOtherUploadMaxSizeHint } from '@/shared/constants/upload-limits' + +export async function postProductReview( + productId: string, + body: { rating: number; text?: string | null; imageUrl?: string | null }, +): Promise { + await apiClient.post(`products/${productId}/reviews`, body) +} + +export async function uploadReviewImage(file: File): Promise<{ url: string }> { + if (file.size > OTHER_UPLOAD_MAX_FILE_BYTES) { + throw new Error(`Файл «${file.name}» слишком большой (максимум ${formatOtherUploadMaxSizeHint()}).`) + } + + const fd = new FormData() + fd.append('file', file, file.name) + const { data } = await apiClient.post<{ url: string }>('reviews/upload-image', fd) + return data +} + +export type PublicReviewFeedItem = { + id: string + rating: number + text: string | null + imageUrl: string | null + createdAt: string + authorId: string + authorDisplay: string + authorAvatar?: string | null + authorAvatarStyle?: string | null + product: { + id: string + title: string + published: boolean + slug: string + } +} + +export type PublicReviewsLatestResponse = { + items: PublicReviewFeedItem[] +} + +export async function fetchLatestApprovedReviews(limit = 5): Promise { + const { data } = await apiClient.get('reviews/latest', { + params: { limit }, + }) + return data +} + +export type PublicProductReviewItem = { + id: string + rating: number + text: string | null + imageUrl: string | null + createdAt: string + authorId: string + authorDisplay: string + authorAvatar?: string | null + authorAvatarStyle?: string | null +} + +export type PublicProductReviewsResponse = { + items: PublicProductReviewItem[] + total: number + page: number + pageSize: number +} + +export async function fetchPublicProductReviews( + productId: string, + params?: { page?: number; pageSize?: number }, +): Promise { + const { data } = await apiClient.get(`products/${productId}/reviews`, { + params: { page: params?.page, pageSize: params?.pageSize }, + }) + return data +} diff --git a/client/src/entities/review/index.ts b/client/src/entities/review/index.ts new file mode 100755 index 0000000..48f31fd --- /dev/null +++ b/client/src/entities/review/index.ts @@ -0,0 +1,12 @@ +export { + postProductReview, + uploadReviewImage, + fetchLatestApprovedReviews, + fetchPublicProductReviews, +} from './api/reviews-api' +export type { + PublicReviewFeedItem, + PublicReviewsLatestResponse, + PublicProductReviewItem, + PublicProductReviewsResponse, +} from './api/reviews-api' diff --git a/client/src/entities/test-checklist/api/test-checklist-api.ts b/client/src/entities/test-checklist/api/test-checklist-api.ts new file mode 100755 index 0000000..5ed7fb7 --- /dev/null +++ b/client/src/entities/test-checklist/api/test-checklist-api.ts @@ -0,0 +1,40 @@ +import { apiClient } from '@/shared/api/client' + +export type ChecklistResultDto = { + passed: boolean + comment: string | null + checkedAt: string +} + +export type TestChecklistResponse = { + results: Record +} + +export type UpdateChecklistItemResponse = { + itemKey: string + passed: boolean + comment: string | null + checkedAt: string +} + +export async function fetchTestChecklistResults(): Promise { + const { data } = await apiClient.get('admin/test-checklist') + return data +} + +export async function updateTestChecklistItem( + itemKey: string, + passed: boolean, + comment?: string | null, +): Promise { + const { data } = await apiClient.patch<{ result: UpdateChecklistItemResponse }>('admin/test-checklist', { + itemKey, + passed, + comment: passed ? null : (comment ?? null), + }) + return data.result +} + +export async function resetTestChecklist(): Promise { + await apiClient.post('admin/test-checklist/reset') +} diff --git a/client/src/entities/user/api/address-api.ts b/client/src/entities/user/api/address-api.ts new file mode 100755 index 0000000..d79656b --- /dev/null +++ b/client/src/entities/user/api/address-api.ts @@ -0,0 +1,49 @@ +import type { ShippingAddress } from '@/entities/user/model/types' +import { apiClient } from '@/shared/api/client' + +export type AddressesListResponse = { items: ShippingAddress[] } + +export async function fetchMyAddresses(): Promise { + const { data } = await apiClient.get('me/addresses') + return data +} + +export async function createMyAddress(body: { + label?: string | null + recipientName: string + recipientPhone: string + addressLine: string + comment?: string | null + lat: number + lng: number + isDefault?: boolean +}): Promise<{ item: ShippingAddress }> { + const { data } = await apiClient.post<{ item: ShippingAddress }>('me/addresses', body) + return data +} + +export async function updateMyAddress( + id: string, + body: Partial<{ + label: string | null + recipientName: string + recipientPhone: string + addressLine: string + comment: string | null + lat: number + lng: number + isDefault: boolean + }>, +): Promise<{ item: ShippingAddress }> { + const { data } = await apiClient.patch<{ item: ShippingAddress }>(`me/addresses/${id}`, body) + return data +} + +export async function deleteMyAddress(id: string): Promise { + await apiClient.delete(`me/addresses/${id}`) +} + +export async function setMyAddressDefault(id: string): Promise<{ item: ShippingAddress }> { + const { data } = await apiClient.post<{ item: ShippingAddress }>(`me/addresses/${id}/default`) + return data +} diff --git a/client/src/entities/user/api/messages-api.ts b/client/src/entities/user/api/messages-api.ts new file mode 100755 index 0000000..8f24233 --- /dev/null +++ b/client/src/entities/user/api/messages-api.ts @@ -0,0 +1,24 @@ +import { apiClient } from '@/shared/api/client' + +export async function fetchUnreadMessageCount(): Promise<{ count: number }> { + const { data } = await apiClient.get<{ count: number }>('me/messages/unread-count') + return data +} + +export async function markOrderMessagesRead(orderId: string): Promise { + await apiClient.post(`me/orders/${orderId}/messages/read`) +} + +export type ConversationSummary = { + orderId: string + status: string + deliveryType: 'delivery' | 'pickup' + lastMessageAt: string + preview: string + unreadCount: number +} + +export async function fetchMyConversations(): Promise<{ items: ConversationSummary[] }> { + const { data } = await apiClient.get<{ items: ConversationSummary[] }>('me/conversations') + return data +} diff --git a/client/src/entities/user/api/user-api.ts b/client/src/entities/user/api/user-api.ts new file mode 100755 index 0000000..60d3076 --- /dev/null +++ b/client/src/entities/user/api/user-api.ts @@ -0,0 +1,45 @@ +import type { AdminUser } from '@/entities/user/model/types' +import { apiClient } from '@/shared/api/client' + +export type AdminUsersListResponse = { + items: AdminUser[] + total: number + page: number + pageSize: number +} + +export async function fetchAdminUsers(params?: { + q?: string + page?: number + pageSize?: number +}): Promise { + const { data } = await apiClient.get('admin/users', { params }) + return data +} + +export async function createAdminUser(body: { email: string; displayName?: string | null }): Promise { + const { data } = await apiClient.post('admin/users', body) + return data +} + +export async function updateAdminUser( + id: string, + body: Partial<{ email: string; displayName: string | null }>, +): Promise { + const { data } = await apiClient.patch(`admin/users/${id}`, body) + return data +} + +export type AdminAvatarResponse = { + avatar: string | null + avatarStyle: string | null +} + +export async function fetchAdminAvatar(): Promise { + const { data } = await apiClient.get('admin/avatar') + return data +} + +export async function deleteAdminUser(id: string): Promise { + await apiClient.delete(`admin/users/${id}`) +} diff --git a/client/src/entities/user/index.ts b/client/src/entities/user/index.ts new file mode 100755 index 0000000..cfb7e7f --- /dev/null +++ b/client/src/entities/user/index.ts @@ -0,0 +1,10 @@ +export type { AdminUser, ShippingAddress } from './model/types' +export { fetchAdminUsers, createAdminUser, updateAdminUser, deleteAdminUser } from './api/user-api' +export type { AdminUsersListResponse } from './api/user-api' +export { + fetchMyAddresses, + createMyAddress, + updateMyAddress, + deleteMyAddress, + setMyAddressDefault, +} from './api/address-api' diff --git a/client/src/entities/user/model/types.ts b/client/src/entities/user/model/types.ts new file mode 100755 index 0000000..2ff1b45 --- /dev/null +++ b/client/src/entities/user/model/types.ts @@ -0,0 +1,23 @@ +export type AdminUser = { + id: string + email: string + displayName: string | null + avatar?: string | null + avatarStyle?: string | null + createdAt: string + updatedAt: string +} + +export type ShippingAddress = { + id: string + label: string | null + recipientName: string + recipientPhone: string + addressLine: string + comment: string | null + lat: number + lng: number + isDefault: boolean + createdAt: string + updatedAt: string +} diff --git a/client/src/features/address-form/index.ts b/client/src/features/address-form/index.ts new file mode 100755 index 0000000..da37f3c --- /dev/null +++ b/client/src/features/address-form/index.ts @@ -0,0 +1,2 @@ +export { AddressFormDialog } from './ui/AddressFormDialog' +export type { AddressFormValues } from './ui/AddressFormDialog' diff --git a/client/src/features/address-form/ui/AddressFormDialog.tsx b/client/src/features/address-form/ui/AddressFormDialog.tsx new file mode 100755 index 0000000..4af5d36 --- /dev/null +++ b/client/src/features/address-form/ui/AddressFormDialog.tsx @@ -0,0 +1,127 @@ +import Button from '@mui/material/Button' +import Dialog from '@mui/material/Dialog' +import DialogActions from '@mui/material/DialogActions' +import DialogContent from '@mui/material/DialogContent' +import DialogTitle from '@mui/material/DialogTitle' +import FormControlLabel from '@mui/material/FormControlLabel' +import Stack from '@mui/material/Stack' +import Switch from '@mui/material/Switch' +import TextField from '@mui/material/TextField' +import { Controller, type UseFormReturn } from 'react-hook-form' +import { AddressMapPicker } from '@/features/address-map-picker' + +export type AddressFormValues = { + label: string + recipientName: string + recipientPhone: string + addressLine: string + comment: string + lat: number | null + lng: number | null + isDefault: boolean +} + +export function AddressFormDialog({ + open, + onClose, + editing, + form, + onSubmit, + isPending, +}: { + open: boolean + onClose: () => void + editing: boolean + form: UseFormReturn + onSubmit: () => void + isPending: boolean +}) { + return ( + + {editing ? 'Редактировать адрес' : 'Новый адрес'} + + + } + /> + + } + /> + } + /> + + + } + /> + } + /> + + ( + ( + { + latField.onChange(v.lat) + lngField.onChange(v.lng) + if (v.addressLine) form.setValue('addressLine', v.addressLine, { shouldDirty: true }) + }} + /> + )} + /> + )} + /> + + ( + field.onChange(v)} />} + label="Адрес по умолчанию" + /> + )} + /> + + + + + + + + ) +} diff --git a/client/src/features/address-map-picker/api/map-geocoding.ts b/client/src/features/address-map-picker/api/map-geocoding.ts new file mode 100755 index 0000000..e448a4e --- /dev/null +++ b/client/src/features/address-map-picker/api/map-geocoding.ts @@ -0,0 +1,28 @@ +import type { LatLng, NominatimItem } from '../model/types' + +export async function reverseGeocode(pos: LatLng): Promise { + const url = new URL('https://nominatim.openstreetmap.org/reverse') + url.searchParams.set('format', 'jsonv2') + url.searchParams.set('lat', String(pos.lat)) + url.searchParams.set('lon', String(pos.lng)) + url.searchParams.set('accept-language', 'ru') + const res = await fetch(url.toString(), { headers: { 'User-Agent': 'craftshop-demo' } }) + if (!res.ok) return null + const data = (await res.json()) as { display_name?: string } + return data.display_name ? String(data.display_name) : null +} + +export async function searchPlaces(q: string, signal?: AbortSignal): Promise { + const url = new URL('https://nominatim.openstreetmap.org/search') + url.searchParams.set('format', 'jsonv2') + url.searchParams.set('q', q) + url.searchParams.set('accept-language', 'ru') + url.searchParams.set('limit', '5') + const res = await fetch(url.toString(), { + headers: { 'User-Agent': 'craftshop-demo' }, + signal, + }) + if (!res.ok) return [] + const data = (await res.json()) as NominatimItem[] + return Array.isArray(data) ? data : [] +} diff --git a/client/src/features/address-map-picker/index.ts b/client/src/features/address-map-picker/index.ts new file mode 100755 index 0000000..1a7d1b6 --- /dev/null +++ b/client/src/features/address-map-picker/index.ts @@ -0,0 +1 @@ +export { AddressMapPicker } from './ui/AddressMapPicker' diff --git a/client/src/features/address-map-picker/model/types.ts b/client/src/features/address-map-picker/model/types.ts new file mode 100755 index 0000000..0661142 --- /dev/null +++ b/client/src/features/address-map-picker/model/types.ts @@ -0,0 +1,3 @@ +export type NominatimItem = { display_name: string; lat: string; lon: string } + +export type LatLng = { lat: number; lng: number } diff --git a/client/src/features/address-map-picker/ui/AddressMapPicker.tsx b/client/src/features/address-map-picker/ui/AddressMapPicker.tsx new file mode 100755 index 0000000..4a2b769 --- /dev/null +++ b/client/src/features/address-map-picker/ui/AddressMapPicker.tsx @@ -0,0 +1,144 @@ +import { useEffect, useMemo, useRef, useState } from 'react' +import Box from '@mui/material/Box' +import Button from '@mui/material/Button' +import CircularProgress from '@mui/material/CircularProgress' +import List from '@mui/material/List' +import ListItemButton from '@mui/material/ListItemButton' +import ListItemText from '@mui/material/ListItemText' +import Stack from '@mui/material/Stack' +import TextField from '@mui/material/TextField' +import Typography from '@mui/material/Typography' +import { reverseGeocode, searchPlaces } from '../api/map-geocoding' +import { MapPickerMap } from './MapPickerMap' +import type { LatLng, NominatimItem } from '../model/types' + +export function AddressMapPicker(props: { + value: { lat: number; lng: number } | null + onChange: (v: { lat: number; lng: number; addressLine?: string | null }) => void +}) { + const { value, onChange } = props + const [q, setQ] = useState('') + const [searching, setSearching] = useState(false) + const [results, setResults] = useState([]) + const [hint, setHint] = useState(null) + const abortRef = useRef(null) + const lastQueryRef = useRef('') + const lastRequestAtRef = useRef(0) + + const qTrimmed = q.trim() + const visibleResults = qTrimmed.length >= 3 ? results : [] + + const center = useMemo(() => { + if (value) return { lat: value.lat, lng: value.lng } + return { lat: 55.751244, lng: 37.618423 } + }, [value]) + + const pick = async (pos: LatLng) => { + setHint(null) + onChange({ lat: pos.lat, lng: pos.lng }) + try { + const addr = await reverseGeocode(pos) + if (addr) { + setHint(addr) + onChange({ lat: pos.lat, lng: pos.lng, addressLine: addr }) + } + } catch (err) { + console.warn('[address-map-picker] Failed to reverse geocode', err) + } + } + + useEffect(() => { + const s = qTrimmed + if (s.length < 3) { + return + } + + const t = window.setTimeout(async () => { + const now = Date.now() + if (now - lastRequestAtRef.current < 900) return + if (s === lastQueryRef.current) return + + lastQueryRef.current = s + lastRequestAtRef.current = now + + abortRef.current?.abort() + const ac = new AbortController() + abortRef.current = ac + + setSearching(true) + try { + setResults(await searchPlaces(s, ac.signal)) + } catch (e) { + if ((e as { name?: string })?.name !== 'AbortError') { + setResults([]) + } + } finally { + setSearching(false) + } + }, 450) + + return () => { + window.clearTimeout(t) + } + }, [qTrimmed]) + + return ( + + Выбор на карте + + + setQ(e.target.value)} fullWidth /> + + + + {visibleResults.length > 0 && ( + + {visibleResults.map((r) => ( + { + const lat = Number(r.lat) + const lng = Number(r.lon) + if (!Number.isFinite(lat) || !Number.isFinite(lng)) return + void pick({ lat, lng }) + }} + > + + + ))} + + )} + + + + + {hint && ( + + Подсказка адреса: {hint} + + )} + + + ) +} diff --git a/client/src/features/address-map-picker/ui/MapPickerMap.tsx b/client/src/features/address-map-picker/ui/MapPickerMap.tsx new file mode 100755 index 0000000..e11cfb2 --- /dev/null +++ b/client/src/features/address-map-picker/ui/MapPickerMap.tsx @@ -0,0 +1,177 @@ +import { useEffect, useRef, useState } from 'react' +import MyLocationOutlinedIcon from '@mui/icons-material/MyLocationOutlined' +import Box from '@mui/material/Box' +import CircularProgress from '@mui/material/CircularProgress' +import IconButton from '@mui/material/IconButton' +import Tooltip from '@mui/material/Tooltip' +import Map, { Marker, type MapMouseEvent, type MapRef } from 'react-map-gl/maplibre' +import { reverseGeocode } from '../api/map-geocoding' +import type { LatLng } from '../model/types' +import type * as maplibregl from 'maplibre-gl' + +let maplibreglPromise: Promise | null = null + +function loadMaplibre() { + if (!maplibreglPromise) { + maplibreglPromise = Promise.all([import('maplibre-gl'), import('maplibre-gl/dist/maplibre-gl.css')]).then( + ([mod]) => mod, + ) + } + return maplibreglPromise +} + +type MapPickerMapProps = { + value: { lat: number; lng: number } | null + onChange: (v: { lat: number; lng: number; addressLine?: string | null }) => void + center: { lat: number; lng: number } +} + +export function MapPickerMap({ value, onChange, center }: MapPickerMapProps) { + const mapRef = useRef(null) + const [maplibre, setMaplibre] = useState(null) + const [loading, setLoading] = useState(true) + const [locating, setLocating] = useState(false) + + useEffect(() => { + let cancelled = false + loadMaplibre().then((mod) => { + if (!cancelled) { + setMaplibre(mod) + setLoading(false) + } + }) + return () => { + cancelled = true + } + }, []) + + const pick = async (pos: LatLng) => { + onChange({ lat: pos.lat, lng: pos.lng }) + try { + const addr = await reverseGeocode(pos) + if (addr) { + onChange({ lat: pos.lat, lng: pos.lng, addressLine: addr }) + } + } catch (err) { + console.warn('[map-picker] Failed to reverse geocode', err) + } + } + + if (loading || !maplibre) { + return ( + + + + ) + } + + return ( + + { + const { lng, lat } = e.lngLat + void pick({ lat, lng }) + }} + > + {value && ( + { + const { lng, lat } = e.lngLat + void pick({ lat, lng }) + }} + anchor="bottom" + > + + + )} + + + + + { + if (!('geolocation' in navigator)) return + setLocating(true) + navigator.geolocation.getCurrentPosition( + (pos) => { + const lat = pos.coords.latitude + const lng = pos.coords.longitude + mapRef.current?.flyTo({ center: [lng, lat], zoom: 15, duration: 800 }) + void pick({ lat, lng }) + setLocating(false) + }, + () => { + setLocating(false) + }, + { enableHighAccuracy: true, timeout: 8000, maximumAge: 60_000 }, + ) + }} + sx={{ + position: 'absolute', + top: 10, + right: 10, + bgcolor: 'background.paper', + border: 1, + borderColor: 'divider', + boxShadow: 2, + '&:hover': { bgcolor: 'background.paper' }, + }} + aria-label="Моё местоположение" + > + {locating ? : } + + + + + ) +} diff --git a/client/src/features/auth-code/index.ts b/client/src/features/auth-code/index.ts new file mode 100755 index 0000000..f5f22a4 --- /dev/null +++ b/client/src/features/auth-code/index.ts @@ -0,0 +1 @@ +export { AuthCodeForm } from './ui/AuthCodeForm' diff --git a/client/src/features/auth-code/ui/AuthCodeForm.test.tsx b/client/src/features/auth-code/ui/AuthCodeForm.test.tsx new file mode 100755 index 0000000..198ca02 --- /dev/null +++ b/client/src/features/auth-code/ui/AuthCodeForm.test.tsx @@ -0,0 +1,67 @@ +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render, screen, fireEvent, waitFor } from '@testing-library/react' +import { MemoryRouter } from 'react-router-dom' +import { describe, expect, it, vi } from 'vitest' +import { AuthCodeForm } from '../ui/AuthCodeForm' + +vi.mock('@/shared/api/client', () => ({ apiClient: { post: vi.fn() } })) +vi.mock('@/shared/model/auth', () => ({ tokenSet: vi.fn() })) + +function renderForm() { + const qc = new QueryClient({ defaultOptions: { queries: { retry: false } } }) + const onSuccess = vi.fn() + return render( + + + + + , + ) +} + +describe('AuthCodeForm', () => { + it('renders email field, code field, and buttons', () => { + renderForm() + expect(screen.getByLabelText(/Email/i)).toBeTruthy() + expect(screen.getByLabelText(/Код/i)).toBeTruthy() + expect(screen.getByRole('button', { name: 'Отправить код' })).toBeTruthy() + expect(screen.getByRole('button', { name: 'Войти' })).toBeTruthy() + }) + + it('disables send button when email is empty', () => { + renderForm() + expect(screen.getByRole('button', { name: 'Отправить код' })).toBeDisabled() + }) + + it('disables login button when code.length !== 6', () => { + renderForm() + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Код/i), { target: { value: '123' } }) + expect(screen.getByRole('button', { name: 'Войти' })).toBeDisabled() + }) + + it('enables login button when code is 6 digits', async () => { + renderForm() + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Код/i), { target: { value: '123456' } }) + await waitFor(() => { + expect(screen.getByRole('button', { name: 'Войти' })).not.toBeDisabled() + }) + }) + + it('calls onSuccess after successful verify', async () => { + const { apiClient } = await import('@/shared/api/client') + const { tokenSet } = await import('@/shared/model/auth') + vi.mocked(apiClient.post).mockResolvedValue({ data: { token: 'test-token' } } as never) + renderForm() + + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Код/i), { target: { value: '123456' } }) + fireEvent.click(screen.getByRole('button', { name: 'Войти' })) + + expect(screen.getByRole('button', { name: 'Войти' })).not.toBeDisabled() + await waitFor(() => { + expect(tokenSet).toHaveBeenCalledWith('test-token') + }) + }) +}) diff --git a/client/src/features/auth-code/ui/AuthCodeForm.tsx b/client/src/features/auth-code/ui/AuthCodeForm.tsx new file mode 100755 index 0000000..7287bcb --- /dev/null +++ b/client/src/features/auth-code/ui/AuthCodeForm.tsx @@ -0,0 +1,115 @@ +import Button from '@mui/material/Button' +import InputAdornment from '@mui/material/InputAdornment' +import Link from '@mui/material/Link' +import Stack from '@mui/material/Stack' +import TextField from '@mui/material/TextField' +import Typography from '@mui/material/Typography' +import { useMutation } from '@tanstack/react-query' +import { Mail } from 'lucide-react' +import { useForm } from 'react-hook-form' +import { Link as RouterLink } from 'react-router-dom' +import { apiClient } from '@/shared/api/client' +import { getApiErrorMessage } from '@/shared/lib/get-api-error-message' +import { tokenSet } from '@/shared/model/auth' + +type AuthResponse = { + token: string + user: { + id: string + email: string + displayName?: string | null + avatar?: string | null + avatarStyle?: string | null + } +} + +type FormValues = { + email: string + code: string +} + +type Props = { + onSuccess: () => void +} + +export function AuthCodeForm({ onSuccess }: Props) { + const { register, watch } = useForm({ + defaultValues: { email: '', code: '' }, + mode: 'onChange', + }) + + const email = watch('email') + const code = watch('code') + + const requestCodeMutation = useMutation({ + mutationFn: async () => { + await apiClient.post('auth/request-code', { email }) + }, + }) + + const verifyCodeMutation = useMutation({ + mutationFn: async () => { + const { data } = await apiClient.post('auth/verify-code', { email, code }) + tokenSet(data.token) + }, + onSuccess, + }) + + return ( + + + + + ), + }, + }} + /> + + + + + + + {(requestCodeMutation.error || verifyCodeMutation.error) && ( + + )} + + + Нажимая «Войти», вы принимаете{' '} + + пользовательское соглашение + {' '} + и{' '} + + политику конфиденциальности + + . + + + ) +} diff --git a/client/src/features/auth-forgot/index.ts b/client/src/features/auth-forgot/index.ts new file mode 100755 index 0000000..81d088a --- /dev/null +++ b/client/src/features/auth-forgot/index.ts @@ -0,0 +1 @@ +export { AuthForgotForm } from './ui/AuthForgotForm' diff --git a/client/src/features/auth-forgot/ui/AuthForgotForm.tsx b/client/src/features/auth-forgot/ui/AuthForgotForm.tsx new file mode 100755 index 0000000..f769891 --- /dev/null +++ b/client/src/features/auth-forgot/ui/AuthForgotForm.tsx @@ -0,0 +1,145 @@ +import { useState } from 'react' +import Button from '@mui/material/Button' +import InputAdornment from '@mui/material/InputAdornment' +import Stack from '@mui/material/Stack' +import TextField from '@mui/material/TextField' +import Typography from '@mui/material/Typography' +import { useMutation } from '@tanstack/react-query' +import { Lock, Mail } from 'lucide-react' +import { useForm } from 'react-hook-form' +import { apiClient } from '@/shared/api/client' +import { getApiErrorMessage } from '@/shared/lib/get-api-error-message' + +type Step = 'request' | 'reset' + +type FormValues = { + email: string + code: string + newPassword: string + passwordConfirm: string +} + +type Props = { + onBack: () => void +} + +export function AuthForgotForm({ onBack }: Props) { + const [step, setStep] = useState('request') + + const { register, watch } = useForm({ + defaultValues: { email: '', code: '', newPassword: '', passwordConfirm: '' }, + mode: 'onChange', + }) + + const email = watch('email') + const code = watch('code') + const newPassword = watch('newPassword') + const passwordConfirm = watch('passwordConfirm') + + const forgotCodeMutation = useMutation({ + mutationFn: async () => { + await apiClient.post('auth/forgot-password', { email }) + }, + onSuccess: () => setStep('reset'), + }) + + const resetPasswordMutation = useMutation({ + mutationFn: async () => { + await apiClient.post('auth/reset-password', { email, code, newPassword }) + }, + }) + + const passwordError = newPassword && passwordConfirm && newPassword !== passwordConfirm ? 'Пароли не совпадают' : null + + return ( + + + {step === 'request' + ? 'Введите email, на который будет отправлен код для сброса пароля' + : 'Введите код и новый пароль'} + + + + + + ), + }, + }} + /> + + {step === 'reset' && ( + <> + + + + + ), + }, + }} + /> + + + )} + + {step === 'request' ? ( + + ) : ( + + )} + + + + {(forgotCodeMutation.error || resetPasswordMutation.error) && ( + + )} + + ) +} diff --git a/client/src/features/auth-oauth/__tests__/OAuthButtons.test.tsx b/client/src/features/auth-oauth/__tests__/OAuthButtons.test.tsx new file mode 100755 index 0000000..d095761 --- /dev/null +++ b/client/src/features/auth-oauth/__tests__/OAuthButtons.test.tsx @@ -0,0 +1,19 @@ +import { render, screen } from '@testing-library/react' +import { describe, it, expect } from 'vitest' +import { OAuthButtons } from '../ui/OAuthButtons' + +describe('OAuthButtons', () => { + it('renders Yandex and VK buttons', () => { + render() + expect(screen.getByText('Войти через Яндекс ID')).toBeDefined() + expect(screen.getByText('Войти через VK ID')).toBeDefined() + }) + + it('buttons have correct href', () => { + render() + const yaBtn = screen.getByText('Войти через Яндекс ID').closest('a') + const vkBtn = screen.getByText('Войти через VK ID').closest('a') + expect(yaBtn?.getAttribute('href')).toContain('/auth/oauth/yandex') + expect(vkBtn?.getAttribute('href')).toContain('/auth/oauth/vk') + }) +}) diff --git a/client/src/features/auth-oauth/index.ts b/client/src/features/auth-oauth/index.ts new file mode 100755 index 0000000..bbe2712 --- /dev/null +++ b/client/src/features/auth-oauth/index.ts @@ -0,0 +1 @@ +export { OAuthButtons } from './ui/OAuthButtons' diff --git a/client/src/features/auth-oauth/lib/oauth-providers.ts b/client/src/features/auth-oauth/lib/oauth-providers.ts new file mode 100755 index 0000000..2d1a510 --- /dev/null +++ b/client/src/features/auth-oauth/lib/oauth-providers.ts @@ -0,0 +1,24 @@ +import { oauthAuthorizeUrl } from '@/shared/lib/oauth-authorize-url' + +export type OAuthProvider = { + id: 'yandex' | 'vk' + label: string + color: string +} + +export const oauthProviders: OAuthProvider[] = [ + { + id: 'yandex', + label: 'Яндекс ID', + color: '#FC3F1D', + }, + { + id: 'vk', + label: 'VK ID', + color: '#0077FF', + }, +] + +export function getOAuthUrl(provider: 'yandex' | 'vk'): string { + return oauthAuthorizeUrl(provider) +} diff --git a/client/src/features/auth-oauth/ui/OAuthButtons.tsx b/client/src/features/auth-oauth/ui/OAuthButtons.tsx new file mode 100755 index 0000000..4a28d87 --- /dev/null +++ b/client/src/features/auth-oauth/ui/OAuthButtons.tsx @@ -0,0 +1,28 @@ +import Button from '@mui/material/Button' +import Stack from '@mui/material/Stack' +import { getOAuthUrl, oauthProviders } from '../lib/oauth-providers' + +export function OAuthButtons() { + return ( + + {oauthProviders.map((p) => ( + + ))} + + ) +} diff --git a/client/src/features/auth-password/index.ts b/client/src/features/auth-password/index.ts new file mode 100755 index 0000000..1cd0df0 --- /dev/null +++ b/client/src/features/auth-password/index.ts @@ -0,0 +1 @@ +export { AuthPasswordForm } from './ui/AuthPasswordForm' diff --git a/client/src/features/auth-password/ui/AuthPasswordForm.test.tsx b/client/src/features/auth-password/ui/AuthPasswordForm.test.tsx new file mode 100755 index 0000000..6596166 --- /dev/null +++ b/client/src/features/auth-password/ui/AuthPasswordForm.test.tsx @@ -0,0 +1,74 @@ +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render, screen, fireEvent, waitFor } from '@testing-library/react' +import { MemoryRouter } from 'react-router-dom' +import { describe, expect, it, vi } from 'vitest' +import { AuthPasswordForm } from '../ui/AuthPasswordForm' + +vi.mock('@/shared/api/client', () => ({ apiClient: { post: vi.fn() } })) +vi.mock('@/shared/model/auth', () => ({ tokenSet: vi.fn() })) + +function renderForm(isRegister: boolean) { + const qc = new QueryClient({ defaultOptions: { queries: { retry: false } } }) + const onSuccess = vi.fn() + return render( + + + + + , + ) +} + +describe('AuthPasswordForm', () => { + it('renders login button when isRegister=false', () => { + renderForm(false) + expect(screen.getByRole('button', { name: 'Войти' })).toBeTruthy() + expect(screen.getByText('Вход')).toBeTruthy() + }) + + it('renders register button and passwordConfirm when isRegister=true', () => { + renderForm(true) + expect(screen.getByRole('button', { name: 'Зарегистрироваться' })).toBeTruthy() + expect(screen.getByLabelText(/Подтверждение пароля/i)).toBeTruthy() + }) + + it('disables button when password < 8 chars', async () => { + const { apiClient } = await import('@/shared/api/client') + vi.mocked(apiClient.post).mockResolvedValue({} as never) + renderForm(true) + + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Пароль/i), { target: { value: '123' } }) + + await waitFor(() => { + expect(screen.getByRole('button', { name: 'Зарегистрироваться' })).toBeDisabled() + }) + }) + + it('shows error when passwords do not match', async () => { + renderForm(true) + + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Пароль/i), { target: { value: 'password123' } }) + fireEvent.change(screen.getByLabelText(/Подтверждение пароля/i), { target: { value: 'different' } }) + + await waitFor(() => { + expect(screen.getByText('Пароли не совпадают')).toBeTruthy() + }) + }) + + it('calls onSuccess after successful login', async () => { + const { apiClient } = await import('@/shared/api/client') + const { tokenSet } = await import('@/shared/model/auth') + vi.mocked(apiClient.post).mockResolvedValue({ data: { token: 'test-token' } } as never) + renderForm(false) + + fireEvent.change(screen.getByLabelText(/Email/i), { target: { value: 'test@test.com' } }) + fireEvent.change(screen.getByLabelText(/Пароль/i), { target: { value: 'password123' } }) + fireEvent.click(screen.getByRole('button', { name: 'Войти' })) + + await waitFor(() => { + expect(tokenSet).toHaveBeenCalledWith('test-token') + }) + }) +}) diff --git a/client/src/features/auth-password/ui/AuthPasswordForm.tsx b/client/src/features/auth-password/ui/AuthPasswordForm.tsx new file mode 100755 index 0000000..d5f8fec --- /dev/null +++ b/client/src/features/auth-password/ui/AuthPasswordForm.tsx @@ -0,0 +1,234 @@ +import Alert from '@mui/material/Alert' +import Box from '@mui/material/Box' +import Button from '@mui/material/Button' +import InputAdornment from '@mui/material/InputAdornment' +import Link from '@mui/material/Link' +import Stack from '@mui/material/Stack' +import TextField from '@mui/material/TextField' +import Typography from '@mui/material/Typography' +import { useMutation } from '@tanstack/react-query' +import { Lock, Mail } from 'lucide-react' +import { useForm } from 'react-hook-form' +import { Link as RouterLink } from 'react-router-dom' +import { apiClient } from '@/shared/api/client' +import { getApiErrorMessage } from '@/shared/lib/get-api-error-message' +import { tokenSet } from '@/shared/model/auth' + +type AuthResponse = { + token: string + user: { + id: string + email: string + displayName?: string | null + avatar?: string | null + avatarStyle?: string | null + } +} + +type FormValues = { + email: string + password: string + passwordConfirm: string + displayName: string +} + +type Props = { + isRegister: boolean + onRegisterChange: (v: boolean) => void + onSuccess: () => void +} + +const EMAIL_PATTERN = /^[^\s@]+@[^\s@]+\.[^\s@]+$/ + +export function AuthPasswordForm({ isRegister, onRegisterChange, onSuccess }: Props) { + const { + register, + handleSubmit, + watch, + formState: { errors }, + } = useForm({ + defaultValues: { email: '', password: '', passwordConfirm: '', displayName: '' }, + mode: 'onChange', + }) + + const password = watch('password') + + const loginMutation = useMutation({ + mutationFn: async (values: FormValues) => { + const { data } = await apiClient.post('auth/login', { + email: values.email, + password: values.password, + }) + tokenSet(data.token) + }, + onSuccess, + }) + + const registerMutation = useMutation({ + mutationFn: async (values: FormValues) => { + const { data } = await apiClient.post('auth/register', { + email: values.email, + password: values.password, + displayName: values.displayName || undefined, + }) + tokenSet(data.token) + }, + onSuccess, + }) + + const apiError = loginMutation.error || registerMutation.error + const apiErrorMessage = apiError ? getApiErrorMessage(apiError) : null + + const onSubmit = isRegister + ? handleSubmit((values) => registerMutation.mutate(values)) + : handleSubmit((values) => loginMutation.mutate(values)) + + const isPending = loginMutation.isPending || registerMutation.isPending + + return ( + + + + + + + + + + + ), + }, + }} + /> + + {isRegister && ( + + )} + + + + + ), + }, + }} + /> + + {isRegister && ( + value === password || 'Пароли не совпадают', + })} + fullWidth + error={Boolean(errors.passwordConfirm)} + helperText={errors.passwordConfirm?.message} + /> + )} + + {apiErrorMessage && ( + { + loginMutation.reset() + registerMutation.reset() + }} + > + {apiErrorMessage} + + )} + + {isRegister ? ( + + ) : ( + + )} + + + Нажимая «{isRegister ? 'Зарегистрироваться' : 'Войти'}», вы принимаете{' '} + + пользовательское соглашение + {' '} + и{' '} + + политику конфиденциальности + + . + + + + ) +} diff --git a/client/src/features/cart/add-to-cart/index.ts b/client/src/features/cart/add-to-cart/index.ts new file mode 100755 index 0000000..6c055a5 --- /dev/null +++ b/client/src/features/cart/add-to-cart/index.ts @@ -0,0 +1 @@ +export { AddToCartButton } from './ui/AddToCartButton' diff --git a/client/src/features/cart/add-to-cart/ui/AddToCartButton.tsx b/client/src/features/cart/add-to-cart/ui/AddToCartButton.tsx new file mode 100755 index 0000000..a860d2d --- /dev/null +++ b/client/src/features/cart/add-to-cart/ui/AddToCartButton.tsx @@ -0,0 +1,46 @@ +import Button from '@mui/material/Button' +import type { ButtonProps } from '@mui/material/Button' +import { useMutation, useQueryClient } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { addToCart } from '@/entities/cart/api/cart-api' +import { $user } from '@/shared/model/auth' +import { addNotification } from '@/shared/model/notification' + +type Props = { + productId: string + qty?: number + loggedOutLabel?: string +} & Omit + +export function AddToCartButton(props: Props) { + const { productId, qty = 1, loggedOutLabel = 'Войдите, чтобы купить', disabled, children, ...rest } = props + const qc = useQueryClient() + const user = useUnit($user) + + const addMut = useMutation({ + mutationFn: () => addToCart({ productId, qty }), + onSuccess: () => { + void qc.invalidateQueries({ queryKey: ['me', 'cart'] }) + addNotification({ + type: 'success', + message: 'Товар добавлен в корзину', + actionLabel: 'Перейти в корзину', + actionPath: '/cart', + }) + }, + }) + + return ( + + ) +} diff --git a/client/src/features/cart/add-to-cart/ui/__tests__/AddToCartButton.test.tsx b/client/src/features/cart/add-to-cart/ui/__tests__/AddToCartButton.test.tsx new file mode 100755 index 0000000..1dc3024 --- /dev/null +++ b/client/src/features/cart/add-to-cart/ui/__tests__/AddToCartButton.test.tsx @@ -0,0 +1,43 @@ +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render, screen, fireEvent } from '@testing-library/react' +import { describe, it, expect, vi, beforeEach } from 'vitest' +import * as notifications from '@/shared/model/notification' +import { AddToCartButton } from '../AddToCartButton' + +vi.mock('@/entities/cart/api/cart-api', () => ({ + addToCart: vi.fn(() => Promise.resolve()), +})) + +vi.mock('effector-react', async () => { + const actual = await vi.importActual('effector-react') + return { ...actual, useUnit: () => ({ id: '1', email: 'test@test.com' }) } +}) + +describe('AddToCartButton', () => { + const qc = new QueryClient({ defaultOptions: { queries: { retry: false } } }) + + beforeEach(() => { + vi.clearAllMocks() + qc.clear() + }) + + it('calls addNotification after successful add', async () => { + const spy = vi.spyOn(notifications, 'addNotification') + render( + + + , + ) + + fireEvent.click(screen.getByRole('button', { name: /в корзину/i })) + + await vi.waitFor(() => { + expect(spy).toHaveBeenCalledWith({ + type: 'success', + message: 'Товар добавлен в корзину', + actionLabel: 'Перейти в корзину', + actionPath: '/cart', + }) + }) + }) +}) diff --git a/client/src/features/cart/cart-badge/index.ts b/client/src/features/cart/cart-badge/index.ts new file mode 100755 index 0000000..436a567 --- /dev/null +++ b/client/src/features/cart/cart-badge/index.ts @@ -0,0 +1 @@ +export { CartBadge } from './ui/CartBadge' diff --git a/client/src/features/cart/cart-badge/ui/CartBadge.tsx b/client/src/features/cart/cart-badge/ui/CartBadge.tsx new file mode 100755 index 0000000..bf72c83 --- /dev/null +++ b/client/src/features/cart/cart-badge/ui/CartBadge.tsx @@ -0,0 +1,31 @@ +import Badge from '@mui/material/Badge' +import IconButton from '@mui/material/IconButton' +import Tooltip from '@mui/material/Tooltip' +import { ShoppingCart } from 'lucide-react' +import type { AuthUser } from '@/shared/model/auth' + +type Props = { + user: AuthUser | null + cartCount: number + onNavigate: (to: string) => void +} + +export function CartBadge({ user, cartCount, onNavigate }: Props) { + return ( + + { + if (!user) onNavigate('/auth') + else onNavigate('/cart') + }} + aria-label={user ? `Корзина (${cartCount})` : 'Авторизуйтесь для совершения покупок'} + > + + + + + + ) +} diff --git a/client/src/features/cart/toggle-cart-icon/index.ts b/client/src/features/cart/toggle-cart-icon/index.ts new file mode 100755 index 0000000..8a884df --- /dev/null +++ b/client/src/features/cart/toggle-cart-icon/index.ts @@ -0,0 +1 @@ +export { ToggleCartIcon } from './ui/ToggleCartIcon' diff --git a/client/src/features/cart/toggle-cart-icon/ui/ToggleCartIcon.tsx b/client/src/features/cart/toggle-cart-icon/ui/ToggleCartIcon.tsx new file mode 100755 index 0000000..f324de9 --- /dev/null +++ b/client/src/features/cart/toggle-cart-icon/ui/ToggleCartIcon.tsx @@ -0,0 +1,77 @@ +import IconButton from '@mui/material/IconButton' +import Tooltip from '@mui/material/Tooltip' +import { useMutation, useQueryClient } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { ShoppingCart } from 'lucide-react' +import { useNavigate } from 'react-router-dom' +import { addToCart, removeCartItem } from '@/entities/cart/api/cart-api' +import { useCartQuery } from '@/entities/cart/lib/use-cart-query' +import { $user } from '@/shared/model/auth' +import { addNotification } from '@/shared/model/notification' + +export function ToggleCartIcon(props: { + productId: string + size?: 'small' | 'medium' + disabledReason?: string | null +}) { + const { productId, size = 'small', disabledReason = null } = props + const user = useUnit($user) + const qc = useQueryClient() + const navigate = useNavigate() + + const cartQuery = useCartQuery() + + const existing = cartQuery.data?.items.find((x) => x.product.id === productId) ?? null + const inCart = Boolean(existing) + + const addMut = useMutation({ + mutationFn: () => addToCart({ productId, qty: 1 }), + onSuccess: () => { + void qc.invalidateQueries({ queryKey: ['me', 'cart'] }) + addNotification({ + type: 'success', + message: 'Товар добавлен в корзину', + actionLabel: 'Перейти в корзину', + actionPath: '/cart', + }) + }, + }) + + const removeMut = useMutation({ + mutationFn: () => removeCartItem(existing!.id), + onSuccess: () => void qc.invalidateQueries({ queryKey: ['me', 'cart'] }), + }) + + const disabled = Boolean(disabledReason) + const busy = addMut.isPending || removeMut.isPending + + const onClick = (e: React.MouseEvent) => { + e.preventDefault() + e.stopPropagation() + if (disabledReason) return + if (!user) { + navigate('/auth') + return + } + if (inCart) removeMut.mutate() + else addMut.mutate() + } + + const tooltip = disabledReason + ? disabledReason + : !user + ? 'Авторизуйтесь для совершения покупок' + : inCart + ? 'Убрать из корзины' + : 'В корзину' + + return ( + + + + {user ? inCart ? : : } + + + + ) +} diff --git a/client/src/features/cart/toggle-cart-icon/ui/__tests__/ToggleCartIcon.test.tsx b/client/src/features/cart/toggle-cart-icon/ui/__tests__/ToggleCartIcon.test.tsx new file mode 100755 index 0000000..0a4b908 --- /dev/null +++ b/client/src/features/cart/toggle-cart-icon/ui/__tests__/ToggleCartIcon.test.tsx @@ -0,0 +1,74 @@ +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render, screen, fireEvent } from '@testing-library/react' +import { MemoryRouter } from 'react-router-dom' +import { describe, it, expect, vi, beforeEach } from 'vitest' +import * as api from '@/entities/cart/api/cart-api' +import * as notifications from '@/shared/model/notification' +import { ToggleCartIcon } from '../ToggleCartIcon' + +vi.mock('@/entities/cart/api/cart-api', () => ({ + addToCart: vi.fn(() => Promise.resolve()), + fetchMyCart: vi.fn(() => Promise.resolve({ items: [] })), + removeCartItem: vi.fn(() => Promise.resolve()), +})) + +vi.mock('effector-react', async () => { + const actual = await vi.importActual('effector-react') + return { ...actual, useUnit: () => ({ id: '1', email: 'test@test.com' }) } +}) + +describe('ToggleCartIcon', () => { + const qc = new QueryClient({ defaultOptions: { queries: { retry: false } } }) + + beforeEach(() => { + vi.clearAllMocks() + qc.clear() + }) + + it('calls addNotification after successful add', async () => { + const spy = vi.spyOn(notifications, 'addNotification') + render( + + + + + , + ) + + fireEvent.click(screen.getByRole('button', { name: /в корзину/i })) + + await vi.waitFor(() => { + expect(spy).toHaveBeenCalledWith({ + type: 'success', + message: 'Товар добавлен в корзину', + actionLabel: 'Перейти в корзину', + actionPath: '/cart', + }) + }) + }) + + it('does not call addNotification on remove', async () => { + vi.mocked(api.fetchMyCart).mockResolvedValueOnce({ + items: [{ id: 'cart-1', qty: 1, product: { id: 'test-product' } as never }], + }) + const spy = vi.spyOn(notifications, 'addNotification') + + render( + + + + + , + ) + + await vi.waitFor(() => { + expect(screen.getByRole('button', { name: /убрать из корзины/i })).toBeInTheDocument() + }) + + fireEvent.click(screen.getByRole('button', { name: /убрать из корзины/i })) + + await vi.waitFor(() => { + expect(spy).not.toHaveBeenCalled() + }) + }) +}) diff --git a/client/src/features/order-chat/index.ts b/client/src/features/order-chat/index.ts new file mode 100755 index 0000000..cb5b17a --- /dev/null +++ b/client/src/features/order-chat/index.ts @@ -0,0 +1 @@ +export { OrderChat } from './ui/OrderChat' diff --git a/client/src/features/order-chat/ui/OrderChat.tsx b/client/src/features/order-chat/ui/OrderChat.tsx new file mode 100755 index 0000000..7346365 --- /dev/null +++ b/client/src/features/order-chat/ui/OrderChat.tsx @@ -0,0 +1,87 @@ +import { useState } from 'react' +import Box from '@mui/material/Box' +import Button from '@mui/material/Button' +import Stack from '@mui/material/Stack' +import Typography from '@mui/material/Typography' +import { useQuery } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { fetchAdminAvatar } from '@/entities/user/api/user-api' +import { $user } from '@/shared/model/auth' +import { ChatMessageBubble } from '@/shared/ui/ChatMessageBubble' +import { OrderMessageBody } from '@/shared/ui/OrderMessageBody' +import { RichTextMessageEditor } from '@/shared/ui/RichTextMessageEditor.lazy' +import { UserAvatar } from '@/shared/ui/UserAvatar' + +type Message = { + id: string + authorType: 'user' | 'admin' + text: string + attachmentUrl?: string | null + createdAt: string +} + +type Props = { + messages: Message[] + isPending: boolean + onSend: (text: string) => void +} + +export function OrderChat({ messages, isPending, onSend }: Props) { + const [text, setText] = useState('') + const canSend = text.replace(/<[^>]*>/g, ' ').trim().length > 0 + const currentUser = useUnit($user) + + const adminAvatarQuery = useQuery({ + queryKey: ['admin', 'avatar'], + queryFn: fetchAdminAvatar, + staleTime: 5 * 60 * 1000, + }) + + const handleSend = () => { + if (!canSend || isPending) return + onSend(text.trim()) + setText('') + } + + return ( + + + Чат по заказу + + + {messages.map((m) => { + const isAdminMsg = m.authorType === 'admin' + const adminAv = adminAvatarQuery.data + const avatarNode = isAdminMsg ? ( + + ) : currentUser ? ( + + ) : null + return ( + + + {isAdminMsg ? 'Админ' : 'Вы'} · {new Date(m.createdAt).toLocaleString()} + + + + ) + })} + {messages.length === 0 && Пока сообщений нет.} + + + + + + + + + + ) +} diff --git a/client/src/features/order-detail/index.ts b/client/src/features/order-detail/index.ts new file mode 100755 index 0000000..9e2f64c --- /dev/null +++ b/client/src/features/order-detail/index.ts @@ -0,0 +1,2 @@ +export { DeliveryFeeAdjustmentForm } from './ui/DeliveryFeeAdjustmentForm' +export { OrderDetailContent } from './ui/OrderDetailContent' diff --git a/client/src/features/order-detail/ui/DeliveryFeeAdjustmentForm.tsx b/client/src/features/order-detail/ui/DeliveryFeeAdjustmentForm.tsx new file mode 100755 index 0000000..cff8f64 --- /dev/null +++ b/client/src/features/order-detail/ui/DeliveryFeeAdjustmentForm.tsx @@ -0,0 +1,55 @@ +import { useState } from 'react' +import Button from '@mui/material/Button' +import Stack from '@mui/material/Stack' +import TextField from '@mui/material/TextField' +import { useMutation, useQueryClient } from '@tanstack/react-query' +import { patchAdminOrderDeliveryFee } from '@/entities/order/api/admin-order-api' +import { invalidateQueryKeys } from '@/shared/lib/invalidate-query-keys' + +export function DeliveryFeeAdjustmentForm({ + orderId, + deliveryFeeCents, +}: { + orderId: string + deliveryFeeCents: number +}) { + const qc = useQueryClient() + const [rub, setRub] = useState(() => String(deliveryFeeCents / 100)) + const feeMut = useMutation({ + mutationFn: () => patchAdminOrderDeliveryFee(orderId, Math.round(Number.parseFloat(rub) * 100)), + onSuccess: async () => { + await invalidateQueryKeys(qc, [ + ['admin', 'orders'], + ['admin', 'orders', 'detail'], + ['admin', 'orders', 'summary'], + ]) + }, + }) + + return ( + + setRub(e.target.value)} + slotProps={{ htmlInput: { min: 0, step: 1 } }} + sx={{ width: { xs: '100%', sm: 200 } }} + /> + + + ) +} diff --git a/client/src/features/order-detail/ui/OrderDetailContent.tsx b/client/src/features/order-detail/ui/OrderDetailContent.tsx new file mode 100755 index 0000000..ca19b9f --- /dev/null +++ b/client/src/features/order-detail/ui/OrderDetailContent.tsx @@ -0,0 +1,245 @@ +import { useMemo, useState } from 'react' +import Alert from '@mui/material/Alert' +import Box from '@mui/material/Box' +import Button from '@mui/material/Button' +import Link from '@mui/material/Link' +import Stack from '@mui/material/Stack' +import Typography from '@mui/material/Typography' +import { useMutation, useQueryClient } from '@tanstack/react-query' +import { useUnit } from 'effector-react' +import { Link as RouterLink } from 'react-router-dom' +import { postAdminOrderMessage, setAdminOrderStatus } from '@/entities/order/api/admin-order-api' +import type { AdminOrderDetailResponse } from '@/entities/order/api/admin-order-api' +import { deliveryCarrierLabelRu } from '@/shared/constants/delivery-carrier' +import { getAdminNextOrderStatuses } from '@/shared/constants/order' +import { formatPriceRub } from '@/shared/lib/format-price' +import { invalidateQueryKeys } from '@/shared/lib/invalidate-query-keys' +import { parseOrderAddressSnapshot } from '@/shared/lib/order-address-snapshot' +import { ORDER_STATUS_MAP } from '@/shared/lib/order-status-data' +import { $user } from '@/shared/model/auth' +import { ChatMessageBubble } from '@/shared/ui/ChatMessageBubble' +import { OrderMessageBody } from '@/shared/ui/OrderMessageBody' +import { RichTextMessageEditor } from '@/shared/ui/RichTextMessageEditor.lazy' +import { UserAvatar } from '@/shared/ui/UserAvatar' +import { DeliveryFeeAdjustmentForm } from './DeliveryFeeAdjustmentForm' + +export function OrderDetailContent({ detail, orderId }: { detail: AdminOrderDetailResponse['item']; orderId: string }) { + const qc = useQueryClient() + const [msg, setMsg] = useState('') + + const statusMut = useMutation({ + mutationFn: (next: string) => setAdminOrderStatus(orderId, next), + onSuccess: async () => { + await invalidateQueryKeys(qc, [ + ['admin', 'orders'], + ['admin', 'orders', 'detail'], + ['admin', 'orders', 'summary'], + ]) + }, + }) + + const msgMut = useMutation({ + mutationFn: () => postAdminOrderMessage(orderId, msg.trim()), + onSuccess: async () => { + setMsg('') + await invalidateQueryKeys(qc, [['admin', 'orders', 'detail']]) + }, + }) + + const deliverySnapshot = useMemo( + () => (detail.deliveryType === 'delivery' ? parseOrderAddressSnapshot(detail.addressSnapshotJson) : null), + [detail], + ) + + const nextStatuses = useMemo( + () => getAdminNextOrderStatuses(detail.status, detail.deliveryType ?? 'delivery'), + [detail], + ) + + const canSendMessage = msg.replace(/<[^>]*>/g, ' ').trim().length > 0 + const currentUser = useUnit($user) + + return ( + + + #{detail.id.slice(-8)} · {detail.user.email} · {ORDER_STATUS_MAP[detail.status] ?? detail.status} ·{' '} + {formatPriceRub(detail.totalCents)} + + + Получение: {detail.deliveryType === 'pickup' ? 'самовывоз' : 'доставка'} + {(detail.paymentMethod ?? 'online') === 'on_pickup' ? ' · оплата при получении' : ' · онлайн-оплата'} + {detail.deliveryType === 'delivery' && deliveryCarrierLabelRu(detail.deliveryCarrier) && ( + <> · служба: {deliveryCarrierLabelRu(detail.deliveryCarrier)} + )} + + + {detail.deliveryType === 'delivery' && ( + + + Адрес и получатель (на момент заказа) + + {deliverySnapshot ? ( + + {deliverySnapshot.label?.trim() && ( + + Метка: {deliverySnapshot.label} + + )} + + + Адрес: + {' '} + {deliverySnapshot.addressLine ?? '—'} + + + + Получатель: + {' '} + {deliverySnapshot.recipientName ?? '—'} + + + + Телефон: + {' '} + {deliverySnapshot.recipientPhone ?? '—'} + + {deliverySnapshot.comment?.trim() && ( + + Комментарий к адресу: {deliverySnapshot.comment} + + )} + + ) : ( + + Данные адреса в заказе отсутствуют или не распознаны. + + )} + + )} + + + + Товары в заказе + + + {detail.items.map((item) => ( + + + + {item.titleSnapshot} + + + {item.qty} × {formatPriceRub(item.priceCentsSnapshot)} + + + {formatPriceRub(item.priceCentsSnapshot * item.qty)} + + ))} + + + + {detail.status === 'PENDING_PAYMENT' && detail.deliveryFeeLocked === false && ( + + Укажите итоговую стоимость доставки (₽). После сохранения клиент сможет оплатить заказ с учётом этой суммы. + + )} + + {detail.status === 'PENDING_PAYMENT' && detail.deliveryFeeLocked === false && ( + + )} + + + + Быстрый переход статуса + + {statusMut.isError && Не удалось сменить статус} + {nextStatuses.length === 0 ? ( + + Статус финальный, смена недоступна + + ) : ( + + {nextStatuses.map((nextStatus) => { + const isCancelled = nextStatus === 'CANCELLED' + return ( + + ) + })} + + )} + + + + + Сообщения + + + {detail.messages.map((m) => { + const isAdminMsg = m.authorType === 'admin' + const avatarNode = isAdminMsg ? ( + currentUser && ( + + ) + ) : ( + + ) + return ( + + + {isAdminMsg ? 'Админ (вы)' : 'Пользователь'} · {new Date(m.createdAt).toLocaleString()} + + + + ) + })} + {detail.messages.length === 0 && Нет сообщений.} + + + + + + + + + + + ) +} diff --git a/client/src/features/order-detail/ui/__tests__/OrderDetailContent.test.tsx b/client/src/features/order-detail/ui/__tests__/OrderDetailContent.test.tsx new file mode 100755 index 0000000..385261f --- /dev/null +++ b/client/src/features/order-detail/ui/__tests__/OrderDetailContent.test.tsx @@ -0,0 +1,211 @@ +import type { ReactNode } from 'react' +import { QueryClient, QueryClientProvider } from '@tanstack/react-query' +import { render, screen, waitFor } from '@testing-library/react' +import userEvent from '@testing-library/user-event' +import { MemoryRouter } from 'react-router-dom' +import { beforeEach, describe, expect, it, vi } from 'vitest' +import { setAdminOrderStatus } from '@/entities/order/api/admin-order-api' +import type { AdminOrderDetailResponse } from '@/entities/order/api/admin-order-api' +import { ORDER_STATUS_MAP } from '@/shared/lib/order-status-data' +import { OrderDetailContent } from '../OrderDetailContent' + +vi.mock('@/entities/order/api/admin-order-api', () => ({ + setAdminOrderStatus: vi.fn(), + postAdminOrderMessage: vi.fn(), +})) + +vi.mock('effector-react', () => ({ + useUnit: vi.fn(), +})) + +vi.mock('@/shared/ui/RichTextMessageEditor.lazy', () => ({ + RichTextMessageEditor: ({ + value, + onChange, + }: { + value: string + onChange: (next: string) => void + placeholder?: string + disabled?: boolean + }) =>