diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml index f99ffb3a..53bddf68 100644 --- a/.github/workflows/deploy.yml +++ b/.github/workflows/deploy.yml @@ -17,12 +17,12 @@ jobs: steps: - name: Checkout code - uses: actions/checkout@v2 + uses: actions/checkout@v4 - name: Set up Node.js uses: actions/setup-node@v4 with: - node-version: '16' + node-version: 18 - name: Install dependencies run: yarn diff --git a/.github/workflows/distribute.yml b/.github/workflows/distribute.yml index 1effd142..a9316bb0 100644 --- a/.github/workflows/distribute.yml +++ b/.github/workflows/distribute.yml @@ -32,7 +32,7 @@ jobs: tag_version: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: fetch-depth: 0 @@ -64,10 +64,10 @@ jobs: needs: tag_version runs-on: ubuntu-latest steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v4 with: ref: master - - uses: actions/cache@v2 + - uses: actions/cache@v4 with: path: '**/node_modules' key: ${{ runner.os }}-modules-${{ hashFiles('**/yarn.lock') }} @@ -91,13 +91,17 @@ jobs: needs: tag_version runs-on: ubuntu-latest steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v4 with: ref: master - - uses: actions/cache@v2 + - uses: actions/cache@v4 with: path: '**/node_modules' key: ${{ runner.os }}-modules-${{ hashFiles('**/yarn.lock') }} + - name: Set up Node.js + uses: actions/setup-node@v4 + with: + node-version: 18 - name: Install yarn dependencies run: yarn diff --git a/.nvmrc b/.nvmrc new file mode 100644 index 00000000..23cc58a7 --- /dev/null +++ b/.nvmrc @@ -0,0 +1 @@ +18.20.0 \ No newline at end of file diff --git a/package.json b/package.json index bf1e0f6c..157a156c 100644 --- a/package.json +++ b/package.json @@ -9,23 +9,24 @@ "@tanstack/react-query-persist-client": "^5.8.4", "@types/dompurify": "^2.3.4", "@types/jspath": "^0.4.0", - "axios": "^0.21.2", + "axios": "^1.8.4", "axios-cache-adapter": "^2.7.3", "country-emoji": "^1.5.4", - "dompurify": "^2.2.7", + "dompurify": "^3.2.4", + "firebase": "^11.2.0", "htmlparser2": "^8.0.1", "jsonpath": "^1.1.1", "normalize.css": "^8.0.1", "prop-types": "^15.0.0-0", "react": "^17.0.1", "react-contexify": "^5.0.0", - "react-device-detect": "^1.17.0", "react-dom": "^17.0.1", "react-easy-sort": "^1.5.1", "react-error-boundary": "^3.1.4", "react-icons": "^5.2.1", "react-markdown": "^7.0.1", "react-modal": "^3.12.1", + "react-responsive": "^10.0.1", "react-router-dom": "^6.21.0", "react-select": "^5.0.1", "react-share": "^4.4.1", @@ -40,6 +41,9 @@ "web-vitals": "^0.2.4", "zustand": "^4.3.3" }, + "engines": { + "node": "18" + }, "scripts": { "start": "vite", "preview": "vite preview", @@ -91,7 +95,7 @@ "prettier": "^2.7.1", "terser": "^5.19.2", "typescript": "^5.1.6", - "vite": "^4.4.5", + "vite": "^6.2.3", "vite-plugin-svgr": "^3.2.0", "vite-tsconfig-paths": "^4.2.0" } diff --git a/public/auth.js b/public/auth.js new file mode 100644 index 00000000..2f6f2d02 --- /dev/null +++ b/public/auth.js @@ -0,0 +1,6 @@ +window.addEventListener('message', (event) => { + if (event.data.type === 'TOKEN_RECEIVED') { + // Forward to content script + window.postMessage(event.data, '*') + } +}) diff --git a/public/background.js b/public/background.js index e7861b7e..df8301a7 100644 --- a/public/background.js +++ b/public/background.js @@ -1,9 +1,4 @@ -/*chrome.browserAction.onClicked.addListener(function (tab) { - chrome.tabs.create({ url: "chrome://newtab" }); -} -);*/ - const uninstallUrl = `https://hackertab.dev/uninstall.html` if (chrome.runtime.setUninstallURL) { chrome.runtime.setUninstallURL(uninstallUrl) -} \ No newline at end of file +} diff --git a/public/base.manifest.json b/public/base.manifest.json index 13f81829..7be1bb4d 100644 --- a/public/base.manifest.json +++ b/public/base.manifest.json @@ -1,10 +1,19 @@ { + "manifest_version": 3, "name": "Hackertab.dev - developer news", "description": "All developer news in one tab", "version": "1.20.1", "chrome_url_overrides": { "newtab": "index.html" }, + "host_permissions": ["https://*.hackertab.dev/*"], + "content_scripts": [ + { + "matches": ["https://hackertab.dev/*"], + "run_at": "document_start", + "js": ["content.js"] + } + ], "icons": { "16": "/logos/logo16.png", "32": "/logos/logo32.png", diff --git a/public/chrome.manifest.json b/public/chrome.manifest.json index 87acc9f8..fcb931e3 100644 --- a/public/chrome.manifest.json +++ b/public/chrome.manifest.json @@ -1,7 +1,5 @@ { - "manifest_version": 3, "background": { "service_worker": "background.js" - }, - "host_permissions": ["https://*.hackertab.dev/*"] + } } diff --git a/public/content.js b/public/content.js new file mode 100644 index 00000000..21e9b99b --- /dev/null +++ b/public/content.js @@ -0,0 +1,12 @@ +const script = document.createElement('script') +script.src = chrome.runtime.getURL('auth.js') +document.documentElement.appendChild(script) + +// Listen for messages from the injected script +window.addEventListener('message', (event) => { + if (event.source !== window || !event.data || event.data.type !== 'TOKEN_RECEIVED') { + return + } + + chrome.runtime.sendMessage({ ...event.data }) +}) diff --git a/public/firefox.manifest.json b/public/firefox.manifest.json index db181686..769e74b6 100644 --- a/public/firefox.manifest.json +++ b/public/firefox.manifest.json @@ -1,17 +1,5 @@ { - "manifest_version": 2, - "background": { - "scripts": [ - "background.js" - ] - }, - "permissions": [ - "https://*.hackertab.dev/*" - ], - "content_security_policy": "script-src 'self' object-src 'self'", - "applications": { - "gecko": { - "id": "{f8793186-e9da-4332-aa1e-dc3d9f7bb04c}" - } - } - } \ No newline at end of file + "background": { + "scripts": ["background.js"] + } +} diff --git a/src/App.tsx b/src/App.tsx index 788dd432..a14f024b 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -1,8 +1,14 @@ import { useEffect, useLayoutEffect, useState } from 'react' import { DNDLayout } from 'src/components/Layout' -import { setupAnalytics, setupIdentification, trackPageView } from 'src/lib/analytics' +import { + identifyAdvBlocked, + setupAnalytics, + setupIdentification, + trackPageView, +} from 'src/lib/analytics' import { useUserPreferences } from 'src/stores/preferences' import { AppContentLayout } from './components/Layout' +import { verifyAdvStatus } from './features/adv/utils/status' import { isWebOrExtensionVersion } from './utils/Environment' import { lazyImport } from './utils/lazyImport' const { OnboardingModal } = lazyImport(() => import('src/features/onboarding'), 'OnboardingModal') @@ -30,6 +36,11 @@ export const App = () => { document.body.classList.remove('preload') setupAnalytics() setupIdentification() + const adVerifier = async () => { + const status = await verifyAdvStatus() + identifyAdvBlocked(status) + } + adVerifier() }, []) useEffect(() => { diff --git a/src/assets/App.css b/src/assets/App.css index d460729b..bb4551f9 100644 --- a/src/assets/App.css +++ b/src/assets/App.css @@ -122,44 +122,32 @@ a { } .extras { - display: none; - flex-direction: row; - align-content: center; order: 2; } -.extraBtn { - background-color: var(--button-background-color); - color: var(--button-text-color); - margin-left: 8px; - height: 40px; - width: 40px; - line-height: 44px; - font-size: 18px; - text-align: center; - border: 0; - border-radius: 20px; - cursor: pointer; - position: relative; +.buttonsFlex { display: inline-flex; - align-items: center; - justify-content: center; + flex-direction: row; + align-content: center; + column-gap: 8px; + row-gap: 8px; } -.extraBtn:first-child { - margin-left: 0; +.dndButton { + font-weight: bold; } -.extraTextBtn { - padding: 0 16px; - width: auto; - min-width: 40px; - font-weight: bold; +.profileImage { + height: 40px; + width: 40px; + border-radius: 20px; } -.darkModeBtn { - background-color: var(--dark-mode-background-color); - color: var(--dark-mode-text-color); +.avatarPlaceholder { + height: 34px; + width: 34px; + margin-top: 10px; + border-radius: 20px; } .badgeCount { @@ -1018,7 +1006,7 @@ Producthunt item *** BREAKPOINTS *******************/ /* Extra small devices (phones, 600px and down) */ -@media (max-width: 600px) { +@media (max-width: 768px) { .bottomNavigation { display: flex; } @@ -1174,9 +1162,52 @@ Producthunt item color: white; font-size: 24px; } +.overflowHidden { + overflow: hidden; +} +/* User */ +.profileImageContainer { + position: relative; + + img { + width: 40px; + height: 40px; + border-radius: 50%; + border: 2px solid #ff8f1f; + } + + .streak { + position: absolute; + display: inline-block; + bottom: -8px; + left: 0; + right: 0; + color: white; + + .content { + background: #f88e0c; + font-size: 11px; + border-radius: 12px; + font-weight: bold; + border: 2px solid var(--background-color); + padding: 1px 6px 1px 22px; + display: inline-flex; + align-items: center; + justify-content: center; + color: white; + } + .icon { + width: 2.2em; + position: absolute; + left: 0; + top: -2px; + filter: drop-shadow(0 -4px 4px rgb(0 0 0 / 0.4)); + } + } +} /* Small devices (portrait tablets and large phones, 600px and up) */ -@media only screen and (min-width: 600px) { +@media only screen and (min-width: 768px) { .floatingFilter { display: none; } @@ -1198,10 +1229,6 @@ Producthunt item position: relative; } - .extras { - display: block; - } - .scrollButton { align-items: center; display: flex; @@ -1349,6 +1376,135 @@ Producthunt item color: var(--primary-text-color); border-radius: 10px; } + +.dangerToast { + background-color: #ff4d4f; + color: white; + padding: 10px 20px; + border-radius: 10px; +} +.successToast { + background-color: #52c41a; + color: white; + padding: 10px 20px; + border-radius: 10px; +} .capitalize { text-transform: capitalize; } + +/** +Modal +**/ + +.Modal { + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + width: 650px; + background-color: var(--card-background-color); + padding: 24px; + border-radius: 10px; + box-shadow: 0 0 20px #00000052; + z-index: 3; + max-height: 80vh; + overflow-y: auto; +} + +.Overlay { + position: fixed; + top: 0; + left: 0; + right: 0; + bottom: 0; + background-color: var(--overlay-background-color); + backdrop-filter: blur(2px); +} + +.modalHeader { + display: flex; + flex-direction: row; + align-items: center; + justify-content: space-between; + margin-bottom: 16px; + position: relative; +} + +.modalTitle { + margin: 0; + padding: 0; + color: var(--primary-text-color); + font-size: 1.3em; + display: inline-flex; + width: 100%; + column-gap: 8px; + align-items: center; +} +.modalCloseBtn { + align-items: center; + position: absolute; + background-color: transparent; + border-radius: 50%; + top: 0; + right: 0; + border: none; + color: var(--primary-text-color); + cursor: pointer; + display: flex; + justify-content: center; + margin: 0; + padding: 0; + text-align: center; +} +.modalCloseBtn:hover { + opacity: 0.7; +} +.settingContent .form { + display: flex; + flex-direction: row; + align-items: center; + gap: 12px; +} + +.settingContent input[type='text'] { + flex: 1; + background-color: var(--settings-input-background-color); + border: 1px solid var(--settings-input-border-color); + border-radius: 50px; + padding: 6px 18px; + color: var(--settings-input-text-color); + font-size: 14px; +} + +.settingContent input[type='text']::placeholder { + /* Chrome, Firefox, Opera, Safari 10.1+ */ + color: var(--settings-input-placeholder-color); + opacity: 1; + font-size: 14px; +} + +.settingContent input[type='text']:focus { + border-color: var(--settings-input-border-focus-color); +} +@media (max-width: 768px) { + .Modal { + left: 0; + top: 0; + margin: 0; + height: 100vh; + max-height: 100vh; + transform: translate(0, 0); + border-radius: 0; + position: relative; + box-shadow: none; + width: auto; + } + .settingContent { + margin-top: 6px; + } + .settingRow { + flex-direction: column; + align-items: flex-start; + } +} diff --git a/src/assets/icons/avatar.svg b/src/assets/icons/avatar.svg new file mode 100644 index 00000000..91b9bd7a --- /dev/null +++ b/src/assets/icons/avatar.svg @@ -0,0 +1,19 @@ + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/fire_icon.svg b/src/assets/icons/fire_icon.svg new file mode 100644 index 00000000..12f8ed21 --- /dev/null +++ b/src/assets/icons/fire_icon.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/variables.css b/src/assets/variables.css index e1d8ac7e..89b7a223 100644 --- a/src/assets/variables.css +++ b/src/assets/variables.css @@ -22,6 +22,7 @@ html.dark { --secondary-text-color: #a2a7ad; --overlay-background-color: #7985958a; --language-dot-background-color: #424d5a; + --icons-accent-color: #ffffff; --dark-mode-background-color: #1c2026; --dark-mode-text-color: #f0c73d; @@ -31,7 +32,9 @@ html.dark { /** Buttons **/ --button-background-color: #1c2026; + --button-hover-background-color: #181c21; --button-text-color: #fff; + --button-hover-text-color: #d0d0d0; /**Card **/ --card-background-color: #0d1116; @@ -108,6 +111,7 @@ html.light { --overlay-background-color: #0000005c; --primary-hover-text-color: #495a6b; --language-dot-background-color: #cfddec; + --icons-accent-color: #3c5065; --scrollbar-background-color: #eff6ff; --scrollbar-accent-color: #b9cadc; @@ -117,6 +121,8 @@ html.light { --button-background-color: #d2deeb; --button-text-color: #3c5065; + --button-hover-background-color: #b9cadc; + --button-hover-text-color: #3c5065; /**Card **/ --card-background-color: #ffffff; diff --git a/src/components/Elements/Button/Button.css b/src/components/Elements/Button/Button.css new file mode 100644 index 00000000..be5b5d51 --- /dev/null +++ b/src/components/Elements/Button/Button.css @@ -0,0 +1,97 @@ +.button { + background-color: var(--button-background-color); + color: var(--button-text-color); + font-size: 18px; + text-align: center; + border: 0; + border-radius: 50px; + cursor: pointer; + position: relative; + display: inline-flex; + align-items: center; + justify-content: center; + gap: 0.5rem; + + &:hover { + background-color: var(--button-hover-background-color); + color: var(--button-hover-text-color); + } + + &.disabled { + cursor: not-allowed; + opacity: 0.8; + } + &.small { + padding: 0.3rem 0.75rem; + font-size: 0.875rem; + } + + &.medium { + padding: 0.75rem 1rem; + font-size: 1.1rem; + height: 40px; + } + + &.large { + padding: 1rem 2rem; + font-size: 1.25rem; + } + + &.loading { + pointer-events: none; + cursor: not-allowed; + opacity: 0.8; + } +} + +.circle-button { + border-radius: 50px; + cursor: pointer; + display: inline-flex; + align-items: center; + justify-content: center; + gap: 0.5rem; + border: none; + padding: 0; + padding: 0; + font-size: 1.1em; + + &.disabled { + cursor: not-allowed; + opacity: 0.8; + } + + &.small { + min-width: 30px; + height: 30px; + } + + &.medium { + min-width: 40px; + height: 40px; + } + + &.large { + min-width: 50px; + height: 50px; + } + + &.primary { + background-color: var(--button-background-color); + color: var(--button-text-color); + + &:hover { + background-color: var(--button-hover-background-color); + color: var(--button-hover-text-color); + } + } + + &.dark-focus { + background-color: var(--dark-mode-background-color); + color: var(--dark-mode-text-color); + + &:hover { + opacity: 0.9; + } + } +} diff --git a/src/components/Elements/Button/Button.tsx b/src/components/Elements/Button/Button.tsx new file mode 100644 index 00000000..1fc82658 --- /dev/null +++ b/src/components/Elements/Button/Button.tsx @@ -0,0 +1,44 @@ +import clsx from 'clsx' +import React from 'react' +import { Spinner } from '../Spinner' +import './Button.css' +const sizes = { + small: 'small', + medium: 'medium', + large: 'large', +} +type ButtonProps = { + children: React.ReactNode + onClick: () => void + className?: string + size?: keyof typeof sizes + startIcon?: React.ReactNode + endIcon?: React.ReactNode + isLoading?: boolean +} +export const Button = ({ + size = 'medium', + onClick, + className, + startIcon, + endIcon, + children, + isLoading = false, +}: ButtonProps) => { + return ( + + ) +} diff --git a/src/components/Elements/Button/CircleButton.tsx b/src/components/Elements/Button/CircleButton.tsx new file mode 100644 index 00000000..71d112dc --- /dev/null +++ b/src/components/Elements/Button/CircleButton.tsx @@ -0,0 +1,46 @@ +import clsx from 'clsx' +import { Spinner } from '../Spinner' + +const sizes = { + small: 'small', + medium: 'medium', + large: 'large', +} + +const variants = { + primary: 'primary', + darkfocus: 'dark-focus', +} + +type CircleButtonProps = { + className?: string + isLoading?: boolean + onClick: () => void + size?: keyof typeof sizes + variant?: keyof typeof variants + children: React.ReactNode +} + +export const CircleButton = ({ + size = 'medium', + variant = 'primary', + onClick, + isLoading, + className, + children, +}: CircleButtonProps) => { + return ( + + ) +} diff --git a/src/components/Elements/Button/index.ts b/src/components/Elements/Button/index.ts new file mode 100644 index 00000000..9a7dca3d --- /dev/null +++ b/src/components/Elements/Button/index.ts @@ -0,0 +1,2 @@ +export * from './Button' +export * from './CircleButton' diff --git a/src/components/Elements/Card/Card.tsx b/src/components/Elements/Card/Card.tsx index b6c47f0b..973c8792 100644 --- a/src/components/Elements/Card/Card.tsx +++ b/src/components/Elements/Card/Card.tsx @@ -1,11 +1,11 @@ import React, { useEffect, useState } from 'react' -import { isDesktop } from 'react-device-detect' import { SortableKnob } from 'react-easy-sort' import { BsBoxArrowInUpRight } from 'react-icons/bs' import { MdOutlineDragIndicator } from 'react-icons/md' import { ref } from 'src/config' import { AdvBanner } from 'src/features/adv' import { useRemoteConfigStore } from 'src/features/remoteConfig' +import { DesktopBreakpoint } from 'src/providers/DesktopBreakpoint' import { useUserPreferences } from 'src/stores/preferences' import { SupportedCardType } from 'src/types' @@ -59,13 +59,13 @@ export const Card = ({ return (
- {isDesktop && ( + - )} + {icon} {titleComponent || label}{' '} {link && ( diff --git a/src/components/Elements/Modal/ConfirmModal.tsx b/src/components/Elements/Modal/ConfirmModal.tsx new file mode 100644 index 00000000..a1e70b00 --- /dev/null +++ b/src/components/Elements/Modal/ConfirmModal.tsx @@ -0,0 +1,42 @@ +import { Button, Modal } from 'src/components/Elements' +import './confirmModal.css' + +interface LogoutModalProps { + showModal: boolean + title: string + description: string + onClose: () => void + onConfirm: () => void +} + +export const ConfirmModal = ({ + showModal, + title, + description, + onClose, + onConfirm, +}: LogoutModalProps) => { + return ( + +
+

{description}

+
+ + +
+
+
+ ) +} diff --git a/src/components/Elements/Modal/Modal.tsx b/src/components/Elements/Modal/Modal.tsx new file mode 100644 index 00000000..46515014 --- /dev/null +++ b/src/components/Elements/Modal/Modal.tsx @@ -0,0 +1,49 @@ +import clsx from 'clsx' +import { VscClose } from 'react-icons/vsc' +import ReactModal from 'react-modal' +type ModalProps = { + showModal: boolean + onClose: () => void + className?: string + header: { + title: string + className?: string + icon?: React.ReactNode + } + children: React.ReactNode +} +export const Modal = ({ showModal, className, header, onClose, children }: ModalProps) => { + return ( + +
+
+

+ {header.icon} + {header.title} +

+ +
+ + {children} +
+
+ ) +} diff --git a/src/components/Elements/Modal/confirmModal.css b/src/components/Elements/Modal/confirmModal.css new file mode 100644 index 00000000..aa2a27f3 --- /dev/null +++ b/src/components/Elements/Modal/confirmModal.css @@ -0,0 +1,20 @@ +@media (min-width: 768px) { + .confirmModal { + width: 400px; + overflow-y: hidden; + } +} + +.confirmModal .content { + display: flex; + flex-direction: column; +} +.confirmModal .description { + font-size: 1em; +} +.confirmModal .buttons { + display: flex; + flex-direction: row; + gap: 10px; + align-self: flex-end; +} diff --git a/src/components/Elements/Modal/index.ts b/src/components/Elements/Modal/index.ts new file mode 100644 index 00000000..701e747d --- /dev/null +++ b/src/components/Elements/Modal/index.ts @@ -0,0 +1,2 @@ +export * from './ConfirmModal' +export * from './Modal' diff --git a/src/components/Elements/Spinner/Spinner.css b/src/components/Elements/Spinner/Spinner.css new file mode 100644 index 00000000..78e63a02 --- /dev/null +++ b/src/components/Elements/Spinner/Spinner.css @@ -0,0 +1,32 @@ +.spinner { + position: relative; + border-radius: 50%; + border: 2px solid transparent; + border-top-color: currentColor; + border-right-color: currentColor; + border-bottom-color: currentColor; + animation: spin 1.5s linear infinite; + &.small { + width: 20px; + height: 20px; + } + + &.medium { + width: 40px; + height: 40px; + } + + &.large { + width: 50px; + height: 50px; + } +} + +@keyframes spin { + 0% { + transform: rotate(0deg); + } + 100% { + transform: rotate(360deg); + } +} diff --git a/src/components/Elements/Spinner/Spinner.tsx b/src/components/Elements/Spinner/Spinner.tsx new file mode 100644 index 00000000..d55133cb --- /dev/null +++ b/src/components/Elements/Spinner/Spinner.tsx @@ -0,0 +1,17 @@ +import clsx from 'clsx' +import './Spinner.css' + +const sizes = { + small: 'small', + medium: 'medium', + large: 'large', +} + +export type SpinnerProps = { + size?: keyof typeof sizes + className?: string +} + +export const Spinner = ({ size = 'medium', className = '' }: SpinnerProps) => { + return
+} diff --git a/src/components/Elements/Spinner/index.ts b/src/components/Elements/Spinner/index.ts new file mode 100644 index 00000000..cf2b71e2 --- /dev/null +++ b/src/components/Elements/Spinner/index.ts @@ -0,0 +1 @@ +export * from './Spinner' diff --git a/src/components/Elements/index.ts b/src/components/Elements/index.ts index 850e09f7..14add55f 100644 --- a/src/components/Elements/index.ts +++ b/src/components/Elements/index.ts @@ -1,4 +1,5 @@ export * from './BottomNavigation' +export * from './Button' export * from './Card' export * from './CardLink' export * from './CardWithActions' @@ -7,8 +8,10 @@ export * from './ClickableItem' export * from './ColoredLanguagesBadges' export * from './FloatingFilter' export * from './InlineTextFilter' +export * from './Modal' export * from './Panel' export * from './SearchBar' export * from './SearchBarWithLogo' +export * from './Spinner' export * from './Steps' export * from './UserTags' diff --git a/src/components/Layout/AppContentLayout.tsx b/src/components/Layout/AppContentLayout.tsx index 5189bccf..03ffbf12 100644 --- a/src/components/Layout/AppContentLayout.tsx +++ b/src/components/Layout/AppContentLayout.tsx @@ -1,5 +1,6 @@ import { useState } from 'react' -import { isDesktop } from 'react-device-detect' +import { DesktopBreakpoint } from 'src/providers/DesktopBreakpoint' +import { MobileBreakpoint } from 'src/providers/MobileBreakpoint' import { useUserPreferences } from 'src/stores/preferences' import { BottomNavigation } from '../Elements' import { ScrollCardsNavigator } from './' @@ -14,13 +15,14 @@ export const AppContentLayout = () => { <>
- {isDesktop ? ( + - ) : ( + +
- )} +
diff --git a/src/components/Layout/AppLayout.tsx b/src/components/Layout/AppLayout.tsx index 5d9fe243..f8ae0748 100644 --- a/src/components/Layout/AppLayout.tsx +++ b/src/components/Layout/AppLayout.tsx @@ -1,18 +1,33 @@ -import React from 'react' +import React, { useEffect } from 'react' import 'react-contexify/dist/ReactContexify.css' import { Outlet } from 'react-router-dom' import { BeatLoader } from 'react-spinners' import 'src/assets/App.css' +import { AuthModal, useAuth } from 'src/features/auth' +import { usePostStreak } from 'src/features/hits' import { MarketingBanner } from 'src/features/MarketingBanner' +import { AuthProvider } from 'src/providers/AuthProvider' import { Header } from './Header' export const AppLayout = () => { + const { isAuthModalOpen, setStreak, isConnected } = useAuth() + const postStreakMutation = usePostStreak() + + useEffect(() => { + if (isConnected) { + postStreakMutation.mutateAsync(undefined).then((data) => { + setStreak(data.streak) + }) + } + }, []) + return ( - <> +
+ @@ -22,6 +37,6 @@ export const AppLayout = () => {
- +
) } diff --git a/src/components/Layout/Header.tsx b/src/components/Layout/Header.tsx index 749928b7..1b2fedbd 100644 --- a/src/components/Layout/Header.tsx +++ b/src/components/Layout/Header.tsx @@ -1,22 +1,25 @@ +import { clsx } from 'clsx' import { useEffect, useState } from 'react' import { BsFillBookmarksFill, BsFillGearFill, BsMoonFill } from 'react-icons/bs' import { CgTab } from 'react-icons/cg' import { IoMdSunny } from 'react-icons/io' import { MdDoDisturbOff } from 'react-icons/md' -import { RxArrowLeft } from 'react-icons/rx' import { Link, useLocation, useNavigate } from 'react-router-dom' +import { ReactComponent as AvatarPlaceholder } from 'src/assets/icons/avatar.svg' +import { ReactComponent as StreakIcon } from 'src/assets/icons/fire_icon.svg' import { ReactComponent as HackertabLogo } from 'src/assets/logo.svg' import { SearchBar } from 'src/components/Elements/SearchBar' import { UserTags } from 'src/components/Elements/UserTags' +import { useAuth } from 'src/features/auth' import { Changelog } from 'src/features/changelog' import { identifyUserTheme, trackDNDDisable, trackThemeSelect } from 'src/lib/analytics' -import { useBookmarks } from 'src/stores/bookmarks' import { useUserPreferences } from 'src/stores/preferences' - +import { Button, CircleButton } from '../Elements' export const Header = () => { + const { openAuthModal, user, isConnected, isConnecting } = useAuth() + const [themeIcon, setThemeIcon] = useState() const { theme, setTheme, setDNDDuration, isDNDModeActive } = useUserPreferences() - const { userBookmarks } = useBookmarks() const navigate = useNavigate() const location = useLocation() @@ -46,16 +49,6 @@ export const Header = () => { navigate('/settings/general') } - const BookmarksBadgeCount = () => { - return userBookmarks.length > 0 ? ( - userBookmarks.length < 10 ? ( - {userBookmarks.length} - ) : ( - +9 - ) - ) : null - } - const onUnpauseClicked = () => { trackDNDDisable() setDNDDuration('never') @@ -74,38 +67,48 @@ export const Header = () => { -
+
{isDNDModeActive() && ( - + Unpause + )} - - - - <> - - - - + + navigate('/settings/bookmarks')}> + + + { + if (isConnected) { + navigate('/settings/general') + } else { + openAuthModal() + } + }}> + {isConnected ? ( + <> + +
+ + {user?.streak || 1} + +
+ + ) : ( + + )} +
- {location.pathname === '/' ? ( - - ) : ( -
- - Back - -
- )} + {location.pathname === '/' && } ) diff --git a/src/components/Layout/SettingsContentLayout/SettingsContentLayout.tsx b/src/components/Layout/SettingsContentLayout/SettingsContentLayout.tsx index fb6b1c31..cd2b596f 100644 --- a/src/components/Layout/SettingsContentLayout/SettingsContentLayout.tsx +++ b/src/components/Layout/SettingsContentLayout/SettingsContentLayout.tsx @@ -4,18 +4,24 @@ type SettingsContentLayoutProps = { title: string description: string children: React.ReactNode + actions?: React.ReactNode } export const SettingsContentLayout = ({ title, description, + actions, children, }: SettingsContentLayoutProps) => { return (
-

{title}

-

{description}

+
+

{title}

+

{description}

+
+ + {actions &&
{actions}
}
{children}
diff --git a/src/components/Layout/SettingsContentLayout/settingsContentLayout.css b/src/components/Layout/SettingsContentLayout/settingsContentLayout.css index 07437110..a79379a8 100644 --- a/src/components/Layout/SettingsContentLayout/settingsContentLayout.css +++ b/src/components/Layout/SettingsContentLayout/settingsContentLayout.css @@ -15,23 +15,33 @@ gap: 16px; } .settingsContent header { + display: flex; + justify-content: space-between; + align-items: center; +} + +.settingsContent .settingsHeader { display: flex; flex-direction: column; row-gap: 8px; } -.settingsContent header .title { +.settingsContent .settingsHeader .title { font-size: 24px; margin: 0; padding: 0; color: var(--primary-text-color); } -.settingsContent header .description { +.settingsContent .settingsHeader .description { font-size: 14px; margin: 0; padding: 0; color: var(--secondary-text-color); } - +.settingsContent .actions { + display: flex; + justify-content: space-between; + align-items: center; +} @media only screen and (max-width: 768px) { .settingsContent { padding: 16px; diff --git a/src/config/index.tsx b/src/config/index.tsx index 2f5f622b..d0553ca6 100644 --- a/src/config/index.tsx +++ b/src/config/index.tsx @@ -3,6 +3,9 @@ export const ANALYTICS_ENDPOINT = import.meta.env.VITE_AMPLITUDE_URL as string export const ANALYTICS_SDK_KEY = import.meta.env.VITE_AMPLITUDE_KEY as string export const API_ENDPOINT = import.meta.env.VITE_API_URL as string export const LS_ANALYTICS_ID_KEY = 'hackerTabAnalyticsId' +export const FIREBASE_API_KEY = import.meta.env.VITE_FIREBASE_API_KEY as string +export const BUILD_TARGET = (import.meta.env.VITE_BUILD_TARGET as 'web' | 'extension') || 'web' + // Meta export const name = 'Hackertab.dev' export const slogan = '— Stay updated with the new technology and trends' diff --git a/src/features/MarketingBanner/components/MarketingBanner.tsx b/src/features/MarketingBanner/components/MarketingBanner.tsx index 902ff258..4f7d6001 100644 --- a/src/features/MarketingBanner/components/MarketingBanner.tsx +++ b/src/features/MarketingBanner/components/MarketingBanner.tsx @@ -1,7 +1,7 @@ import DOMPurify from 'dompurify' import jsonPath from 'jsonpath' import { useEffect, useMemo, useState } from 'react' -import { isMobile } from 'react-device-detect' +import { useMediaQuery } from 'react-responsive' import { trackMarketingCampaignClose, trackMarketingCampaignOpen, @@ -25,7 +25,7 @@ export const MarketingBanner = () => { cacheTime: 600000, }, }) - + const isMobile = useMediaQuery({ maxWidth: 767 }) const userAtttributes = useMemo(() => { return { platform: isWebOrExtensionVersion(), diff --git a/src/features/adv/utils/status.ts b/src/features/adv/utils/status.ts new file mode 100644 index 00000000..4c355fe8 --- /dev/null +++ b/src/features/adv/utils/status.ts @@ -0,0 +1,11 @@ +export const verifyAdvStatus = async () => { + try { + const resp = await fetch('https://srv.buysellads.com', { + mode: 'no-cors', + }) + await resp.text() + return false + } catch (e) { + return true + } +} diff --git a/src/features/auth/api/deleteAccount.ts b/src/features/auth/api/deleteAccount.ts new file mode 100644 index 00000000..c448ea4c --- /dev/null +++ b/src/features/auth/api/deleteAccount.ts @@ -0,0 +1,23 @@ +import { useMutation } from '@tanstack/react-query' +import { axios } from 'src/lib/axios' +import { MutationConfig } from 'src/lib/react-query' + +type DeleteAccountDTO = { + userId: string +} +const deleteAccount = ({ userId }: DeleteAccountDTO): Promise<{ authLink: string }> => { + return axios.delete(`/engine/user/${userId}`, {}) +} + +type QueryFnType = typeof deleteAccount + +type UseGetArticlesOptions = { + config?: MutationConfig +} + +export const useDeleteAccount = ({ config }: UseGetArticlesOptions = {}) => { + return useMutation({ + ...config, + mutationFn: deleteAccount, + }) +} diff --git a/src/features/auth/api/getOauthLink.ts b/src/features/auth/api/getOauthLink.ts new file mode 100644 index 00000000..3785f8bb --- /dev/null +++ b/src/features/auth/api/getOauthLink.ts @@ -0,0 +1,26 @@ +import { useMutation } from '@tanstack/react-query' +import { axios } from 'src/lib/axios' +import { MutationConfig } from 'src/lib/react-query' + +export type GetOauthLinkDTO = { + data: { + provider: string + state: string + } +} +const getOauthLink = ({ data }: GetOauthLinkDTO): Promise<{ authLink: string }> => { + return axios.post('/engine/auth/oauth-link', data) +} + +type QueryFnType = typeof getOauthLink + +type UseGetArticlesOptions = { + config?: MutationConfig +} + +export const useGetOauthLink = ({ config }: UseGetArticlesOptions = {}) => { + return useMutation({ + ...config, + mutationFn: getOauthLink, + }) +} diff --git a/src/features/auth/components/AuthModal.tsx b/src/features/auth/components/AuthModal.tsx new file mode 100644 index 00000000..18423fb7 --- /dev/null +++ b/src/features/auth/components/AuthModal.tsx @@ -0,0 +1,130 @@ +import { AuthProvider, GithubAuthProvider, GoogleAuthProvider } from 'firebase/auth' +import { useCallback, useState } from 'react' +import { FaGithub } from 'react-icons/fa' +import { FcGoogle } from 'react-icons/fc' +import { IoHeartCircle } from 'react-icons/io5' +import { Button, Modal } from 'src/components/Elements' +import { BUILD_TARGET, privacyPolicyLink, termsAndConditionsLink } from 'src/config' +import { useAuth } from 'src/features/auth' +import { getBrowserName } from 'src/utils/Environment' +import { checkHostPermissions, requestHostPermissions } from 'src/utils/Permissions' +import { useGetOauthLink } from '../api/getOauthLink' +import './authModal.css' + +type AuthModalProps = { + showAuth: boolean +} + +const googleAuthProvider = new GoogleAuthProvider() +const githubAuthProvider = new GithubAuthProvider() + +export const AuthModal = ({ showAuth }: AuthModalProps) => { + const { closeAuthModal, authError, setAuthError } = useAuth() + const [selectedProvider, setSelectedProvider] = useState(googleAuthProvider) + const getOauthLink = useGetOauthLink() + + const requestOauthLink = useCallback( + async (provider: AuthProvider) => { + getOauthLink + .mutateAsync({ + data: { + provider: provider.providerId, + state: BUILD_TARGET === 'web' ? window.location.origin : getBrowserName(), + }, + }) + .then(({ authLink }) => { + window.open(authLink, BUILD_TARGET === 'web' ? '_self' : '_blank') + }) + }, + [getOauthLink] + ) + + const signIn = useCallback(async (provider: AuthProvider) => { + setSelectedProvider(provider) + setAuthError(null) + + if (BUILD_TARGET === 'web') { + requestOauthLink(provider) + } else { + const permissionCheck = await checkHostPermissions() + + if (!permissionCheck) { + setAuthError({ + message: 'Hackertab needs permission to Sign in, Please request it and try again', + }) + } else { + requestOauthLink(provider) + } + } + }, []) + + const requestMissingPermission = useCallback(async () => { + const permissionRequest = await requestHostPermissions() + + if (!permissionRequest) { + setAuthError({ + message: 'Permission not granted, Request it again', + }) + } else { + setAuthError(null) + requestOauthLink(selectedProvider) + } + }, []) + + return ( + , + }} + className="authModal"> +
+

Create an account to sync, save bookmarks, and earn rewards.

+
+ + +
+ {authError && ( +
+

{authError.message}

+ {authError.retry && ( + + )} +
+ )} +
+

+ By signing in, you agree to our{' '} + + terms of Service + {' '} + and{' '} + + privacy Policy + + . +

+
+
+ + ) +} diff --git a/src/features/auth/components/authModal.css b/src/features/auth/components/authModal.css new file mode 100644 index 00000000..9a3ccfb8 --- /dev/null +++ b/src/features/auth/components/authModal.css @@ -0,0 +1,48 @@ +@media (min-width: 768px) { + .authModal { + width: 400px; + } +} +.authModal .header { + text-align: center; + justify-content: center; +} +.authModal .buttons { + display: inline-flex; + align-items: center; + justify-items: center; + flex-direction: column; + row-gap: 16px; + width: 100%; + margin: 0 0 20px 0; +} + +.authModal .description { + padding: 20px; + text-align: center; + font-size: 1em; +} +.authModal .errors { + background-color: rgba(255, 0, 0, 0.064); + border: 1px solid rgba(255, 0, 0, 0.363); + border-radius: 6px; + padding: 8px; + display: flex; + justify-content: center; + flex-direction: row; + p { + padding: 0; + margin: 0; + font-size: 0.8em; + color: red; + text-align: center; + } + .cta { + flex: 1; + } +} + +.authModal .footer { + font-size: 0.7em; + text-align: center; +} diff --git a/src/features/auth/hooks/useAuth.ts b/src/features/auth/hooks/useAuth.ts new file mode 100644 index 00000000..e9ae5578 --- /dev/null +++ b/src/features/auth/hooks/useAuth.ts @@ -0,0 +1,25 @@ +import { signOut } from 'firebase/auth' +import { AuthModalStore, AuthStore } from 'src/features/auth' +import { trackUserDisconnect } from 'src/lib/analytics' +import { firebaseAuth } from 'src/lib/firebase' + +export const useAuth = () => { + const authModalStore = AuthModalStore() + const authStore = AuthStore() + + const isConnected = authStore.user != null + + const logout = async () => { + trackUserDisconnect() + signOut(firebaseAuth) + authStore.clear() + return await firebaseAuth.signOut() + } + + return { + ...authModalStore, + ...authStore, + isConnected, + logout, + } +} diff --git a/src/features/auth/index.ts b/src/features/auth/index.ts new file mode 100644 index 00000000..ab44d99e --- /dev/null +++ b/src/features/auth/index.ts @@ -0,0 +1,6 @@ +export * from './components/AuthModal' +export * from './hooks/useAuth' +export * from './stores/authModalStore' +export * from './stores/authStore' +export * from './types' +export * from './utils/auth' diff --git a/src/features/auth/stores/authModalStore.ts b/src/features/auth/stores/authModalStore.ts new file mode 100644 index 00000000..0dbd94f3 --- /dev/null +++ b/src/features/auth/stores/authModalStore.ts @@ -0,0 +1,27 @@ +import { create } from 'zustand' + +type AuthError = { + message: string + retry?: { + label: string + } | null +} +interface AuthModalState { + isAuthModalOpen: boolean + isConnecting: boolean + authError: AuthError | null + openAuthModal: () => void + closeAuthModal: () => void + setAuthError: (error: AuthError | null) => void + setConnecting: (isConnecting: boolean) => void +} + +export const AuthModalStore = create((set) => ({ + isAuthModalOpen: false, + authError: null, + isConnecting: false, + setAuthError: (error) => set({ authError: error }), + openAuthModal: () => set({ isAuthModalOpen: true }), + closeAuthModal: () => set({ isAuthModalOpen: false }), + setConnecting: (isConnecting) => set({ isConnecting }), +})) diff --git a/src/features/auth/stores/authStore.ts b/src/features/auth/stores/authStore.ts new file mode 100644 index 00000000..17c18ebc --- /dev/null +++ b/src/features/auth/stores/authStore.ts @@ -0,0 +1,39 @@ +import { User } from 'src/features/auth/types' +import { create } from 'zustand' +import { persist } from 'zustand/middleware' + +type AuthState = { + user: User | null + providerId: string | null +} + +type AuthActions = { + initState: (state: AuthState) => void + setStreak: (streak: number) => void + clear: () => void +} + +export const AuthStore = create( + persist( + (set) => ({ + user: null, + providerId: null, + initState: (newState: AuthState) => + set({ + user: newState.user, + providerId: newState.providerId, + }), + setStreak: (streak: number) => + set((state) => ({ + user: { + ...state.user!, + streak, + }, + })), + clear: () => set({ user: null }), + }), + { + name: 'auth-storage', // key in localStorage + } + ) +) diff --git a/src/features/auth/types/index.ts b/src/features/auth/types/index.ts new file mode 100644 index 00000000..51b82333 --- /dev/null +++ b/src/features/auth/types/index.ts @@ -0,0 +1,6 @@ +export type User = { + id: string + name: string + imageURL?: string + streak?: number +} diff --git a/src/features/auth/utils/auth.ts b/src/features/auth/utils/auth.ts new file mode 100644 index 00000000..35409604 --- /dev/null +++ b/src/features/auth/utils/auth.ts @@ -0,0 +1,16 @@ +import { firebaseAuth } from 'src/lib/firebase' + +export const getUserToken = async () => { + return new Promise((resolve, _) => { + const unsub = firebaseAuth.onAuthStateChanged(async (user) => { + if (user) { + const token = await user.getIdToken() + resolve(token) + } else { + console.log('User not logged in') + resolve(null) + } + unsub() + }) + }) +} diff --git a/src/features/hits/api/postStreak.ts b/src/features/hits/api/postStreak.ts new file mode 100644 index 00000000..2fc2a8aa --- /dev/null +++ b/src/features/hits/api/postStreak.ts @@ -0,0 +1,18 @@ +import { useMutation } from '@tanstack/react-query' +import { axios } from 'src/lib/axios' +import { MutationConfig } from 'src/lib/react-query' +import { Streak } from '../types' + +const postStreak = async (): Promise => { + return axios.post('/engine/user/streak', {}) +} + +type UsePostStreakOptions = { + config?: MutationConfig +} +export const usePostStreak = ({ config }: UsePostStreakOptions = {}) => { + return useMutation({ + ...config, + mutationFn: postStreak, + }) +} diff --git a/src/features/hits/index.ts b/src/features/hits/index.ts new file mode 100644 index 00000000..ca92e270 --- /dev/null +++ b/src/features/hits/index.ts @@ -0,0 +1,2 @@ +export * from './api/postStreak' +export * from './types' diff --git a/src/features/hits/types/index.ts b/src/features/hits/types/index.ts new file mode 100644 index 00000000..57b63d29 --- /dev/null +++ b/src/features/hits/types/index.ts @@ -0,0 +1,3 @@ +export type Streak = { + streak: number +} diff --git a/src/features/settings/components/AddSearchEngine.tsx b/src/features/settings/components/AddSearchEngine.tsx index 48fb8d26..c3613947 100644 --- a/src/features/settings/components/AddSearchEngine.tsx +++ b/src/features/settings/components/AddSearchEngine.tsx @@ -2,6 +2,7 @@ import { useState } from 'react' import { isValidURL } from 'src/utils/UrlUtils' import { TiPlus } from 'react-icons/ti' +import { Button } from 'src/components/Elements' import { useUserPreferences } from 'src/stores/preferences' export const AddSearchEngine = () => { @@ -45,9 +46,9 @@ export const AddSearchEngine = () => { placeholder="https://google.com?q=" />
- +
{RssInputFeedback && ( diff --git a/src/features/settings/components/BookmarkSettings/BookmarkSettings.tsx b/src/features/settings/components/BookmarkSettings/BookmarkSettings.tsx index 86ca31d3..b27d8ce6 100644 --- a/src/features/settings/components/BookmarkSettings/BookmarkSettings.tsx +++ b/src/features/settings/components/BookmarkSettings/BookmarkSettings.tsx @@ -1,6 +1,8 @@ +import { useRef } from 'react' import { BiBookmarkMinus } from 'react-icons/bi' +import { RiFileDownloadFill, RiFileUploadFill } from 'react-icons/ri' import toast from 'react-simple-toasts' -import { CardLink } from 'src/components/Elements' +import { Button, CardLink, CircleButton } from 'src/components/Elements' import { SettingsContentLayout } from 'src/components/Layout/SettingsContentLayout' import { SUPPORTED_CARDS } from 'src/config/supportedCards' import { BookmarkedPost } from 'src/features/bookmarks' @@ -13,7 +15,8 @@ type BookmarkItemProps = { item: BookmarkedPost appendRef?: boolean } -const BookmarkItem = ({ item, appendRef = false }: BookmarkItemProps) => { + +export const BookmarkItem = ({ item, appendRef = false }: BookmarkItemProps) => { const { unbookmarkPost } = useBookmarks() const { userCustomCards } = useUserPreferences() @@ -60,17 +63,82 @@ const BookmarkItem = ({ item, appendRef = false }: BookmarkItemProps) => { } export const BookmarkSettings = () => { - const { userBookmarks } = useBookmarks() + const inputFile = useRef(null) + const { initState, userBookmarks } = useBookmarks() + + const importBookmarks = () => { + inputFile.current?.click() + } + + const exportBookmarks = () => { + const blob = new Blob([JSON.stringify(userBookmarks, null, 2)], { + type: 'application/json', + }) + const downloadURL = URL.createObjectURL(blob) + const link = document.createElement('a') + link.href = downloadURL + link.download = 'hackertabBookmarks' + link.click() + } + + const handleFileChange = (event: any) => { + const file = event.target.files?.[0] + if (file) { + const reader = new FileReader() + reader.onload = () => { + const importData: BookmarkedPost[] = JSON.parse(reader.result as string) + const validatedData = importData + .filter( + (data: BookmarkedPost) => + data.title && + data.url && + !userBookmarks.some((bm) => bm.title === data.title && bm.url === data.url) + ) + .map((data) => ({ + title: data.title, + url: data.url, + source: data.source || '', + sourceType: data.sourceType || '', + })) + initState({ + userBookmarks: [...userBookmarks, ...validatedData], + }) + toast('Your bookmarks have been successfully imported', { theme: 'defaultToast' }) + } + reader.readAsText(file) + } + } return ( - -
- {userBookmarks.map((bm) => ( - - ))} -
-
+ <> + + + + exportBookmarks()}> + + +
+ }> +
+ {userBookmarks.map((bm) => ( + + ))} +
+ + ) } diff --git a/src/features/settings/components/BookmarkSettings/bookmarkSettings.css b/src/features/settings/components/BookmarkSettings/bookmarkSettings.css index 6775a57c..889f7920 100644 --- a/src/features/settings/components/BookmarkSettings/bookmarkSettings.css +++ b/src/features/settings/components/BookmarkSettings/bookmarkSettings.css @@ -53,3 +53,31 @@ align-items: center; justify-content: center; } + +/* Export/Import Bookmark Buttons */ + +/* removing that ugly default button design */ +.notbtn { + background: none; + color: inherit; + border: none; + padding: 0; + font: inherit; + cursor: pointer; + outline: inherit; +} + +.btn-group { + padding: 10px; + display: flex; +} + +.btn { + border-radius: 12px; + padding: 6px 12px; + margin-right: 5px; + background-color: var(--chip-background); +} +.btn:hover { + filter: brightness(85%); +} diff --git a/src/features/settings/components/GeneralSettings/DNDSettings.tsx b/src/features/settings/components/GeneralSettings/DNDSettings.tsx index 2385ec01..1ac8a1b5 100644 --- a/src/features/settings/components/GeneralSettings/DNDSettings.tsx +++ b/src/features/settings/components/GeneralSettings/DNDSettings.tsx @@ -1,6 +1,7 @@ import { useState } from 'react' import { useNavigate } from 'react-router-dom' import Select, { SingleValue } from 'react-select' +import { Button } from 'src/components/Elements' import { trackDNDEnable } from 'src/lib/analytics' import { useUserPreferences } from 'src/stores/preferences' import { diffBetweenTwoDatesInMinutes } from 'src/utils/DateUtils' @@ -101,7 +102,9 @@ export const DNDSettings = () => { />
- + {timeoutLabel() && (
diff --git a/src/features/settings/components/GeneralSettings/GeneralSettings.tsx b/src/features/settings/components/GeneralSettings/GeneralSettings.tsx index 0915c0ef..80bbe10f 100644 --- a/src/features/settings/components/GeneralSettings/GeneralSettings.tsx +++ b/src/features/settings/components/GeneralSettings/GeneralSettings.tsx @@ -16,6 +16,8 @@ import { } from 'src/lib/analytics' import { useUserPreferences } from 'src/stores/preferences' import { Option } from 'src/types' +import { DeleteAccount } from '../UserSettings/DeleteAccount' +import { UserInfo } from '../UserSettings/UserInfo' import { DNDSettings } from './DNDSettings' import './generalSettings.css' @@ -24,7 +26,6 @@ export const GeneralSettings = () => { openLinksNewTab, listingMode, theme, - searchEngine, maxVisibleCards, setTheme, setListingMode, @@ -69,6 +70,7 @@ export const GeneralSettings = () => { 'Customize your experience by selecting the number of cards you want to see, the search engine you want to use and more.' }>
+

Max number of cards to display

@@ -133,6 +135,8 @@ export const GeneralSettings = () => { + +
diff --git a/src/features/settings/components/GeneralSettings/generalSettings.css b/src/features/settings/components/GeneralSettings/generalSettings.css index 81a3b7dd..ac030e82 100644 --- a/src/features/settings/components/GeneralSettings/generalSettings.css +++ b/src/features/settings/components/GeneralSettings/generalSettings.css @@ -1,32 +1,3 @@ -/** -Modal -**/ - -.Modal { - position: absolute; - left: 50%; - top: 50%; - transform: translate(-50%, -50%); - width: 650px; - background-color: var(--card-background-color); - padding: 24px; - border-radius: 10px; - box-shadow: 0 0 20px #00000052; - z-index: 3; - max-height: 80vh; - overflow-y: scroll; -} - -.Overlay { - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - background-color: var(--overlay-background-color); - backdrop-filter: blur(2px); -} - .settingTitle { width: 300px; margin: 0; @@ -45,34 +16,6 @@ Modal border-bottom: 1px solid var(--card-content-divider); } -.settingContent .form { - display: flex; - flex-direction: row; - align-items: center; - gap: 12px; -} - -.settingContent input[type='text'] { - flex: 1; - background-color: var(--settings-input-background-color); - border: 1px solid var(--settings-input-border-color); - border-radius: 50px; - padding: 6px 18px; - color: var(--settings-input-text-color); - font-size: 14px; -} - -.settingContent input[type='text']::placeholder { - /* Chrome, Firefox, Opera, Safari 10.1+ */ - color: var(--settings-input-placeholder-color); - opacity: 1; - font-size: 14px; -} - -.settingContent input[type='text']:focus { - border-color: var(--settings-input-border-focus-color); -} - .settingContent button { display: flex; align-items: center; @@ -92,10 +35,12 @@ Modal .rssButton { background-color: #ee802f; color: white; + &:hover { + background-color: #f99147; + color: white; + } } -.rssButton:hover { - opacity: 0.9; -} + .settingContent { width: 100%; flex: 1; @@ -110,37 +55,6 @@ Modal font-weight: 500; color: var(--primary-text-color); } -.modalHeader { - display: flex; - flex-direction: row; - justify-content: space-between; - margin-bottom: 16px; -} - -.modalTitle { - margin: 0; - padding: 0; - color: var(--primary-text-color); -} -.modalCloseBtn { - align-items: center; - background-color: transparent; - border-radius: 50%; - border: none; - color: var(--primary-text-color); - cursor: pointer; - display: flex; - height: 40px; - justify-content: center; - margin: 0; - padding: 0; - text-align: center; - width: 40px; -} -.modalCloseBtn:hover { - opacity: 0.7; -} - /** Select styles **/ @@ -182,24 +96,121 @@ Select styles color: var(--tag-secondary-color) !important; } -@media (max-width: 768px) { - .Modal { - left: 0; - top: 0; - margin: 0; - height: 100vh; - max-height: 100vh; - transform: translate(0, 0); - border-radius: 0; - position: relative; - box-shadow: none; +.userContent { + width: auto; + padding: 20px 0px; + display: flex; + align-items: center; + border-bottom: 1px solid var(--card-content-divider); + + .userImage { + width: 50px; + height: 50px; + border-radius: 100%; + align-self: flex-start; + } + .userInfos { + flex: 1; + display: flex; + flex-direction: column; + gap: 8px; width: auto; + margin-left: 10px; } - .settingContent { + .userName { + font-weight: 600; + } + .actions { margin-top: 6px; } - .settingRow { - flex-direction: column; - align-items: flex-start; + .sub { + font-size: 0.9em; + opacity: 0.9; + display: flex; + flex-direction: row; + align-items: center; + column-gap: 4px; + } +} + +.streaks { + display: flex; + flex-direction: column; + row-gap: 16px; + align-self: flex-start; + + .title { + margin: 0; + padding: 0; + font-size: 0.9em; + } + .icon { + width: 1.6em; + vertical-align: bottom; + position: relative; + top: 2px; + } + .highlight { + font-weight: 900; + color: #ff8f1f; } } + +.streaks ul.streaksWeek::before { + content: ''; + display: block; + width: 100%; + height: 1px; + background-color: var(--card-content-divider); + z-index: 1; + position: absolute; + margin-top: 9px; +} +.streaks ul.streaksWeek { + position: relative; + margin: 0; + padding: 0; + display: flex; + flex-direction: row; + list-style-type: none; + overflow: hidden; +} + +.streaks .dayWrapper { + color: white; + display: flex; + align-items: center; + justify-content: flex-start; + z-index: 2; + width: 48px; + position: relative; +} +.streaks .dayWrapper:last-child { + width: auto; +} +.streaks .dayWrapper .day { + border: 1px solid var(--card-content-divider); + background-color: var(--card-content-divider); + border-radius: 20px; + width: 16px; + height: 16px; + position: relative; + z-index: 5; +} +.streaks .dayWrapper::before { + content: ''; + display: block; + width: 100%; + background-color: var(--card-content-divider); + z-index: -1; + height: 2px; + position: absolute; +} + +.streaks .dayWrapper.checked .day { + border: 2px solid #18bc2d; + background-color: #18bc2d; +} +.streaks .checked:is(:has(+ .checked))::before { + background-color: #18bc2d; +} diff --git a/src/features/settings/components/RssSetting.tsx b/src/features/settings/components/RssSetting.tsx index 54be6d6f..daefce43 100644 --- a/src/features/settings/components/RssSetting.tsx +++ b/src/features/settings/components/RssSetting.tsx @@ -5,6 +5,7 @@ import { identifyUserCards, trackRssSourceAdd } from 'src/lib/analytics' import { isValidURL } from 'src/utils/UrlUtils' import { BsRssFill } from 'react-icons/bs' +import { Button } from 'src/components/Elements' import { useUserPreferences } from 'src/stores/preferences' import { SupportedCardType } from 'src/types' @@ -78,9 +79,13 @@ export const RssSetting = () => { ) : (
- +
)}
diff --git a/src/features/settings/components/SearchEngineSettings.tsx b/src/features/settings/components/SearchEngineSettings.tsx index ebf2d6ee..f3cc2154 100644 --- a/src/features/settings/components/SearchEngineSettings.tsx +++ b/src/features/settings/components/SearchEngineSettings.tsx @@ -49,10 +49,12 @@ export const SearchEngineSettings = () => {
<>
-

Add new Search Engine

-

- Can't find your favorite search engine? Add it here and it. -

+
+

Add new Search Engine

+

+ Can't find your favorite search engine? Add it here and it. +

+
diff --git a/src/features/settings/components/SourceSettings.tsx b/src/features/settings/components/SourceSettings.tsx index a1099179..b9da4deb 100644 --- a/src/features/settings/components/SourceSettings.tsx +++ b/src/features/settings/components/SourceSettings.tsx @@ -69,11 +69,13 @@ export const SourceSettings = () => {
<>
-

Add new Source

-

- Can't find your favorite source? Add its RSS feed URL here and it will be available in - your feed. -

+
+

Add new Source

+

+ Can't find your favorite source? Add its RSS feed URL here and it will be available + in your feed. +

+
diff --git a/src/features/settings/components/UserSettings/DeleteAccount.tsx b/src/features/settings/components/UserSettings/DeleteAccount.tsx new file mode 100644 index 00000000..02183067 --- /dev/null +++ b/src/features/settings/components/UserSettings/DeleteAccount.tsx @@ -0,0 +1,59 @@ +import { useState } from 'react' +import toast from 'react-simple-toasts' +import { Button, ConfirmModal } from 'src/components/Elements' +import { useAuth } from 'src/features/auth' +import { useDeleteAccount } from 'src/features/auth/api/deleteAccount' +import { trackUserDelete } from 'src/lib/analytics' + +export const DeleteAccount = () => { + const deleteAccountMutation = useDeleteAccount() + const { user, logout } = useAuth() + const [confirmDelete, setConfirmDelete] = useState(false) + + if (!user) { + return null + } + + const onDeleteAccount = () => { + deleteAccountMutation + .mutateAsync({ + userId: user.id, + }) + .then(() => { + logout() + trackUserDelete() + toast('Account deleted successfully', { theme: 'successToast' }) + }) + .catch(() => { + toast('Failed to delete account', { theme: 'defaultToast' }) + }) + } + + return ( +
+

+ Delete account? +
+ + This action is irreversible and will delete all your data. Proceed with caution. + +

+
+ setConfirmDelete(false)} + onConfirm={onDeleteAccount} + /> + + +
+
+ ) +} diff --git a/src/features/settings/components/UserSettings/UserInfo.tsx b/src/features/settings/components/UserSettings/UserInfo.tsx new file mode 100644 index 00000000..d17430ab --- /dev/null +++ b/src/features/settings/components/UserSettings/UserInfo.tsx @@ -0,0 +1,97 @@ +import { useState } from 'react' +import { FaGithub } from 'react-icons/fa' +import { FcGoogle } from 'react-icons/fc' +import { IoCheckmarkOutline } from 'react-icons/io5' +import { ReactComponent as AvatarPlaceholder } from 'src/assets/icons/avatar.svg' +import { ReactComponent as StreakIcon } from 'src/assets/icons/fire_icon.svg' +import { Button, ConfirmModal } from 'src/components/Elements' +import { useAuth } from 'src/features/auth' +import { pluralize } from 'src/utils/String' + +export const UserInfo = () => { + const { user } = useAuth() + const { logout, providerId, openAuthModal } = useAuth() + const providerName = providerId?.split('.')[0] || 'Unknown' + const [showLogout, setShowLogout] = useState(false) + + if (!user) { + return ( +
+ +
+
+ Join Hackertab +
+
+ Create a free account on Hackertab to sync, save bookmarks, and earn rewards. +
+
+ +
+
+
+ ) + } + + return ( +
+ setShowLogout(false)} + onConfirm={logout} + /> + {user?.imageURL && } +
+
{user.name}
+
+ {providerId == 'github.com' ? ( + + ) : providerId == 'google.com' ? ( + + ) : null} + Connected with {providerName} +
+
+ +
+
+ +
+

+ You're on{' '} + + {pluralize(user.streak || 1, 'day')} streak + +

+
+
    + {Array.from({ length: 5 }, (_, i) => { + const streak = user.streak || 1 + if (i < streak) { + return ( +
  • + + + +
  • + ) + } else { + return ( +
  • + +
  • + ) + } + })} +
+
+
+
+ ) +} diff --git a/src/features/settings/index.ts b/src/features/settings/index.ts index 0e607d08..74f1de9f 100644 --- a/src/features/settings/index.ts +++ b/src/features/settings/index.ts @@ -1,5 +1,6 @@ export * from './components/BookmarkSettings' export * from './components/GeneralSettings' + export * from './components/SearchEngineSettings' export * from './components/SourceSettings' export * from './components/TopicSettings' diff --git a/src/features/shareModal/components/ShareModal.tsx b/src/features/shareModal/components/ShareModal.tsx index cab95c4c..103e5bd1 100644 --- a/src/features/shareModal/components/ShareModal.tsx +++ b/src/features/shareModal/components/ShareModal.tsx @@ -17,6 +17,7 @@ import { WhatsappShareButton, } from 'react-share' import toast from 'react-simple-toasts' +import { Button } from 'src/components/Elements' import { twitterHandle } from 'src/config' import { trackLinkCopy, trackLinkShare } from 'src/lib/analytics' import { ShareModalData } from '../types' @@ -118,9 +119,14 @@ export const ShareModal = ({ showModal, closeModal, shareData }: ShareModalProps
- + +
diff --git a/src/lib/analytics.ts b/src/lib/analytics.ts index c03674e1..9d2b8d59 100644 --- a/src/lib/analytics.ts +++ b/src/lib/analytics.ts @@ -44,6 +44,9 @@ enum Verbs { DISABLE = 'Disable', SHARE = 'Share', COPY = 'Copy', + CONNECT = 'Connect', + DISCONNECT = 'Disconnect', + DELETE = 'Delete', } export enum Attributes { @@ -69,6 +72,7 @@ export enum Attributes { MAX_VISIBLE_CARDS = 'Max Visible Cards', DURATION = 'Duration', PROVIDER = 'Provider', + ADV = 'ADV', } const _SEP_ = ' ' @@ -376,6 +380,28 @@ export const trackLinkCopy = ({ }) } +export const trackUserConnect = (provider: string) => { + trackEvent({ + object: Objects.USER, + verb: Verbs.CONNECT, + attributes: { [Attributes.PROVIDER]: provider }, + }) +} + +export const trackUserDisconnect = () => { + trackEvent({ + object: Objects.USER, + verb: Verbs.DISCONNECT, + }) +} + +export const trackUserDelete = () => { + trackEvent({ + object: Objects.USER, + verb: Verbs.DELETE, + }) +} + // Identification export const identifyUserLanguages = (languages: string[]) => { @@ -405,6 +431,9 @@ export const identifyUserOccupation = (occupation: string) => { export const identifyUserMaxVisibleCards = (maxVisibleCards: number) => { identifyUserProperty(Attributes.MAX_VISIBLE_CARDS, maxVisibleCards) } +export const identifyAdvBlocked = (blocked: boolean) => { + identifyUserProperty(Attributes.ADV, blocked) +} // Private functions type trackEventProps = { @@ -452,7 +481,10 @@ const trackEvent = ({ object, verb, attributes }: trackEventProps) => { } } -const identifyUserProperty = (attributes: Attributes, value: string | number | string[]) => { +const identifyUserProperty = ( + attributes: Attributes, + value: string | number | string[] | boolean +) => { try { let formatedValue if (Array.isArray(value)) { diff --git a/src/lib/firebase.ts b/src/lib/firebase.ts new file mode 100644 index 00000000..8417befc --- /dev/null +++ b/src/lib/firebase.ts @@ -0,0 +1,15 @@ +import { initializeApp } from 'firebase/app' +import { getAuth } from 'firebase/auth' +import { FIREBASE_API_KEY } from 'src/config' + +const firebaseConfig = { + apiKey: FIREBASE_API_KEY, +} + +if (!FIREBASE_API_KEY) { + console.warn('Missing Firebase api Key') +} +// Initialize Firebase +const app = initializeApp(firebaseConfig) +const firebaseAuth = getAuth(app) +export { firebaseAuth } diff --git a/src/lib/interceptors/DefaultRequestInterceptor.ts b/src/lib/interceptors/DefaultRequestInterceptor.ts index be883e3c..8683e376 100644 --- a/src/lib/interceptors/DefaultRequestInterceptor.ts +++ b/src/lib/interceptors/DefaultRequestInterceptor.ts @@ -1,13 +1,19 @@ -import { AxiosRequestConfig } from 'axios' +import { InternalAxiosRequestConfig } from 'axios' import { API_ENDPOINT } from 'src/config' +import { getUserToken } from 'src/features/auth' import { isProduction } from 'src/utils/Environment' -export function DefaultRequestInterceptor(config: AxiosRequestConfig) { +export async function DefaultRequestInterceptor(config: InternalAxiosRequestConfig) { if (config) { config.baseURL = isProduction() ? API_ENDPOINT : '/api' if (config.headers) { config.headers.Accept = 'application/json' } + + const token = await getUserToken() + if (token) { + config.headers.authorization = `Bearer ${token}` + } } return config diff --git a/src/providers/AuthProvider.tsx b/src/providers/AuthProvider.tsx new file mode 100644 index 00000000..b0bc75e2 --- /dev/null +++ b/src/providers/AuthProvider.tsx @@ -0,0 +1,101 @@ +import { GithubAuthProvider, GoogleAuthProvider, signInWithCredential } from 'firebase/auth' +import { useCallback, useEffect } from 'react' +import { useNavigate, useSearchParams } from 'react-router-dom' +import toast from 'react-simple-toasts' +import { useAuth } from 'src/features/auth' +import { firebaseAuth } from 'src/lib/firebase' + +export const AuthProvider = ({ children }: { children: React.ReactNode }) => { + const navigate = useNavigate() + const [searchParams] = useSearchParams() + const { closeAuthModal, initState, setAuthError, openAuthModal, setConnecting } = useAuth() + + const connectTheUser = useCallback((token?: string | null, provider?: string | null) => { + const allowedProviders = ['google', 'github'] + if ((provider && !allowedProviders.includes(provider)) || !token) { + return Promise.resolve() + } + + setConnecting(true) + const authProvider = + provider === 'google' + ? GoogleAuthProvider.credential(null, token) + : GithubAuthProvider.credential(token) + + return signInWithCredential(firebaseAuth, authProvider) + .then((userCredential) => { + const user = userCredential.user + + initState({ + user: { + id: user.uid, + name: user.displayName || 'Anonymous', + imageURL: user.photoURL || '', + }, + providerId: authProvider.providerId, + }) + if (user.displayName) { + toast(`Welcome, ${user.displayName}`, { theme: 'successToast' }) + } + closeAuthModal() + navigate(window.location.pathname, { replace: true }) + }) + .finally(() => { + setConnecting(false) + }) + }, []) + + /** + * This effect is used to connect the user when the token is received from the background script + * on Chrome and Firefox extensions + */ + useEffect(() => { + const messageListener = (message: { + action: string + type?: string + access_token?: string + provider?: string + }) => { + if (message.type === 'TOKEN_RECEIVED') { + const { access_token: token, provider } = message + connectTheUser(token, provider).catch((error) => { + if (error && error.code === 'auth/account-exists-with-different-credential') { + setAuthError({ + message: + 'You have an account with a different provider. Please sign in with that provider to continue.', + }) + } + }) + } + } + + chrome.runtime?.onMessage.addListener(messageListener) + + return () => { + chrome.runtime?.onMessage.removeListener(messageListener) + } + }, []) + + /** + * This effect is used to connect the user when the user when the token is received from the URL + * on the web + */ + useEffect(() => { + const token = searchParams.get('access_token') + const provider = searchParams.get('provider') + connectTheUser(token, provider).catch((error) => { + openAuthModal() + if (error && error.code === 'auth/account-exists-with-different-credential') { + setAuthError({ + message: + 'This account is already connected with a different provider. Please sign in with that provider to continue.', + }) + } else { + setAuthError({ + message: 'Error signing in, Please try again', + }) + } + }) + }, [searchParams]) + return <>{children} +} diff --git a/src/providers/DesktopBreakpoint.tsx b/src/providers/DesktopBreakpoint.tsx new file mode 100644 index 00000000..32b31119 --- /dev/null +++ b/src/providers/DesktopBreakpoint.tsx @@ -0,0 +1,6 @@ +import { useMediaQuery } from 'react-responsive' + +export const DesktopBreakpoint = ({ children }: { children: React.ReactNode }) => { + const isDesktop = useMediaQuery({ minWidth: 768 }) + return isDesktop ? children : null +} diff --git a/src/providers/MobileBreakpoint.tsx b/src/providers/MobileBreakpoint.tsx new file mode 100644 index 00000000..194e3d17 --- /dev/null +++ b/src/providers/MobileBreakpoint.tsx @@ -0,0 +1,6 @@ +import { useMediaQuery } from 'react-responsive' + +export const MobileBreakpoint = ({ children }: { children: React.ReactNode }) => { + const isMobile = useMediaQuery({ maxWidth: 767 }) + return isMobile ? children : null +} diff --git a/src/stores/bookmarks.ts b/src/stores/bookmarks.ts index d99827b0..71f63f5c 100644 --- a/src/stores/bookmarks.ts +++ b/src/stores/bookmarks.ts @@ -1,24 +1,37 @@ -import { create } from 'zustand'; +import { create } from 'zustand' -import { BookmarkedPost } from "src/features/bookmarks"; -import { persist } from 'zustand/middleware'; +import { BookmarkedPost } from 'src/features/bookmarks' +import { persist } from 'zustand/middleware' type BookmarksState = { userBookmarks: BookmarkedPost[] -}; +} type BookmarksActions = { - bookmarkPost: (post: BookmarkedPost) => void; - unbookmarkPost: (post: BookmarkedPost) => void; - initState: (state: BookmarksState) => void; + bookmarkPost: (post: BookmarkedPost) => void + unbookmarkPost: (post: BookmarkedPost) => void + initState: (state: BookmarksState) => void + clear: () => void } - -export const useBookmarks = create(persist((set) => ({ - userBookmarks: [], - bookmarkPost: (post: BookmarkedPost) => set((state) => ({ userBookmarks: [post, ...state.userBookmarks] })), - unbookmarkPost: (post: BookmarkedPost) => set((state) => ({ userBookmarks: state.userBookmarks.filter((bookmarkedPost) => bookmarkedPost.url !== post.url), })), - initState: (newState: BookmarksState) => set(() => ({ userBookmarks: newState.userBookmarks })) -}), { - name: 'bookmarks_storage' -})); +export const useBookmarks = create( + persist( + (set) => ({ + userBookmarks: [], + bookmarkPost: (post: BookmarkedPost) => + set((state) => ({ userBookmarks: [post, ...state.userBookmarks] })), + unbookmarkPost: (post: BookmarkedPost) => + set((state) => ({ + userBookmarks: state.userBookmarks.filter( + (bookmarkedPost) => bookmarkedPost.url !== post.url + ), + })), + initState: (newState: BookmarksState) => + set(() => ({ userBookmarks: newState.userBookmarks })), + clear: () => set({ userBookmarks: [] }), + }), + { + name: 'bookmarks_storage', + } + ) +) diff --git a/src/utils/Environment.ts b/src/utils/Environment.ts index a80a14ef..1a2f781c 100644 --- a/src/utils/Environment.ts +++ b/src/utils/Environment.ts @@ -6,12 +6,12 @@ export const isDevelopment = (): boolean => { return import.meta.env.DEV } -export const isWebOrExtensionVersion = (): string => { +export const isWebOrExtensionVersion = (): 'web' | 'extension' => { const buildTarget = import.meta.env.VITE_BUILD_TARGET as 'web' | 'extension' | undefined return buildTarget || 'web' } -export const getBrowserName = (): string => { +export const getBrowserName = (): 'chrome' | 'firefox' | 'other' => { let userAgent = navigator.userAgent if (userAgent.match(/chrome|chromium|crios/i)) { return 'chrome' diff --git a/src/utils/Permissions.ts b/src/utils/Permissions.ts new file mode 100644 index 00000000..1390ee21 --- /dev/null +++ b/src/utils/Permissions.ts @@ -0,0 +1,28 @@ +import { getBrowserName } from './Environment' + +// Check host permissions before allowing the user to sign in with a provider. +// as Firefox may block the Signin on v3 if the extension does not have host permissions. +export const checkHostPermissions = async () => { + if (getBrowserName() !== 'firefox') { + return true + } + + const HOST_PERMISSIONS = chrome.runtime.getManifest().content_scripts || [] + const requiredHosts = HOST_PERMISSIONS.flatMap((permission) => { + return permission.matches || [] + }) + + return await chrome.permissions.contains({ origins: requiredHosts }) +} + +export const requestHostPermissions = async () => { + if (getBrowserName() !== 'firefox') { + return true + } + const HOST_PERMISSIONS = chrome.runtime.getManifest().content_scripts || [] + const requiredHosts = HOST_PERMISSIONS.flatMap((permission) => { + return permission.matches || [] + }) + + return await chrome.permissions.request({ origins: requiredHosts }) +} diff --git a/src/utils/String.ts b/src/utils/String.ts new file mode 100644 index 00000000..8e9eac37 --- /dev/null +++ b/src/utils/String.ts @@ -0,0 +1,2 @@ +export const pluralize = (count: number, noun: string, suffix = 's') => + `${count} ${noun}${count !== 1 ? suffix : ''}` diff --git a/vite.config.ts b/vite.config.mjs similarity index 100% rename from vite.config.ts rename to vite.config.mjs diff --git a/yarn.lock b/yarn.lock index b38ede3b..e76e564d 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1376,115 +1376,130 @@ resolved "https://registry.yarnpkg.com/@emotion/weak-memoize/-/weak-memoize-0.3.1.tgz#d0fce5d07b0620caa282b5131c297bb60f9d87e6" integrity sha512-EsBwpc7hBUJWAsNPBmJy4hxWx12v6bshQsldrVmjxJoc3isbxhOrF2IcCpaXxfvq03NwkI7sbsOLXbYuqF/8Ww== -"@esbuild/android-arm64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.18.20.tgz#984b4f9c8d0377443cc2dfcef266d02244593622" - integrity sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ== - -"@esbuild/android-arm@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.18.20.tgz#fedb265bc3a589c84cc11f810804f234947c3682" - integrity sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw== - -"@esbuild/android-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.18.20.tgz#35cf419c4cfc8babe8893d296cd990e9e9f756f2" - integrity sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg== - -"@esbuild/darwin-arm64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.18.20.tgz#08172cbeccf95fbc383399a7f39cfbddaeb0d7c1" - integrity sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA== - -"@esbuild/darwin-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.18.20.tgz#d70d5790d8bf475556b67d0f8b7c5bdff053d85d" - integrity sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ== - -"@esbuild/freebsd-arm64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.18.20.tgz#98755cd12707f93f210e2494d6a4b51b96977f54" - integrity sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw== - -"@esbuild/freebsd-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.18.20.tgz#c1eb2bff03915f87c29cece4c1a7fa1f423b066e" - integrity sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ== - -"@esbuild/linux-arm64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.18.20.tgz#bad4238bd8f4fc25b5a021280c770ab5fc3a02a0" - integrity sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA== - -"@esbuild/linux-arm@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.18.20.tgz#3e617c61f33508a27150ee417543c8ab5acc73b0" - integrity sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg== - -"@esbuild/linux-ia32@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.18.20.tgz#699391cccba9aee6019b7f9892eb99219f1570a7" - integrity sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA== - -"@esbuild/linux-loong64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.18.20.tgz#e6fccb7aac178dd2ffb9860465ac89d7f23b977d" - integrity sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg== - -"@esbuild/linux-mips64el@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.18.20.tgz#eeff3a937de9c2310de30622a957ad1bd9183231" - integrity sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ== - -"@esbuild/linux-ppc64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.18.20.tgz#2f7156bde20b01527993e6881435ad79ba9599fb" - integrity sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA== - -"@esbuild/linux-riscv64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.18.20.tgz#6628389f210123d8b4743045af8caa7d4ddfc7a6" - integrity sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A== - -"@esbuild/linux-s390x@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.18.20.tgz#255e81fb289b101026131858ab99fba63dcf0071" - integrity sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ== - -"@esbuild/linux-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.18.20.tgz#c7690b3417af318a9b6f96df3031a8865176d338" - integrity sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w== - -"@esbuild/netbsd-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.18.20.tgz#30e8cd8a3dded63975e2df2438ca109601ebe0d1" - integrity sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A== - -"@esbuild/openbsd-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.18.20.tgz#7812af31b205055874c8082ea9cf9ab0da6217ae" - integrity sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg== - -"@esbuild/sunos-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.18.20.tgz#d5c275c3b4e73c9b0ecd38d1ca62c020f887ab9d" - integrity sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ== - -"@esbuild/win32-arm64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.18.20.tgz#73bc7f5a9f8a77805f357fab97f290d0e4820ac9" - integrity sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg== - -"@esbuild/win32-ia32@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.18.20.tgz#ec93cbf0ef1085cc12e71e0d661d20569ff42102" - integrity sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g== - -"@esbuild/win32-x64@0.18.20": - version "0.18.20" - resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.18.20.tgz#786c5f41f043b07afb1af37683d7c33668858f6d" - integrity sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ== +"@esbuild/aix-ppc64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.25.1.tgz#c33cf6bbee34975626b01b80451cbb72b4c6c91d" + integrity sha512-kfYGy8IdzTGy+z0vFGvExZtxkFlA4zAxgKEahG9KE1ScBjpQnFsNOX8KTU5ojNru5ed5CVoJYXFtoxaq5nFbjQ== + +"@esbuild/android-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.25.1.tgz#ea766015c7d2655164f22100d33d7f0308a28d6d" + integrity sha512-50tM0zCJW5kGqgG7fQ7IHvQOcAn9TKiVRuQ/lN0xR+T2lzEFvAi1ZcS8DiksFcEpf1t/GYOeOfCAgDHFpkiSmA== + +"@esbuild/android-arm@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.25.1.tgz#e84d2bf2fe2e6177a0facda3a575b2139fd3cb9c" + integrity sha512-dp+MshLYux6j/JjdqVLnMglQlFu+MuVeNrmT5nk6q07wNhCdSnB7QZj+7G8VMUGh1q+vj2Bq8kRsuyA00I/k+Q== + +"@esbuild/android-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.25.1.tgz#58337bee3bc6d78d10425e5500bd11370cfdfbed" + integrity sha512-GCj6WfUtNldqUzYkN/ITtlhwQqGWu9S45vUXs7EIYf+7rCiiqH9bCloatO9VhxsL0Pji+PF4Lz2XXCES+Q8hDw== + +"@esbuild/darwin-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.25.1.tgz#a46805c1c585d451aa83be72500bd6e8495dd591" + integrity sha512-5hEZKPf+nQjYoSr/elb62U19/l1mZDdqidGfmFutVUjjUZrOazAtwK+Kr+3y0C/oeJfLlxo9fXb1w7L+P7E4FQ== + +"@esbuild/darwin-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.25.1.tgz#0643e003bb238c63fc93ddbee7d26a003be3cd98" + integrity sha512-hxVnwL2Dqs3fM1IWq8Iezh0cX7ZGdVhbTfnOy5uURtao5OIVCEyj9xIzemDi7sRvKsuSdtCAhMKarxqtlyVyfA== + +"@esbuild/freebsd-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.1.tgz#cff18da5469c09986b93e87979de5d6872fe8f8e" + integrity sha512-1MrCZs0fZa2g8E+FUo2ipw6jw5qqQiH+tERoS5fAfKnRx6NXH31tXBKI3VpmLijLH6yriMZsxJtaXUyFt/8Y4A== + +"@esbuild/freebsd-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.25.1.tgz#362fc09c2de14987621c1878af19203c46365dde" + integrity sha512-0IZWLiTyz7nm0xuIs0q1Y3QWJC52R8aSXxe40VUxm6BB1RNmkODtW6LHvWRrGiICulcX7ZvyH6h5fqdLu4gkww== + +"@esbuild/linux-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.25.1.tgz#aa90d5b02efc97a271e124e6d1cea490634f7498" + integrity sha512-jaN3dHi0/DDPelk0nLcXRm1q7DNJpjXy7yWaWvbfkPvI+7XNSc/lDOnCLN7gzsyzgu6qSAmgSvP9oXAhP973uQ== + +"@esbuild/linux-arm@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.25.1.tgz#dfcefcbac60a20918b19569b4b657844d39db35a" + integrity sha512-NdKOhS4u7JhDKw9G3cY6sWqFcnLITn6SqivVArbzIaf3cemShqfLGHYMx8Xlm/lBit3/5d7kXvriTUGa5YViuQ== + +"@esbuild/linux-ia32@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.25.1.tgz#6f9527077ccb7953ed2af02e013d4bac69f13754" + integrity sha512-OJykPaF4v8JidKNGz8c/q1lBO44sQNUQtq1KktJXdBLn1hPod5rE/Hko5ugKKZd+D2+o1a9MFGUEIUwO2YfgkQ== + +"@esbuild/linux-loong64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.25.1.tgz#287d2412a5456e5860c2839d42a4b51284d1697c" + integrity sha512-nGfornQj4dzcq5Vp835oM/o21UMlXzn79KobKlcs3Wz9smwiifknLy4xDCLUU0BWp7b/houtdrgUz7nOGnfIYg== + +"@esbuild/linux-mips64el@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.25.1.tgz#530574b9e1bc5d20f7a4f44c5f045e26f3783d57" + integrity sha512-1osBbPEFYwIE5IVB/0g2X6i1qInZa1aIoj1TdL4AaAb55xIIgbg8Doq6a5BzYWgr+tEcDzYH67XVnTmUzL+nXg== + +"@esbuild/linux-ppc64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.25.1.tgz#5d7e6b283a0b321ea42c6bc0abeb9eb99c1f5589" + integrity sha512-/6VBJOwUf3TdTvJZ82qF3tbLuWsscd7/1w+D9LH0W/SqUgM5/JJD0lrJ1fVIfZsqB6RFmLCe0Xz3fmZc3WtyVg== + +"@esbuild/linux-riscv64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.25.1.tgz#14fa0cd073c26b4ee2465d18cd1e18eea7859fa8" + integrity sha512-nSut/Mx5gnilhcq2yIMLMe3Wl4FK5wx/o0QuuCLMtmJn+WeWYoEGDN1ipcN72g1WHsnIbxGXd4i/MF0gTcuAjQ== + +"@esbuild/linux-s390x@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.25.1.tgz#e677b4b9d1b384098752266ccaa0d52a420dc1aa" + integrity sha512-cEECeLlJNfT8kZHqLarDBQso9a27o2Zd2AQ8USAEoGtejOrCYHNtKP8XQhMDJMtthdF4GBmjR2au3x1udADQQQ== + +"@esbuild/linux-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.25.1.tgz#f1c796b78fff5ce393658313e8c58613198d9954" + integrity sha512-xbfUhu/gnvSEg+EGovRc+kjBAkrvtk38RlerAzQxvMzlB4fXpCFCeUAYzJvrnhFtdeyVCDANSjJvOvGYoeKzFA== + +"@esbuild/netbsd-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.1.tgz#0d280b7dfe3973f111b02d5fe9f3063b92796d29" + integrity sha512-O96poM2XGhLtpTh+s4+nP7YCCAfb4tJNRVZHfIE7dgmax+yMP2WgMd2OecBuaATHKTHsLWHQeuaxMRnCsH8+5g== + +"@esbuild/netbsd-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.25.1.tgz#be663893931a4bb3f3a009c5cc24fa9681cc71c0" + integrity sha512-X53z6uXip6KFXBQ+Krbx25XHV/NCbzryM6ehOAeAil7X7oa4XIq+394PWGnwaSQ2WRA0KI6PUO6hTO5zeF5ijA== + +"@esbuild/openbsd-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.1.tgz#d9021b884233673a05dc1cc26de0bf325d824217" + integrity sha512-Na9T3szbXezdzM/Kfs3GcRQNjHzM6GzFBeU1/6IV/npKP5ORtp9zbQjvkDJ47s6BCgaAZnnnu/cY1x342+MvZg== + +"@esbuild/openbsd-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.25.1.tgz#9f1dc1786ed2e2938c404b06bcc48be9a13250de" + integrity sha512-T3H78X2h1tszfRSf+txbt5aOp/e7TAz3ptVKu9Oyir3IAOFPGV6O9c2naym5TOriy1l0nNf6a4X5UXRZSGX/dw== + +"@esbuild/sunos-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.25.1.tgz#89aac24a4b4115959b3f790192cf130396696c27" + integrity sha512-2H3RUvcmULO7dIE5EWJH8eubZAI4xw54H1ilJnRNZdeo8dTADEZ21w6J22XBkXqGJbe0+wnNJtw3UXRoLJnFEg== + +"@esbuild/win32-arm64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.25.1.tgz#354358647a6ea98ea6d243bf48bdd7a434999582" + integrity sha512-GE7XvrdOzrb+yVKB9KsRMq+7a2U/K5Cf/8grVFRAGJmfADr/e/ODQ134RK2/eeHqYV5eQRFxb1hY7Nr15fv1NQ== + +"@esbuild/win32-ia32@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.25.1.tgz#8cea7340f2647eba951a041dc95651e3908cd4cb" + integrity sha512-uOxSJCIcavSiT6UnBhBzE8wy3n0hOkJsBOzy7HDAuTDE++1DJMRRVCPGisULScHL+a/ZwdXPpXD3IyFKjA7K8A== + +"@esbuild/win32-x64@0.25.1": + version "0.25.1" + resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.25.1.tgz#7d79922cb2d88f9048f06393dbf62d2e4accb584" + integrity sha512-Y1EQdcfwMSeQN/ujR5VayLOJ1BHaK+ssyk0AEzPjC+t1lITgsnccPqFjb6V+LsTp/9Iov4ysfjxLaGJ9RPtkVg== "@eslint-community/eslint-utils@^4.2.0": version "4.4.0" @@ -1498,6 +1513,396 @@ resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.10.0.tgz#548f6de556857c8bb73bbee70c35dc82a2e74d63" integrity sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA== +"@firebase/analytics-compat@0.2.17": + version "0.2.17" + resolved "https://registry.yarnpkg.com/@firebase/analytics-compat/-/analytics-compat-0.2.17.tgz#c3cfc8ffb863d574ec26d86f9c8344d752832995" + integrity sha512-SJNVOeTvzdqZQvXFzj7yAirXnYcLDxh57wBFROfeowq/kRN1AqOw1tG6U4OiFOEhqi7s3xLze/LMkZatk2IEww== + dependencies: + "@firebase/analytics" "0.10.11" + "@firebase/analytics-types" "0.8.3" + "@firebase/component" "0.6.12" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/analytics-types@0.8.3": + version "0.8.3" + resolved "https://registry.yarnpkg.com/@firebase/analytics-types/-/analytics-types-0.8.3.tgz#d08cd39a6209693ca2039ba7a81570dfa6c1518f" + integrity sha512-VrIp/d8iq2g501qO46uGz3hjbDb8xzYMrbu8Tp0ovzIzrvJZ2fvmj649gTjge/b7cCCcjT0H37g1gVtlNhnkbg== + +"@firebase/analytics@0.10.11": + version "0.10.11" + resolved "https://registry.yarnpkg.com/@firebase/analytics/-/analytics-0.10.11.tgz#6896413e92613573af775c45050af889a43676da" + integrity sha512-zwuPiRE0+hgcS95JZbJ6DFQN4xYFO8IyGxpeePTV51YJMwCf3lkBa6FnZ/iXIqDKcBPMgMuuEZozI0BJWaLEYg== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/installations" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/app-check-compat@0.3.18": + version "0.3.18" + resolved "https://registry.yarnpkg.com/@firebase/app-check-compat/-/app-check-compat-0.3.18.tgz#abe63858fca86b61ea431e0d9e58ccb8bac1b275" + integrity sha512-qjozwnwYmAIdrsVGrJk+hnF1WBois54IhZR6gO0wtZQoTvWL/GtiA2F31TIgAhF0ayUiZhztOv1RfC7YyrZGDQ== + dependencies: + "@firebase/app-check" "0.8.11" + "@firebase/app-check-types" "0.5.3" + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/app-check-interop-types@0.3.3": + version "0.3.3" + resolved "https://registry.yarnpkg.com/@firebase/app-check-interop-types/-/app-check-interop-types-0.3.3.tgz#ed9c4a4f48d1395ef378f007476db3940aa5351a" + integrity sha512-gAlxfPLT2j8bTI/qfe3ahl2I2YcBQ8cFIBdhAQA4I2f3TndcO+22YizyGYuttLHPQEpWkhmpFW60VCFEPg4g5A== + +"@firebase/app-check-types@0.5.3": + version "0.5.3" + resolved "https://registry.yarnpkg.com/@firebase/app-check-types/-/app-check-types-0.5.3.tgz#38ba954acf4bffe451581a32fffa20337f11d8e5" + integrity sha512-hyl5rKSj0QmwPdsAxrI5x1otDlByQ7bvNvVt8G/XPO2CSwE++rmSVf3VEhaeOR4J8ZFaF0Z0NDSmLejPweZ3ng== + +"@firebase/app-check@0.8.11": + version "0.8.11" + resolved "https://registry.yarnpkg.com/@firebase/app-check/-/app-check-0.8.11.tgz#3c67148046fea0a0a9a1eecf1a17fdc31a76eda7" + integrity sha512-42zIfRI08/7bQqczAy7sY2JqZYEv3a1eNa4fLFdtJ54vNevbBIRSEA3fZgRqWFNHalh5ohsBXdrYgFqaRIuCcQ== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/app-compat@0.2.48": + version "0.2.48" + resolved "https://registry.yarnpkg.com/@firebase/app-compat/-/app-compat-0.2.48.tgz#4cc013dc53b12c9c2ebda6369bbeb99f3cc59975" + integrity sha512-wVNU1foBIaJncUmiALyRxhHHHC3ZPMLIETTAk+2PG87eP9B/IDBsYUiTpHyboDPEI8CgBPat/zN2v+Snkz6lBw== + dependencies: + "@firebase/app" "0.10.18" + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/app-types@0.9.3": + version "0.9.3" + resolved "https://registry.yarnpkg.com/@firebase/app-types/-/app-types-0.9.3.tgz#8408219eae9b1fb74f86c24e7150a148460414ad" + integrity sha512-kRVpIl4vVGJ4baogMDINbyrIOtOxqhkZQg4jTq3l8Lw6WSk0xfpEYzezFu+Kl4ve4fbPl79dvwRtaFqAC/ucCw== + +"@firebase/app@0.10.18": + version "0.10.18" + resolved "https://registry.yarnpkg.com/@firebase/app/-/app-0.10.18.tgz#219d897beedcc833ab6d7bdc4ea810ece9e32df6" + integrity sha512-VuqEwD/QRisKd/zsFsqgvSAx34mZ3WEF47i97FD6Vw4GWAhdjepYf0Hmi6K0b4QMSgWcv/x0C30Slm5NjjERXg== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + idb "7.1.1" + tslib "^2.1.0" + +"@firebase/auth-compat@0.5.17": + version "0.5.17" + resolved "https://registry.yarnpkg.com/@firebase/auth-compat/-/auth-compat-0.5.17.tgz#60d9222dae734fb8740ac565e342dc11f92d2392" + integrity sha512-Shi6rqLqzU9KLXnUCmlLvVByq1kiG3oe7Wpbf5m1CgS7NiRx2pSSn0HLaRRozdkaizNzMGGj+3oHmNYQ7kU6xA== + dependencies: + "@firebase/auth" "1.8.2" + "@firebase/auth-types" "0.12.3" + "@firebase/component" "0.6.12" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/auth-interop-types@0.2.4": + version "0.2.4" + resolved "https://registry.yarnpkg.com/@firebase/auth-interop-types/-/auth-interop-types-0.2.4.tgz#176a08686b0685596ff03d7879b7e4115af53de0" + integrity sha512-JPgcXKCuO+CWqGDnigBtvo09HeBs5u/Ktc2GaFj2m01hLarbxthLNm7Fk8iOP1aqAtXV+fnnGj7U28xmk7IwVA== + +"@firebase/auth-types@0.12.3": + version "0.12.3" + resolved "https://registry.yarnpkg.com/@firebase/auth-types/-/auth-types-0.12.3.tgz#650e54a36060b5ea012075ddbd3cb26566334c41" + integrity sha512-Zq9zI0o5hqXDtKg6yDkSnvMCMuLU6qAVS51PANQx+ZZX5xnzyNLEBO3GZgBUPsV5qIMFhjhqmLDxUqCbnAYy2A== + +"@firebase/auth@1.8.2": + version "1.8.2" + resolved "https://registry.yarnpkg.com/@firebase/auth/-/auth-1.8.2.tgz#4559dfabe40bc7a0605fb2a73f401d83cb32fe80" + integrity sha512-q+071y2LWe0bVnjqaX3BscqZwzdP0GKN2YBKapLq4bV88MPfCtWwGKmDhNDEDUmioOjudGXkUY5cvvKqk3mlUg== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/component@0.6.12": + version "0.6.12" + resolved "https://registry.yarnpkg.com/@firebase/component/-/component-0.6.12.tgz#08905a534e9b769164e7e1b1e80f6e7611eb67f3" + integrity sha512-YnxqjtohLbnb7raXt2YuA44cC1wA9GiehM/cmxrsoxKlFxBLy2V0OkRSj9gpngAE0UoJ421Wlav9ycO7lTPAUw== + dependencies: + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/data-connect@0.2.0": + version "0.2.0" + resolved "https://registry.yarnpkg.com/@firebase/data-connect/-/data-connect-0.2.0.tgz#7133cb40096466dc17ad01a50a5f2c383605df20" + integrity sha512-7OrZtQoLSk2fiGijhIdUnTSqEFti3h1EMhw9nNiSZ6jJGduw4Pz6jrVvxjpZJtGH/JiljbMkBnPBS2h8CTRKEw== + dependencies: + "@firebase/auth-interop-types" "0.2.4" + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/database-compat@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@firebase/database-compat/-/database-compat-2.0.2.tgz#9ddf474b593766a41ea576185cdf115e28f0cb50" + integrity sha512-5zvdnMsfDHvrQAVM6jBS7CkBpu+z3YbpFdhxRsrK1FP45IEfxlzpeuEUb17D/tpM10vfq4Ok0x5akIBaCv7gfA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/database" "1.0.11" + "@firebase/database-types" "1.0.8" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/database-types@1.0.8": + version "1.0.8" + resolved "https://registry.yarnpkg.com/@firebase/database-types/-/database-types-1.0.8.tgz#eddcce594be118bf9aebb043b5a6d51cfb6de620" + integrity sha512-6lPWIGeufhUq1heofZULyVvWFhD01TUrkkB9vyhmksjZ4XF7NaivQp9rICMk7QNhqwa+uDCaj4j+Q8qqcSVZ9g== + dependencies: + "@firebase/app-types" "0.9.3" + "@firebase/util" "1.10.3" + +"@firebase/database@1.0.11": + version "1.0.11" + resolved "https://registry.yarnpkg.com/@firebase/database/-/database-1.0.11.tgz#5b9960a07a0d49361f67fc69affc779cae4e07e3" + integrity sha512-gLrw/XeioswWUXgpVKCPAzzoOuvYNqK5fRUeiJTzO7Mlp9P6ylFEyPJlRBl1djqYye641r3MX6AmIeMXwjgwuQ== + dependencies: + "@firebase/app-check-interop-types" "0.3.3" + "@firebase/auth-interop-types" "0.2.4" + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + faye-websocket "0.11.4" + tslib "^2.1.0" + +"@firebase/firestore-compat@0.3.41": + version "0.3.41" + resolved "https://registry.yarnpkg.com/@firebase/firestore-compat/-/firestore-compat-0.3.41.tgz#434d86fd603b5ebcde19b5695b9b2a53bb23be8b" + integrity sha512-J/PgWKEt0yugETOE7lOabT16hsV21cLzSxERD7ZhaiwBQkBTSf0Mx9RhjZRT0Ttqe4weM90HGZFyUBqYA73fVA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/firestore" "4.7.6" + "@firebase/firestore-types" "3.0.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/firestore-types@3.0.3": + version "3.0.3" + resolved "https://registry.yarnpkg.com/@firebase/firestore-types/-/firestore-types-3.0.3.tgz#7d0c3dd8850c0193d8f5ee0cc8f11961407742c1" + integrity sha512-hD2jGdiWRxB/eZWF89xcK9gF8wvENDJkzpVFb4aGkzfEaKxVRD1kjz1t1Wj8VZEp2LCB53Yx1zD8mrhQu87R6Q== + +"@firebase/firestore@4.7.6": + version "4.7.6" + resolved "https://registry.yarnpkg.com/@firebase/firestore/-/firestore-4.7.6.tgz#877a37b615d86c61ac9ba7ddd0a967feb99380ac" + integrity sha512-aVDboR+upR/44qZDLR4tnZ9pepSOFBbDJnwk7eWzmTyQq2nZAVG+HIhrqpQawmUVcDRkuJv2K2UT2+oqR8F8TA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + "@firebase/webchannel-wrapper" "1.0.3" + "@grpc/grpc-js" "~1.9.0" + "@grpc/proto-loader" "^0.7.8" + tslib "^2.1.0" + +"@firebase/functions-compat@0.3.18": + version "0.3.18" + resolved "https://registry.yarnpkg.com/@firebase/functions-compat/-/functions-compat-0.3.18.tgz#c6a4f6b0762c6990db0aab244420c3e1240fda2b" + integrity sha512-N7+RN5GVus2ORB8cqfSNhfSn4iaYws6F8uCCfn4mtjC7zYS/KH6muzNAhZUdUqlv5YazbVmvxlAoYYF39i8Qzg== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/functions" "0.12.1" + "@firebase/functions-types" "0.6.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/functions-types@0.6.3": + version "0.6.3" + resolved "https://registry.yarnpkg.com/@firebase/functions-types/-/functions-types-0.6.3.tgz#f5faf770248b13f45d256f614230da6a11bfb654" + integrity sha512-EZoDKQLUHFKNx6VLipQwrSMh01A1SaL3Wg6Hpi//x6/fJ6Ee4hrAeswK99I5Ht8roiniKHw4iO0B1Oxj5I4plg== + +"@firebase/functions@0.12.1": + version "0.12.1" + resolved "https://registry.yarnpkg.com/@firebase/functions/-/functions-0.12.1.tgz#09ca7386619b0c50c535f6695e5be0712d3e4730" + integrity sha512-QucRiFrvMMmIGTRhL7ZK2IeBnAWP7lAmfFREMpEtX47GjVqDqGxdFs+Mg7XBzxSc9UjDO4Rxf+aE9xJHU6bGwg== + dependencies: + "@firebase/app-check-interop-types" "0.3.3" + "@firebase/auth-interop-types" "0.2.4" + "@firebase/component" "0.6.12" + "@firebase/messaging-interop-types" "0.2.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/installations-compat@0.2.12": + version "0.2.12" + resolved "https://registry.yarnpkg.com/@firebase/installations-compat/-/installations-compat-0.2.12.tgz#ee6396f3cc787c0dd4fc5dd87fec1db9dbb40c97" + integrity sha512-RhcGknkxmFu92F6Jb3rXxv6a4sytPjJGifRZj8MSURPuv2Xu+/AispCXEfY1ZraobhEHTG5HLGsP6R4l9qB5aA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/installations" "0.6.12" + "@firebase/installations-types" "0.5.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/installations-types@0.5.3": + version "0.5.3" + resolved "https://registry.yarnpkg.com/@firebase/installations-types/-/installations-types-0.5.3.tgz#cac8a14dd49f09174da9df8ae453f9b359c3ef2f" + integrity sha512-2FJI7gkLqIE0iYsNQ1P751lO3hER+Umykel+TkLwHj6plzWVxqvfclPUZhcKFVQObqloEBTmpi2Ozn7EkCABAA== + +"@firebase/installations@0.6.12": + version "0.6.12" + resolved "https://registry.yarnpkg.com/@firebase/installations/-/installations-0.6.12.tgz#6d9ad14e60caa8fae4ec0120c0e46ceb9d6fbdae" + integrity sha512-ES/WpuAV2k2YtBTvdaknEo7IY8vaGjIjS3zhnHSAIvY9KwTR8XZFXOJoZ3nSkjN1A5R4MtEh+07drnzPDg9vaw== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/util" "1.10.3" + idb "7.1.1" + tslib "^2.1.0" + +"@firebase/logger@0.4.4": + version "0.4.4" + resolved "https://registry.yarnpkg.com/@firebase/logger/-/logger-0.4.4.tgz#29e8379d20fd1149349a195ee6deee4573a86f48" + integrity sha512-mH0PEh1zoXGnaR8gD1DeGeNZtWFKbnz9hDO91dIml3iou1gpOnLqXQ2dJfB71dj6dpmUjcQ6phY3ZZJbjErr9g== + dependencies: + tslib "^2.1.0" + +"@firebase/messaging-compat@0.2.16": + version "0.2.16" + resolved "https://registry.yarnpkg.com/@firebase/messaging-compat/-/messaging-compat-0.2.16.tgz#533af4542a54b932146d175d5687aedd428be972" + integrity sha512-9HZZ88Ig3zQ0ok/Pwt4gQcNsOhoEy8hDHoGsV1am6ulgMuGuDVD2gl11Lere2ksL+msM12Lddi2x/7TCqmODZw== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/messaging" "0.12.16" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/messaging-interop-types@0.2.3": + version "0.2.3" + resolved "https://registry.yarnpkg.com/@firebase/messaging-interop-types/-/messaging-interop-types-0.2.3.tgz#e647c9cd1beecfe6a6e82018a6eec37555e4da3e" + integrity sha512-xfzFaJpzcmtDjycpDeCUj0Ge10ATFi/VHVIvEEjDNc3hodVBQADZ7BWQU7CuFpjSHE+eLuBI13z5F/9xOoGX8Q== + +"@firebase/messaging@0.12.16": + version "0.12.16" + resolved "https://registry.yarnpkg.com/@firebase/messaging/-/messaging-0.12.16.tgz#bd8a768274bdc4368396bd9eaa356bffb998bef2" + integrity sha512-VJ8sCEIeP3+XkfbJA7410WhYGHdloYFZXoHe/vt+vNVDGw8JQPTQSVTRvjrUprEf5I4Tbcnpr2H34lS6zhCHSA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/installations" "0.6.12" + "@firebase/messaging-interop-types" "0.2.3" + "@firebase/util" "1.10.3" + idb "7.1.1" + tslib "^2.1.0" + +"@firebase/performance-compat@0.2.12": + version "0.2.12" + resolved "https://registry.yarnpkg.com/@firebase/performance-compat/-/performance-compat-0.2.12.tgz#069284005e3f29339b570ee517b813d9bdbc0a89" + integrity sha512-DyCbDTIwtBTGsEiQxTz/TD23a0na2nrDozceQ5kVkszyFYvliB0YK/9el0wAGIG91SqgTG9pxHtYErzfZc0VWw== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/performance" "0.6.12" + "@firebase/performance-types" "0.2.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/performance-types@0.2.3": + version "0.2.3" + resolved "https://registry.yarnpkg.com/@firebase/performance-types/-/performance-types-0.2.3.tgz#5ce64e90fa20ab5561f8b62a305010cf9fab86fb" + integrity sha512-IgkyTz6QZVPAq8GSkLYJvwSLr3LS9+V6vNPQr0x4YozZJiLF5jYixj0amDtATf1X0EtYHqoPO48a9ija8GocxQ== + +"@firebase/performance@0.6.12": + version "0.6.12" + resolved "https://registry.yarnpkg.com/@firebase/performance/-/performance-0.6.12.tgz#58763cbbe31673351e1494875a5c10c6b68e1322" + integrity sha512-8mYL4z2jRlKXAi2hjk4G7o2sQLnJCCuTbyvti/xmHf5ZvOIGB01BZec0aDuBIXO+H1MLF62dbye/k91Fr+yc8g== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/installations" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/remote-config-compat@0.2.12": + version "0.2.12" + resolved "https://registry.yarnpkg.com/@firebase/remote-config-compat/-/remote-config-compat-0.2.12.tgz#ae0b597b3228deef0e3c6b2c6e631f19213eca4c" + integrity sha512-91jLWPtubIuPBngg9SzwvNCWzhMLcyBccmt7TNZP+y1cuYFNOWWHKUXQ3IrxCLB7WwLqQaEu7fTDAjHsTyBsSw== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/remote-config" "0.5.0" + "@firebase/remote-config-types" "0.4.0" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/remote-config-types@0.4.0": + version "0.4.0" + resolved "https://registry.yarnpkg.com/@firebase/remote-config-types/-/remote-config-types-0.4.0.tgz#91b9a836d5ca30ced68c1516163b281fbb544537" + integrity sha512-7p3mRE/ldCNYt8fmWMQ/MSGRmXYlJ15Rvs9Rk17t8p0WwZDbeK7eRmoI1tvCPaDzn9Oqh+yD6Lw+sGLsLg4kKg== + +"@firebase/remote-config@0.5.0": + version "0.5.0" + resolved "https://registry.yarnpkg.com/@firebase/remote-config/-/remote-config-0.5.0.tgz#30212fa77adba8a62fc6408eb32122147ae80790" + integrity sha512-weiEbpBp5PBJTHUWR4GwI7ZacaAg68BKha5QnZ8Go65W4oQjEWqCW/rfskABI/OkrGijlL3CUmCB/SA6mVo0qA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/installations" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/storage-compat@0.3.15": + version "0.3.15" + resolved "https://registry.yarnpkg.com/@firebase/storage-compat/-/storage-compat-0.3.15.tgz#02a18e0e1866849206dba7b075a4dcd99489aae7" + integrity sha512-Z9afjrK2O9o1ZHWCpprCGZ1BTc3BbvpZvi6tkSteC8H3W/fMM6x+RoSunlzD3hEVV5bkbwdJIqNClLMchvyoPA== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/storage" "0.13.5" + "@firebase/storage-types" "0.8.3" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/storage-types@0.8.3": + version "0.8.3" + resolved "https://registry.yarnpkg.com/@firebase/storage-types/-/storage-types-0.8.3.tgz#2531ef593a3452fc12c59117195d6485c6632d3d" + integrity sha512-+Muk7g9uwngTpd8xn9OdF/D48uiQ7I1Fae7ULsWPuKoCH3HU7bfFPhxtJYzyhjdniowhuDpQcfPmuNRAqZEfvg== + +"@firebase/storage@0.13.5": + version "0.13.5" + resolved "https://registry.yarnpkg.com/@firebase/storage/-/storage-0.13.5.tgz#108c86c9cd359aebd306882eb61ce6a8b1deb417" + integrity sha512-sB/7HNuW0N9tITyD0RxVLNCROuCXkml5i/iPqjwOGKC0xiUfpCOjBE+bb0ABMoN1qYZfqk0y9IuI2TdomjmkNw== + dependencies: + "@firebase/component" "0.6.12" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/util@1.10.3": + version "1.10.3" + resolved "https://registry.yarnpkg.com/@firebase/util/-/util-1.10.3.tgz#63fc5fea7b36236219c4875731597494416678d1" + integrity sha512-wfoF5LTy0m2ufUapV0ZnpcGQvuavTbJ5Qr1Ze9OJGL70cSMvhDyjS4w2121XdA3lGZSTOsDOyGhpoDtYwck85A== + dependencies: + tslib "^2.1.0" + +"@firebase/vertexai@1.0.3": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@firebase/vertexai/-/vertexai-1.0.3.tgz#815efed8c16105676ea090daed963be72a01fb68" + integrity sha512-SQHg/RPb3LwQs/xiLcvAZYz9NXyDSZUIIwvgsKh6e4wdULAfyPCZIu6Y2ZYIhZLfk9Q44cKZ+++7RPTaqQJdYA== + dependencies: + "@firebase/app-check-interop-types" "0.3.3" + "@firebase/component" "0.6.12" + "@firebase/logger" "0.4.4" + "@firebase/util" "1.10.3" + tslib "^2.1.0" + +"@firebase/webchannel-wrapper@1.0.3": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@firebase/webchannel-wrapper/-/webchannel-wrapper-1.0.3.tgz#a73bab8eb491d7b8b7be2f0e6c310647835afe83" + integrity sha512-2xCRM9q9FlzGZCdgDMJwc0gyUkWFtkosy7Xxr6sFgQwn+wMNIWd7xIvYNauU1r64B5L5rsGKy/n9TKJ0aAFeqQ== + "@floating-ui/core@^1.4.2": version "1.5.2" resolved "https://registry.yarnpkg.com/@floating-ui/core/-/core-1.5.2.tgz#53a0f7a98c550e63134d504f26804f6b83dbc071" @@ -1518,6 +1923,24 @@ resolved "https://registry.yarnpkg.com/@floating-ui/utils/-/utils-0.1.6.tgz#22958c042e10b67463997bd6ea7115fe28cbcaf9" integrity sha512-OfX7E2oUDYxtBvsuS4e/jSn4Q9Qb6DzgeYtsAdkPZ47znpoNsMgZw0+tVijiv3uGNR6dgNlty6r9rzIzHjtd/A== +"@grpc/grpc-js@~1.9.0": + version "1.9.15" + resolved "https://registry.yarnpkg.com/@grpc/grpc-js/-/grpc-js-1.9.15.tgz#433d7ac19b1754af690ea650ab72190bd700739b" + integrity sha512-nqE7Hc0AzI+euzUwDAy0aY5hCp10r734gMGRdU+qOPX0XSceI2ULrcXB5U2xSc5VkWwalCj4M7GzCAygZl2KoQ== + dependencies: + "@grpc/proto-loader" "^0.7.8" + "@types/node" ">=12.12.47" + +"@grpc/proto-loader@^0.7.8": + version "0.7.13" + resolved "https://registry.yarnpkg.com/@grpc/proto-loader/-/proto-loader-0.7.13.tgz#f6a44b2b7c9f7b609f5748c6eac2d420e37670cf" + integrity sha512-AiXO/bfe9bmxBjxxtYxFAXGZvMaN5s8kO+jBHAJCON8rJoB5YS/D6X7ZNc6XQkuHNmyl4CYaMI1fJ/Gn27RGGw== + dependencies: + lodash.camelcase "^4.3.0" + long "^5.0.0" + protobufjs "^7.2.5" + yargs "^17.7.2" + "@jest/expect-utils@^29.7.0": version "29.7.0" resolved "https://registry.yarnpkg.com/@jest/expect-utils/-/expect-utils-29.7.0.tgz#023efe5d26a8a70f21677d0a1afc0f0a44e3a1c6" @@ -1628,6 +2051,59 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" +"@protobufjs/aspromise@^1.1.1", "@protobufjs/aspromise@^1.1.2": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@protobufjs/aspromise/-/aspromise-1.1.2.tgz#9b8b0cc663d669a7d8f6f5d0893a14d348f30fbf" + integrity sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ== + +"@protobufjs/base64@^1.1.2": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@protobufjs/base64/-/base64-1.1.2.tgz#4c85730e59b9a1f1f349047dbf24296034bb2735" + integrity sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg== + +"@protobufjs/codegen@^2.0.4": + version "2.0.4" + resolved "https://registry.yarnpkg.com/@protobufjs/codegen/-/codegen-2.0.4.tgz#7ef37f0d010fb028ad1ad59722e506d9262815cb" + integrity sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg== + +"@protobufjs/eventemitter@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz#355cbc98bafad5978f9ed095f397621f1d066b70" + integrity sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q== + +"@protobufjs/fetch@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@protobufjs/fetch/-/fetch-1.1.0.tgz#ba99fb598614af65700c1619ff06d454b0d84c45" + integrity sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ== + dependencies: + "@protobufjs/aspromise" "^1.1.1" + "@protobufjs/inquire" "^1.1.0" + +"@protobufjs/float@^1.0.2": + version "1.0.2" + resolved "https://registry.yarnpkg.com/@protobufjs/float/-/float-1.0.2.tgz#5e9e1abdcb73fc0a7cb8b291df78c8cbd97b87d1" + integrity sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ== + +"@protobufjs/inquire@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@protobufjs/inquire/-/inquire-1.1.0.tgz#ff200e3e7cf2429e2dcafc1140828e8cc638f089" + integrity sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q== + +"@protobufjs/path@^1.1.2": + version "1.1.2" + resolved "https://registry.yarnpkg.com/@protobufjs/path/-/path-1.1.2.tgz#6cc2b20c5c9ad6ad0dccfd21ca7673d8d7fbf68d" + integrity sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA== + +"@protobufjs/pool@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@protobufjs/pool/-/pool-1.1.0.tgz#09fd15f2d6d3abfa9b65bc366506d6ad7846ff54" + integrity sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw== + +"@protobufjs/utf8@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@protobufjs/utf8/-/utf8-1.1.0.tgz#a777360b5b39a1a2e5106f8e858f2fd2d060c570" + integrity sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw== + "@reach/portal@^0.13.0": version "0.13.2" resolved "https://registry.yarnpkg.com/@reach/portal/-/portal-0.13.2.tgz#6f2a6f4afc14894bde9c6435667bb9b660887ed9" @@ -1659,6 +2135,106 @@ estree-walker "^2.0.2" picomatch "^2.3.1" +"@rollup/rollup-android-arm-eabi@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.37.0.tgz#9bedc746a97fe707154086365f269ced92ff4aa9" + integrity sha512-l7StVw6WAa8l3vA1ov80jyetOAEo1FtHvZDbzXDO/02Sq/QVvqlHkYoFwDJPIMj0GKiistsBudfx5tGFnwYWDQ== + +"@rollup/rollup-android-arm64@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.37.0.tgz#6edc6ffc8af8773e4bc28c72894dd5e846b8ee6c" + integrity sha512-6U3SlVyMxezt8Y+/iEBcbp945uZjJwjZimu76xoG7tO1av9VO691z8PkhzQ85ith2I8R2RddEPeSfcbyPfD4hA== + +"@rollup/rollup-darwin-arm64@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.37.0.tgz#737a7b8be9ff79bd24a7efaae0903e8c66ac0676" + integrity sha512-+iTQ5YHuGmPt10NTzEyMPbayiNTcOZDWsbxZYR1ZnmLnZxG17ivrPSWFO9j6GalY0+gV3Jtwrrs12DBscxnlYA== + +"@rollup/rollup-darwin-x64@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.37.0.tgz#a6a697bb685ca9462a7caeea5f22f6a686acff1f" + integrity sha512-m8W2UbxLDcmRKVjgl5J/k4B8d7qX2EcJve3Sut7YGrQoPtCIQGPH5AMzuFvYRWZi0FVS0zEY4c8uttPfX6bwYQ== + +"@rollup/rollup-freebsd-arm64@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.37.0.tgz#18113e8e133ccb6de4b9dc9d3e09f7acff344cb7" + integrity sha512-FOMXGmH15OmtQWEt174v9P1JqqhlgYge/bUjIbiVD1nI1NeJ30HYT9SJlZMqdo1uQFyt9cz748F1BHghWaDnVA== + +"@rollup/rollup-freebsd-x64@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.37.0.tgz#5e56ffd4a0d7ccfcbc86867c40b8f0e6a2c0c81e" + integrity sha512-SZMxNttjPKvV14Hjck5t70xS3l63sbVwl98g3FlVVx2YIDmfUIy29jQrsw06ewEYQ8lQSuY9mpAPlmgRD2iSsA== + +"@rollup/rollup-linux-arm-gnueabihf@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.37.0.tgz#5addf1a51e1495ae7ff28d26442a88adf629c980" + integrity sha512-hhAALKJPidCwZcj+g+iN+38SIOkhK2a9bqtJR+EtyxrKKSt1ynCBeqrQy31z0oWU6thRZzdx53hVgEbRkuI19w== + +"@rollup/rollup-linux-arm-musleabihf@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.37.0.tgz#00cddb9ab51086c5f2cd33cd4738259e24be4e73" + integrity sha512-jUb/kmn/Gd8epbHKEqkRAxq5c2EwRt0DqhSGWjPFxLeFvldFdHQs/n8lQ9x85oAeVb6bHcS8irhTJX2FCOd8Ag== + +"@rollup/rollup-linux-arm64-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.37.0.tgz#c3b4324496236b6fd9f31fda5701c6d6060b1512" + integrity sha512-oNrJxcQT9IcbcmKlkF+Yz2tmOxZgG9D9GRq+1OE6XCQwCVwxixYAa38Z8qqPzQvzt1FCfmrHX03E0pWoXm1DqA== + +"@rollup/rollup-linux-arm64-musl@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.37.0.tgz#b5222180bb1a50e6e9bc8263efd771c1ce770b6f" + integrity sha512-pfxLBMls+28Ey2enpX3JvjEjaJMBX5XlPCZNGxj4kdJyHduPBXtxYeb8alo0a7bqOoWZW2uKynhHxF/MWoHaGQ== + +"@rollup/rollup-linux-loongarch64-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.37.0.tgz#5660181c1c1efb7b19c7a531d496e685236c5ce7" + integrity sha512-yCE0NnutTC/7IGUq/PUHmoeZbIwq3KRh02e9SfFh7Vmc1Z7atuJRYWhRME5fKgT8aS20mwi1RyChA23qSyRGpA== + +"@rollup/rollup-linux-powerpc64le-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.37.0.tgz#8273166495d2f5d3fbc556cf42a5a6e24b78bdab" + integrity sha512-NxcICptHk06E2Lh3a4Pu+2PEdZ6ahNHuK7o6Np9zcWkrBMuv21j10SQDJW3C9Yf/A/P7cutWoC/DptNLVsZ0VQ== + +"@rollup/rollup-linux-riscv64-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.37.0.tgz#9677e39288ccc91ebcd707cdd794732d701cd174" + integrity sha512-PpWwHMPCVpFZLTfLq7EWJWvrmEuLdGn1GMYcm5MV7PaRgwCEYJAwiN94uBuZev0/J/hFIIJCsYw4nLmXA9J7Pw== + +"@rollup/rollup-linux-riscv64-musl@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.37.0.tgz#71cc5ca7be1ed263357618bfe4f8f50c09725a7e" + integrity sha512-DTNwl6a3CfhGTAOYZ4KtYbdS8b+275LSLqJVJIrPa5/JuIufWWZ/QFvkxp52gpmguN95eujrM68ZG+zVxa8zHA== + +"@rollup/rollup-linux-s390x-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.37.0.tgz#6b0b7df33eb32b0ee7423898b183acc1b5fee33e" + integrity sha512-hZDDU5fgWvDdHFuExN1gBOhCuzo/8TMpidfOR+1cPZJflcEzXdCy1LjnklQdW8/Et9sryOPJAKAQRw8Jq7Tg+A== + +"@rollup/rollup-linux-x64-gnu@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.37.0.tgz#52c27717d3c4819d13b5ebc2373ddea099d2e71b" + integrity sha512-pKivGpgJM5g8dwj0ywBwe/HeVAUSuVVJhUTa/URXjxvoyTT/AxsLTAbkHkDHG7qQxLoW2s3apEIl26uUe08LVQ== + +"@rollup/rollup-linux-x64-musl@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.37.0.tgz#c134a22d30642345de8b799c816345674bf68019" + integrity sha512-E2lPrLKE8sQbY/2bEkVTGDEk4/49UYRVWgj90MY8yPjpnGBQ+Xi1Qnr7b7UIWw1NOggdFQFOLZ8+5CzCiz143w== + +"@rollup/rollup-win32-arm64-msvc@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.37.0.tgz#8063d5f8195dd1845e056d069366fbe06a424d09" + integrity sha512-Jm7biMazjNzTU4PrQtr7VS8ibeys9Pn29/1bm4ph7CP2kf21950LgN+BaE2mJ1QujnvOc6p54eWWiVvn05SOBg== + +"@rollup/rollup-win32-ia32-msvc@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.37.0.tgz#891d90e3b5517f9d290bb416afdfe2ebfb12139e" + integrity sha512-e3/1SFm1OjefWICB2Ucstg2dxYDkDTZGDYgwufcbsxTHyqQps1UQf33dFEChBNmeSsTOyrjw2JJq0zbG5GF6RA== + +"@rollup/rollup-win32-x64-msvc@4.37.0": + version "4.37.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.37.0.tgz#a54d7304c3bd45573d8bcd1270de89771f8195fe" + integrity sha512-LWbXUBwn/bcLx2sSsqy7pK5o+Nr+VCoRoAohfJ5C/aBio9nfJmGQqHAhU6pwxV/RmyTk5AqdySma7uwWGlmeuA== + "@rushstack/eslint-patch@^1.1.0": version "1.6.1" resolved "https://registry.yarnpkg.com/@rushstack/eslint-patch/-/eslint-patch-1.6.1.tgz#9ab8f811930d7af3e3d549183a50884f9eb83f36" @@ -1895,6 +2471,11 @@ dependencies: "@types/trusted-types" "*" +"@types/estree@1.0.6": + version "1.0.6" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.6.tgz#628effeeae2064a1b4e79f78e81d87b7e5fc7b50" + integrity sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw== + "@types/estree@^1.0.0": version "1.0.5" resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" @@ -1995,6 +2576,13 @@ dependencies: undici-types "~5.26.4" +"@types/node@>=12.12.47", "@types/node@>=13.7.0": + version "22.10.10" + resolved "https://registry.yarnpkg.com/@types/node/-/node-22.10.10.tgz#85fe89f8bf459dc57dfef1689bd5b52ad1af07e6" + integrity sha512-X47y/mPNzxviAGY5TcYPtYL8JsY3kAq2n8fMmKoRCxq/c4v4pyGNCzM2R6+M5/umG4ZfHuT+sgqDYqWc9rJ6ww== + dependencies: + undici-types "~6.20.0" + "@types/parse-json@^4.0.0": version "4.0.2" resolved "https://registry.yarnpkg.com/@types/parse-json/-/parse-json-4.0.2.tgz#5950e50960793055845e956c427fc2b0d70c5239" @@ -2064,7 +2652,7 @@ dependencies: "@types/jest" "*" -"@types/trusted-types@*": +"@types/trusted-types@*", "@types/trusted-types@^2.0.7": version "2.0.7" resolved "https://registry.yarnpkg.com/@types/trusted-types/-/trusted-types-2.0.7.tgz#baccb07a970b91707df3a3e8ba6896c57ead2d11" integrity sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw== @@ -2213,7 +2801,7 @@ acorn@^8.8.2: resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.11.2.tgz#ca0d78b51895be5390a5903c5b3bdcdaf78ae40b" integrity sha512-nc0Axzp/0FILLEVsm4fNwLCwMttvhEI263QtVPQcbpfZZ3ts0hLsZGOpE6czNlid7CJ9MlyH8reXkpsf3YUY4w== -ansi-regex@^5.0.0: +ansi-regex@^5.0.0, ansi-regex@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== @@ -2358,6 +2946,11 @@ asynciterator.prototype@^1.0.0: dependencies: has-symbols "^1.0.3" +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q== + available-typed-arrays@^1.0.5: version "1.0.5" resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz#92f95616501069d07d10edb2fc37d3e1c65123b7" @@ -2376,12 +2969,14 @@ axios-cache-adapter@^2.7.3: cache-control-esm "1.0.0" md5 "^2.2.1" -axios@^0.21.2: - version "0.21.4" - resolved "https://registry.yarnpkg.com/axios/-/axios-0.21.4.tgz#c67b90dc0568e5c1cf2b0b858c43ba28e2eda575" - integrity sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg== +axios@^1.8.4: + version "1.8.4" + resolved "https://registry.yarnpkg.com/axios/-/axios-1.8.4.tgz#78990bb4bc63d2cae072952d374835950a82f447" + integrity sha512-eBSYY4Y68NNlHbHBMdeDmKNtDgXWhQsJcGqzO3iLUM0GraQFSS9cVgPX5I9b3lbdFKyYoAEGAZF1DwhTaljNAw== dependencies: - follow-redirects "^1.14.0" + follow-redirects "^1.15.6" + form-data "^4.0.0" + proxy-from-env "^1.1.0" axobject-query@^3.2.1: version "3.2.1" @@ -2537,6 +3132,14 @@ cache-control-esm@1.0.0: resolved "https://registry.yarnpkg.com/cache-control-esm/-/cache-control-esm-1.0.0.tgz#417647ecf1837a5e74155f55d5a4ae32a84e2581" integrity sha512-Fa3UV4+eIk4EOih8FTV6EEsVKO0W5XWtNs6FC3InTfVz+EjurjPfDXY5wZDo/lxjDxg5RjNcurLyxEJBcEUx9g== +call-bind-apply-helpers@^1.0.1, call-bind-apply-helpers@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz#4b5428c222be985d79c3d82657479dbe0b59b2d6" + integrity sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ== + dependencies: + es-errors "^1.3.0" + function-bind "^1.1.2" + call-bind@^1.0.0, call-bind@^1.0.2, call-bind@^1.0.4, call-bind@^1.0.5: version "1.0.5" resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.5.tgz#6fa2b7845ce0ea49bf4d8b9ef64727a2c2e2e513" @@ -2557,9 +3160,9 @@ camelcase@^6.2.0: integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== caniuse-lite@^1.0.30001565: - version "1.0.30001570" - resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001570.tgz#b4e5c1fa786f733ab78fc70f592df6b3f23244ca" - integrity sha512-+3e0ASu4sw1SWaoCtvPeyXp+5PsjigkSt8OXZbF9StH5pQWbxEjLAZE3n8Aup5udop1uRiKA7a4utUk/uoSpUw== + version "1.0.30001707" + resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001707.tgz" + integrity sha512-3qtRjw/HQSMlDWf+X79N206fepf4SOOU6SQLMaq/0KkZLmSjPxAkBOQQ+FxbHKfHmYLZFfdWsO3KA90ceHPSnw== chalk@^2.4.2: version "2.4.2" @@ -2606,6 +3209,15 @@ classnames@^2.2.5, classnames@^2.3.2: resolved "https://registry.yarnpkg.com/classnames/-/classnames-2.3.2.tgz#351d813bf0137fcc6a76a16b88208d2560a0d924" integrity sha512-CSbhY4cFEJRe6/GQzIk5qXZ4Jeg5pcsP7b5peFSDpffpe1cqjASH/n9UTjBwOp6XpMSTwQ8Za2K5V02ueA7Tmw== +cliui@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-8.0.1.tgz#0c04b075db02cbfe60dc8e6cf2f5486b1a3608aa" + integrity sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ== + dependencies: + string-width "^4.2.0" + strip-ansi "^6.0.1" + wrap-ansi "^7.0.0" + clsx@^1.1.1: version "1.2.1" resolved "https://registry.yarnpkg.com/clsx/-/clsx-1.2.1.tgz#0ddc4a20a549b59c93a4116bb26f5294ca17dc12" @@ -2635,6 +3247,13 @@ color-name@~1.1.4: resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== +combined-stream@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" + integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== + dependencies: + delayed-stream "~1.0.0" + comma-separated-tokens@^2.0.0: version "2.0.3" resolved "https://registry.yarnpkg.com/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz#4e89c9458acb61bc8fef19f4529973b2392839ee" @@ -2719,6 +3338,11 @@ crypt@0.0.2: resolved "https://registry.yarnpkg.com/crypt/-/crypt-0.0.2.tgz#88d7ff7ec0dfb86f713dc87bbb42d044d3e6c41b" integrity sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow== +css-mediaquery@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/css-mediaquery/-/css-mediaquery-0.1.2.tgz#6a2c37344928618631c54bd33cedd301da18bea0" + integrity sha512-COtn4EROW5dBGlE/4PiKnh6rZpAPxDeFLaEEwt4i10jpDMFt2EhQGS79QmmrO+iKCHv0PU/HrOWEhijFd1x99Q== + css.escape@^1.5.1: version "1.5.1" resolved "https://registry.yarnpkg.com/css.escape/-/css.escape-1.5.1.tgz#42e27d4fa04ae32f931a4b4d4191fa9cddee97cb" @@ -2790,6 +3414,11 @@ define-properties@^1.1.3, define-properties@^1.2.0, define-properties@^1.2.1: has-property-descriptors "^1.0.0" object-keys "^1.1.1" +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ== + dequal@^2.0.0, dequal@^2.0.3: version "2.0.3" resolved "https://registry.yarnpkg.com/dequal/-/dequal-2.0.3.tgz#2644214f1997d39ed0ee0ece72335490a7ac67be" @@ -2853,10 +3482,12 @@ domhandler@^5.0.2, domhandler@^5.0.3: dependencies: domelementtype "^2.3.0" -dompurify@^2.2.7: - version "2.4.7" - resolved "https://registry.yarnpkg.com/dompurify/-/dompurify-2.4.7.tgz#277adeb40a2c84be2d42a8bcd45f582bfa4d0cfc" - integrity sha512-kxxKlPEDa6Nc5WJi+qRgPbOAbgTpSULL+vI3NUXsZMlkJxTqYI9wg5ZTay2sFrdZRWHPWNi+EdAhcJf81WtoMQ== +dompurify@^3.2.4: + version "3.2.4" + resolved "https://registry.yarnpkg.com/dompurify/-/dompurify-3.2.4.tgz#af5a5a11407524431456cf18836c55d13441cd8e" + integrity sha512-ysFSFEDVduQpyhzAob/kkuJjf5zWkZD8/A9ywSp1byueyuCfHamrCBa14/Oc2iiB0e51B+NpxSl5gmzn+Ms/mg== + optionalDependencies: + "@types/trusted-types" "^2.0.7" domutils@^3.0.1: version "3.1.0" @@ -2875,6 +3506,15 @@ dot-case@^3.0.4: no-case "^3.0.4" tslib "^2.0.3" +dunder-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/dunder-proto/-/dunder-proto-1.0.1.tgz#d7ae667e1dc83482f8b70fd0f6eefc50da30f58a" + integrity sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A== + dependencies: + call-bind-apply-helpers "^1.0.1" + es-errors "^1.3.0" + gopd "^1.2.0" + ejs@^3.1.9: version "3.1.9" resolved "https://registry.yarnpkg.com/ejs/-/ejs-3.1.9.tgz#03c9e8777fe12686a9effcef22303ca3d8eeb361" @@ -2887,6 +3527,11 @@ electron-to-chromium@^1.4.601: resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.614.tgz#2fe789d61fa09cb875569f37c309d0c2701f91c0" integrity sha512-X4ze/9Sc3QWs6h92yerwqv7aB/uU8vCjZcrMjA8N9R1pjMFRe44dLsck5FzLilOYvcXuDn93B+bpGYyufc70gQ== +emoji-regex@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" + integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== + emoji-regex@^9.2.2: version "9.2.2" resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" @@ -2949,6 +3594,16 @@ es-abstract@^1.22.1: unbox-primitive "^1.0.2" which-typed-array "^1.1.13" +es-define-property@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/es-define-property/-/es-define-property-1.0.1.tgz#983eb2f9a6724e9303f61addf011c72e09e0b0fa" + integrity sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g== + +es-errors@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/es-errors/-/es-errors-1.3.0.tgz#05f75a25dab98e4fb1dcd5e1472c0546d5057c8f" + integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw== + es-iterator-helpers@^1.0.12, es-iterator-helpers@^1.0.15: version "1.0.15" resolved "https://registry.yarnpkg.com/es-iterator-helpers/-/es-iterator-helpers-1.0.15.tgz#bd81d275ac766431d19305923707c3efd9f1ae40" @@ -2969,6 +3624,13 @@ es-iterator-helpers@^1.0.12, es-iterator-helpers@^1.0.15: iterator.prototype "^1.1.2" safe-array-concat "^1.0.1" +es-object-atoms@^1.0.0, es-object-atoms@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/es-object-atoms/-/es-object-atoms-1.1.1.tgz#1c4f2c4837327597ce69d2ca190a7fdd172338c1" + integrity sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA== + dependencies: + es-errors "^1.3.0" + es-set-tostringtag@^2.0.1: version "2.0.2" resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.0.2.tgz#11f7cc9f63376930a5f20be4915834f4bc74f9c9" @@ -2978,6 +3640,16 @@ es-set-tostringtag@^2.0.1: has-tostringtag "^1.0.0" hasown "^2.0.0" +es-set-tostringtag@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz#f31dbbe0c183b00a6d26eb6325c810c0fd18bd4d" + integrity sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA== + dependencies: + es-errors "^1.3.0" + get-intrinsic "^1.2.6" + has-tostringtag "^1.0.2" + hasown "^2.0.2" + es-shim-unscopables@^1.0.0: version "1.0.2" resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.0.2.tgz#1f6942e71ecc7835ed1c8a83006d8771a63a3763" @@ -2994,33 +3666,36 @@ es-to-primitive@^1.2.1: is-date-object "^1.0.1" is-symbol "^1.0.2" -esbuild@^0.18.10: - version "0.18.20" - resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.18.20.tgz#4709f5a34801b43b799ab7d6d82f7284a9b7a7a6" - integrity sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA== +esbuild@^0.25.0: + version "0.25.1" + resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.25.1.tgz#a16b8d070b6ad4871935277bda6ccfe852e3fa2f" + integrity sha512-BGO5LtrGC7vxnqucAe/rmvKdJllfGaYWdyABvyMoXQlfYMb2bbRuReWR5tEGE//4LcNJj9XrkovTqNYRFZHAMQ== optionalDependencies: - "@esbuild/android-arm" "0.18.20" - "@esbuild/android-arm64" "0.18.20" - "@esbuild/android-x64" "0.18.20" - "@esbuild/darwin-arm64" "0.18.20" - "@esbuild/darwin-x64" "0.18.20" - "@esbuild/freebsd-arm64" "0.18.20" - "@esbuild/freebsd-x64" "0.18.20" - "@esbuild/linux-arm" "0.18.20" - "@esbuild/linux-arm64" "0.18.20" - "@esbuild/linux-ia32" "0.18.20" - "@esbuild/linux-loong64" "0.18.20" - "@esbuild/linux-mips64el" "0.18.20" - "@esbuild/linux-ppc64" "0.18.20" - "@esbuild/linux-riscv64" "0.18.20" - "@esbuild/linux-s390x" "0.18.20" - "@esbuild/linux-x64" "0.18.20" - "@esbuild/netbsd-x64" "0.18.20" - "@esbuild/openbsd-x64" "0.18.20" - "@esbuild/sunos-x64" "0.18.20" - "@esbuild/win32-arm64" "0.18.20" - "@esbuild/win32-ia32" "0.18.20" - "@esbuild/win32-x64" "0.18.20" + "@esbuild/aix-ppc64" "0.25.1" + "@esbuild/android-arm" "0.25.1" + "@esbuild/android-arm64" "0.25.1" + "@esbuild/android-x64" "0.25.1" + "@esbuild/darwin-arm64" "0.25.1" + "@esbuild/darwin-x64" "0.25.1" + "@esbuild/freebsd-arm64" "0.25.1" + "@esbuild/freebsd-x64" "0.25.1" + "@esbuild/linux-arm" "0.25.1" + "@esbuild/linux-arm64" "0.25.1" + "@esbuild/linux-ia32" "0.25.1" + "@esbuild/linux-loong64" "0.25.1" + "@esbuild/linux-mips64el" "0.25.1" + "@esbuild/linux-ppc64" "0.25.1" + "@esbuild/linux-riscv64" "0.25.1" + "@esbuild/linux-s390x" "0.25.1" + "@esbuild/linux-x64" "0.25.1" + "@esbuild/netbsd-arm64" "0.25.1" + "@esbuild/netbsd-x64" "0.25.1" + "@esbuild/openbsd-arm64" "0.25.1" + "@esbuild/openbsd-x64" "0.25.1" + "@esbuild/sunos-x64" "0.25.1" + "@esbuild/win32-arm64" "0.25.1" + "@esbuild/win32-ia32" "0.25.1" + "@esbuild/win32-x64" "0.25.1" escalade@^3.1.1: version "3.1.1" @@ -3338,6 +4013,13 @@ fastq@^1.6.0: dependencies: reusify "^1.0.4" +faye-websocket@0.11.4: + version "0.11.4" + resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.4.tgz#7f0d9275cfdd86a1c963dc8b65fcc451edcbb1da" + integrity sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g== + dependencies: + websocket-driver ">=0.5.1" + filelist@^1.0.4: version "1.0.4" resolved "https://registry.yarnpkg.com/filelist/-/filelist-1.0.4.tgz#f78978a1e944775ff9e62e744424f215e58352b5" @@ -3357,6 +4039,40 @@ find-root@^1.1.0: resolved "https://registry.yarnpkg.com/find-root/-/find-root-1.1.0.tgz#abcfc8ba76f708c42a97b3d685b7e9450bfb9ce4" integrity sha512-NKfW6bec6GfKc0SGx1e07QZY9PE99u0Bft/0rzSD5k3sO/vwkVUpDUKVm5Gpp5Ue3YfShPFTX2070tDs5kB9Ng== +firebase@^11.2.0: + version "11.2.0" + resolved "https://registry.yarnpkg.com/firebase/-/firebase-11.2.0.tgz#843de48382fcaf8050a4a278f35d094fb9960fda" + integrity sha512-ztwPhBLAZMVNZjBeQzzTM4rk2rsRXmdFYcnvjAXh+StbiFVshHKaPO9VRGMUzF48du4Mkz6jN1wkmYCuUJPxLA== + dependencies: + "@firebase/analytics" "0.10.11" + "@firebase/analytics-compat" "0.2.17" + "@firebase/app" "0.10.18" + "@firebase/app-check" "0.8.11" + "@firebase/app-check-compat" "0.3.18" + "@firebase/app-compat" "0.2.48" + "@firebase/app-types" "0.9.3" + "@firebase/auth" "1.8.2" + "@firebase/auth-compat" "0.5.17" + "@firebase/data-connect" "0.2.0" + "@firebase/database" "1.0.11" + "@firebase/database-compat" "2.0.2" + "@firebase/firestore" "4.7.6" + "@firebase/firestore-compat" "0.3.41" + "@firebase/functions" "0.12.1" + "@firebase/functions-compat" "0.3.18" + "@firebase/installations" "0.6.12" + "@firebase/installations-compat" "0.2.12" + "@firebase/messaging" "0.12.16" + "@firebase/messaging-compat" "0.2.16" + "@firebase/performance" "0.6.12" + "@firebase/performance-compat" "0.2.12" + "@firebase/remote-config" "0.5.0" + "@firebase/remote-config-compat" "0.2.12" + "@firebase/storage" "0.13.5" + "@firebase/storage-compat" "0.3.15" + "@firebase/util" "1.10.3" + "@firebase/vertexai" "1.0.3" + focus-trap@^6.2.2: version "6.9.4" resolved "https://registry.yarnpkg.com/focus-trap/-/focus-trap-6.9.4.tgz#436da1a1d935c48b97da63cd8f361c6f3aa16444" @@ -3364,10 +4080,10 @@ focus-trap@^6.2.2: dependencies: tabbable "^5.3.3" -follow-redirects@^1.14.0: - version "1.15.3" - resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.3.tgz#fe2f3ef2690afce7e82ed0b44db08165b207123a" - integrity sha512-1VzOtuEM8pC9SFU1E+8KfTjZyMztRsgEfwQl44z8A25uy13jSzTj6dyK2Df52iV0vgHCfBwLhDWevLn95w5v6Q== +follow-redirects@^1.15.6: + version "1.15.9" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.9.tgz#a604fa10e443bf98ca94228d9eebcc2e8a2c8ee1" + integrity sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ== for-each@^0.3.3: version "0.3.3" @@ -3376,7 +4092,17 @@ for-each@^0.3.3: dependencies: is-callable "^1.1.3" -fsevents@~2.3.2: +form-data@^4.0.0: + version "4.0.2" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.2.tgz#35cabbdd30c3ce73deb2c42d3c8d3ed9ca51794c" + integrity sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w== + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.8" + es-set-tostringtag "^2.1.0" + mime-types "^2.1.12" + +fsevents@~2.3.2, fsevents@~2.3.3: version "2.3.3" resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== @@ -3406,6 +4132,11 @@ gensync@^1.0.0-beta.2: resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.2.tgz#32a6ee76c3d7f52d46b2b1ae5d93fea8580a25e0" integrity sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg== +get-caller-file@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" + integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== + get-intrinsic@^1.0.2, get-intrinsic@^1.1.1, get-intrinsic@^1.1.3, get-intrinsic@^1.2.0, get-intrinsic@^1.2.1, get-intrinsic@^1.2.2: version "1.2.2" resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.2.tgz#281b7622971123e1ef4b3c90fd7539306da93f3b" @@ -3416,6 +4147,30 @@ get-intrinsic@^1.0.2, get-intrinsic@^1.1.1, get-intrinsic@^1.1.3, get-intrinsic@ has-symbols "^1.0.3" hasown "^2.0.0" +get-intrinsic@^1.2.6: + version "1.3.0" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.3.0.tgz#743f0e3b6964a93a5491ed1bffaae054d7f98d01" + integrity sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ== + 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" + +get-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/get-proto/-/get-proto-1.0.1.tgz#150b3f2743869ef3e851ec0c49d15b1d14d00ee1" + integrity sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g== + dependencies: + dunder-proto "^1.0.1" + es-object-atoms "^1.0.0" + get-symbol-description@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.0.tgz#7fdb81c900101fbd564dd5f1a30af5aadc1e58d6" @@ -3467,6 +4222,11 @@ gopd@^1.0.1: dependencies: get-intrinsic "^1.1.3" +gopd@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.2.0.tgz#89f56b8217bdbc8802bd299df6d7f1081d7e51a1" + integrity sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg== + graceful-fs@^4.2.9: version "4.2.11" resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" @@ -3509,6 +4269,11 @@ has-symbols@^1.0.2, has-symbols@^1.0.3: resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== +has-symbols@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.1.0.tgz#fc9c6a783a084951d0b971fe1018de813707a338" + integrity sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ== + has-tostringtag@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.0.tgz#7e133818a7d394734f941e73c3d3f9291e658b25" @@ -3516,6 +4281,13 @@ has-tostringtag@^1.0.0: dependencies: has-symbols "^1.0.2" +has-tostringtag@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.2.tgz#2cdc42d40bef2e5b4eeab7c01a73c54ce7ab5abc" + integrity sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw== + dependencies: + has-symbols "^1.0.3" + hasown@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.0.tgz#f4c513d454a57b7c7e1650778de226b11700546c" @@ -3523,6 +4295,13 @@ hasown@^2.0.0: dependencies: function-bind "^1.1.2" +hasown@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003" + integrity sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ== + dependencies: + function-bind "^1.1.2" + hast-util-whitespace@^2.0.0: version "2.0.1" resolved "https://registry.yarnpkg.com/hast-util-whitespace/-/hast-util-whitespace-2.0.1.tgz#0ec64e257e6fc216c7d14c8a1b74d27d650b4557" @@ -3545,6 +4324,20 @@ htmlparser2@^8.0.1: domutils "^3.0.1" entities "^4.4.0" +hyphenate-style-name@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/hyphenate-style-name/-/hyphenate-style-name-1.1.0.tgz#1797bf50369588b47b72ca6d5e65374607cf4436" + integrity sha512-WDC/ui2VVRrz3jOVi+XtjqkDjiVjTtFaAGiW37k6b+ohyQ5wYDOGkvCZa8+H0nx3gyvv0+BST9xuOgIyGQ00gw== +http-parser-js@>=0.5.1: + version "0.5.9" + resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.5.9.tgz#b817b3ca0edea6236225000d795378707c169cec" + integrity sha512-n1XsPy3rXVxlqxVioEWdC+0+M+SQw0DpJynwtOPo1X+ZlvdzTLtDBIJJlDQTnwZIFJrZSzSGmIOUdP8tu+SgLw== + +idb@7.1.1: + version "7.1.1" + resolved "https://registry.yarnpkg.com/idb/-/idb-7.1.1.tgz#d910ded866d32c7ced9befc5bfdf36f572ced72b" + integrity sha512-gchesWBzyvGHRO9W8tzUWFDycow5gwjvFKfyV9FF32Y7F50yZMp7mP+T2mJIWFx49zicqyC4uefHM17o6xKIVQ== + ignore@^5.2.0: version "5.3.0" resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.0.tgz#67418ae40d34d6999c95ff56016759c718c82f78" @@ -3654,6 +4447,11 @@ is-finalizationregistry@^1.0.2: dependencies: call-bind "^1.0.2" +is-fullwidth-code-point@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" + integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== + is-generator-function@^1.0.10: version "1.0.10" resolved "https://registry.yarnpkg.com/is-generator-function/-/is-generator-function-1.0.10.tgz#f1558baf1ac17e0deea7c0415c438351ff2b3c72" @@ -3929,6 +4727,11 @@ lines-and-columns@^1.1.6: resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz#eca284f75d2965079309dc0ad9255abb2ebc1632" integrity sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg== +lodash.camelcase@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6" + integrity sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA== + lodash.debounce@^4.0.8: version "4.0.8" resolved "https://registry.yarnpkg.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz#82d79bff30a67c4005ffd5e2515300ad9ca4d7af" @@ -3939,6 +4742,11 @@ lodash@^4.17.15, lodash@^4.17.21: resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== +long@^5.0.0: + version "5.2.4" + resolved "https://registry.yarnpkg.com/long/-/long-5.2.4.tgz#ee651d5c7c25901cfca5e67220ae9911695e99b2" + integrity sha512-qtzLbJE8hq7VabR3mISmVGtoXP8KGc2Z/AT8OuqlYD7JTR3oqrgwdjnk07wpj1twXxYmgDXgoKVWUG/fReSzHg== + loose-envify@^1.0.0, loose-envify@^1.1.0, loose-envify@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf" @@ -3972,6 +4780,17 @@ lz-string@^1.4.4: resolved "https://registry.yarnpkg.com/lz-string/-/lz-string-1.5.0.tgz#c1ab50f77887b712621201ba9fd4e3a6ed099941" integrity sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ== +matchmediaquery@^0.4.2: + version "0.4.2" + resolved "https://registry.yarnpkg.com/matchmediaquery/-/matchmediaquery-0.4.2.tgz#22582bd4ae63ad9f54c53001bba80cbed0f7eafa" + integrity sha512-wrZpoT50ehYOudhDjt/YvUJc6eUzcdFPdmbizfgvswCKNHD1/OBOHYJpHie+HXpu6bSkEGieFMYk6VuutaiRfA== + dependencies: + css-mediaquery "^0.1.2" +math-intrinsics@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/math-intrinsics/-/math-intrinsics-1.1.0.tgz#a0dd74be81e2aa5c2f27e65ce283605ee4e2b7f9" + integrity sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g== + md5@^2.2.1: version "2.3.0" resolved "https://registry.yarnpkg.com/md5/-/md5-2.3.0.tgz#c3da9a6aae3a30b46b7b0c349b87b110dc3bda4f" @@ -4247,6 +5066,18 @@ micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" +mime-db@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.12: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + min-indent@^1.0.0: version "1.0.1" resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869" @@ -4291,10 +5122,10 @@ ms@^2.1.1: resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== -nanoid@^3.3.7: - version "3.3.7" - resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8" - integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g== +nanoid@^3.3.8: + version "3.3.11" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.11.tgz#4f4f112cefbe303202f2199838128936266d185b" + integrity sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w== natural-compare-lite@^1.4.0: version "1.4.0" @@ -4433,19 +5264,24 @@ picocolors@^1.0.0: resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== +picocolors@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.1.1.tgz#3d321af3eab939b083c8f929a1d12cda81c26b6b" + integrity sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA== + picomatch@^2.2.3, picomatch@^2.3.1: version "2.3.1" resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== -postcss@^8.4.27: - version "8.4.32" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.32.tgz#1dac6ac51ab19adb21b8b34fd2d93a86440ef6c9" - integrity sha512-D/kj5JNu6oo2EIy+XL/26JEDTlIbB8hw85G8StOE6L74RQAVVP5rej6wxCNqyMbR4RkPfqvezVbPw81Ngd6Kcw== +postcss@^8.5.3: + version "8.5.3" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.5.3.tgz#1463b6f1c7fb16fe258736cba29a2de35237eafb" + integrity sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A== dependencies: - nanoid "^3.3.7" - picocolors "^1.0.0" - source-map-js "^1.0.2" + nanoid "^3.3.8" + picocolors "^1.1.1" + source-map-js "^1.2.1" prelude-ls@~1.1.2: version "1.1.2" @@ -4483,7 +5319,7 @@ pretty-format@^29.0.0, pretty-format@^29.7.0: ansi-styles "^5.0.0" react-is "^18.0.0" -prop-types@^15.0.0, prop-types@^15.0.0-0, prop-types@^15.5.8, prop-types@^15.6.0, prop-types@^15.6.2, prop-types@^15.7.2, prop-types@^15.8.1: +prop-types@^15.0.0, prop-types@^15.0.0-0, prop-types@^15.5.8, prop-types@^15.6.0, prop-types@^15.6.1, prop-types@^15.6.2, prop-types@^15.7.2, prop-types@^15.8.1: version "15.8.1" resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5" integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg== @@ -4497,6 +5333,29 @@ property-information@^6.0.0: resolved "https://registry.yarnpkg.com/property-information/-/property-information-6.4.0.tgz#6bc4c618b0c2d68b3bb8b552cbb97f8e300a0f82" integrity sha512-9t5qARVofg2xQqKtytzt+lZ4d1Qvj8t5B8fEwXK6qOfgRLgH/b13QlgEyDh033NOS31nXeFbYv7CLUDG1CeifQ== +protobufjs@^7.2.5: + version "7.4.0" + resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-7.4.0.tgz#7efe324ce9b3b61c82aae5de810d287bc08a248a" + integrity sha512-mRUWCc3KUU4w1jU8sGxICXH/gNS94DvI1gxqDvBzhj1JpcsimQkYiOJfwsPUykUI5ZaspFbSgmBLER8IrQ3tqw== + dependencies: + "@protobufjs/aspromise" "^1.1.2" + "@protobufjs/base64" "^1.1.2" + "@protobufjs/codegen" "^2.0.4" + "@protobufjs/eventemitter" "^1.1.0" + "@protobufjs/fetch" "^1.1.0" + "@protobufjs/float" "^1.0.2" + "@protobufjs/inquire" "^1.1.0" + "@protobufjs/path" "^1.1.2" + "@protobufjs/pool" "^1.1.0" + "@protobufjs/utf8" "^1.1.0" + "@types/node" ">=13.7.0" + long "^5.0.0" + +proxy-from-env@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2" + integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg== + queue-microtask@^1.2.2: version "1.2.3" resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" @@ -4509,13 +5368,6 @@ react-contexify@^5.0.0: dependencies: clsx "^1.1.1" -react-device-detect@^1.17.0: - version "1.17.0" - resolved "https://registry.yarnpkg.com/react-device-detect/-/react-device-detect-1.17.0.tgz#a00b4fd6880cebfab3fd8a42a79dc0290cdddca9" - integrity sha512-bBblIStwpHmoS281JFIVqeimcN3LhpoP5YKDWzxQdBIUP8S2xPvHDgizLDhUq2ScguLfVPmwfF5y268EEQR60w== - dependencies: - ua-parser-js "^0.7.24" - react-dom@^17.0.1: version "17.0.2" resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-17.0.2.tgz#ecffb6845e3ad8dbfcdc498f0d0a939736502c23" @@ -4605,6 +5457,16 @@ react-refresh@^0.14.0: resolved "https://registry.yarnpkg.com/react-refresh/-/react-refresh-0.14.0.tgz#4e02825378a5f227079554d4284889354e5f553e" integrity sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ== +react-responsive@^10.0.1: + version "10.0.1" + resolved "https://registry.yarnpkg.com/react-responsive/-/react-responsive-10.0.1.tgz#293d4d2562da93409861216f0110d146c5676eb3" + integrity sha512-OM5/cRvbtUWEX8le8RCT8scA8y2OPtb0Q/IViEyCEM5FBN8lRrkUOZnu87I88A6njxDldvxG+rLBxWiA7/UM9g== + dependencies: + hyphenate-style-name "^1.0.0" + matchmediaquery "^0.4.2" + prop-types "^15.6.1" + shallow-equal "^3.1.0" + react-router-dom@^6.21.0: version "6.21.0" resolved "https://registry.yarnpkg.com/react-router-dom/-/react-router-dom-6.21.0.tgz#aa4c6bc046a8e8723095bc09b3c0ab2254532712" @@ -4806,6 +5668,11 @@ remark-rehype@^9.0.0: mdast-util-to-hast "^11.0.0" unified "^10.0.0" +require-directory@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" + integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q== + resolve-from@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" @@ -4834,11 +5701,33 @@ reusify@^1.0.4: resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== -rollup@^3.27.1: - version "3.29.4" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-3.29.4.tgz#4d70c0f9834146df8705bfb69a9a19c9e1109981" - integrity sha512-oWzmBZwvYrU0iJHtDmhsm662rC15FRXmcjCk1xD771dFDx5jJ02ufAQQTn0etB2emNk4J9EZg/yWKpsn9BWGRw== +rollup@^4.30.1: + version "4.37.0" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.37.0.tgz#e4172f8bdb6ea7df08a1b0acf99abeccb2250378" + integrity sha512-iAtQy/L4QFU+rTJ1YUjXqJOJzuwEghqWzCEYD2FEghT7Gsy1VdABntrO4CLopA5IkflTyqNiLNwPcOJ3S7UKLg== + dependencies: + "@types/estree" "1.0.6" optionalDependencies: + "@rollup/rollup-android-arm-eabi" "4.37.0" + "@rollup/rollup-android-arm64" "4.37.0" + "@rollup/rollup-darwin-arm64" "4.37.0" + "@rollup/rollup-darwin-x64" "4.37.0" + "@rollup/rollup-freebsd-arm64" "4.37.0" + "@rollup/rollup-freebsd-x64" "4.37.0" + "@rollup/rollup-linux-arm-gnueabihf" "4.37.0" + "@rollup/rollup-linux-arm-musleabihf" "4.37.0" + "@rollup/rollup-linux-arm64-gnu" "4.37.0" + "@rollup/rollup-linux-arm64-musl" "4.37.0" + "@rollup/rollup-linux-loongarch64-gnu" "4.37.0" + "@rollup/rollup-linux-powerpc64le-gnu" "4.37.0" + "@rollup/rollup-linux-riscv64-gnu" "4.37.0" + "@rollup/rollup-linux-riscv64-musl" "4.37.0" + "@rollup/rollup-linux-s390x-gnu" "4.37.0" + "@rollup/rollup-linux-x64-gnu" "4.37.0" + "@rollup/rollup-linux-x64-musl" "4.37.0" + "@rollup/rollup-win32-arm64-msvc" "4.37.0" + "@rollup/rollup-win32-ia32-msvc" "4.37.0" + "@rollup/rollup-win32-x64-msvc" "4.37.0" fsevents "~2.3.2" run-parallel@^1.1.9: @@ -4865,6 +5754,11 @@ safe-array-concat@^1.0.1: has-symbols "^1.0.3" isarray "^2.0.5" +safe-buffer@>=5.1.0: + version "5.2.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" + integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== + safe-regex-test@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.0.0.tgz#793b874d524eb3640d1873aad03596db2d4f2295" @@ -4913,6 +5807,11 @@ set-function-name@^2.0.0, set-function-name@^2.0.1: functions-have-names "^1.2.3" has-property-descriptors "^1.0.0" +shallow-equal@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/shallow-equal/-/shallow-equal-3.1.0.tgz#e7a54bac629c7f248eff6c2f5b63122ba4320bec" + integrity sha512-pfVOw8QZIXpMbhBWvzBISicvToTiM5WBF1EeAUZDDSb5Dt29yl4AYbyywbJFSEsRUMr7gJaxqCdr4L3tQf9wVg== + side-channel@^1.0.4: version "1.0.4" resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.4.tgz#efce5c8fdc104ee751b25c58d4290011fa5ea2cf" @@ -4935,10 +5834,10 @@ snake-case@^3.0.4: dot-case "^3.0.4" tslib "^2.0.3" -source-map-js@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.0.2.tgz#adbc361d9c62df380125e7f161f71c826f1e490c" - integrity sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw== +source-map-js@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.2.1.tgz#1ce5650fddd87abc099eda37dcff024c2667ae46" + integrity sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA== source-map-support@~0.5.20: version "0.5.21" @@ -4982,6 +5881,15 @@ string-natural-compare@^3.0.1: resolved "https://registry.yarnpkg.com/string-natural-compare/-/string-natural-compare-3.0.1.tgz#7a42d58474454963759e8e8b7ae63d71c1e7fdf4" integrity sha512-n3sPwynL1nwKi3WJ6AIsClwBMa0zTi54fn2oLU6ndfTSIO05xaznjSf15PcBZU6FNWbmN5Q6cxT4V5hGvB4taw== +string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + string.prototype.matchall@^4.0.8: version "4.0.10" resolved "https://registry.yarnpkg.com/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz#a1553eb532221d4180c51581d6072cd65d1ee100" @@ -5024,6 +5932,13 @@ string.prototype.trimstart@^1.0.7: define-properties "^1.2.0" es-abstract "^1.22.1" +strip-ansi@^6.0.0, strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + strip-bom@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" @@ -5202,11 +6117,6 @@ typescript@^5.1.6: resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== -ua-parser-js@^0.7.24: - version "0.7.37" - resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.37.tgz#e464e66dac2d33a7a1251d7d7a99d6157ec27832" - integrity sha512-xV8kqRKM+jhMvcHWUKthV9fNebIzrNy//2O9ZwWcfiBFR5f25XVZPLlEajk/sf3Ra15V92isyQqnIEXRDaZWEA== - unbox-primitive@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz#29032021057d5e6cdbd08c5129c226dff8ed6f9e" @@ -5227,6 +6137,11 @@ undici-types@~5.26.4: resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== +undici-types@~6.20.0: + version "6.20.0" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.20.0.tgz#8171bf22c1f588d1554d55bf204bc624af388433" + integrity sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg== + unicode-canonical-property-names-ecmascript@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz#301acdc525631670d39f6146e0e77ff6bbdebddc" @@ -5396,16 +6311,16 @@ vite-tsconfig-paths@^4.2.0: globrex "^0.1.2" tsconfck "^2.1.0" -vite@^4.4.5: - version "4.5.1" - resolved "https://registry.yarnpkg.com/vite/-/vite-4.5.1.tgz#3370986e1ed5dbabbf35a6c2e1fb1e18555b968a" - integrity sha512-AXXFaAJ8yebyqzoNB9fu2pHoo/nWX+xZlaRwoeYUxEqBO+Zj4msE5G+BhGBll9lYEKv9Hfks52PAF2X7qDYXQA== +vite@^6.2.3: + version "6.2.3" + resolved "https://registry.yarnpkg.com/vite/-/vite-6.2.3.tgz#249e92d32886981ab46bc1f049ac72abc6fa81e2" + integrity sha512-IzwM54g4y9JA/xAeBPNaDXiBF8Jsgl3VBQ2YQ/wOY6fyW3xMdSoltIV3Bo59DErdqdE6RxUfv8W69DvUorE4Eg== dependencies: - esbuild "^0.18.10" - postcss "^8.4.27" - rollup "^3.27.1" + esbuild "^0.25.0" + postcss "^8.5.3" + rollup "^4.30.1" optionalDependencies: - fsevents "~2.3.2" + fsevents "~2.3.3" warning@^4.0.3: version "4.0.3" @@ -5419,6 +6334,20 @@ web-vitals@^0.2.4: resolved "https://registry.yarnpkg.com/web-vitals/-/web-vitals-0.2.4.tgz#ec3df43c834a207fd7cdefd732b2987896e08511" integrity sha512-6BjspCO9VriYy12z356nL6JBS0GYeEcA457YyRzD+dD6XYCQ75NKhcOHUMHentOE7OcVCIXXDvOm0jKFfQG2Gg== +websocket-driver@>=0.5.1: + version "0.7.4" + resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.7.4.tgz#89ad5295bbf64b480abcba31e4953aca706f5760" + integrity sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg== + dependencies: + http-parser-js ">=0.5.1" + safe-buffer ">=5.1.0" + websocket-extensions ">=0.1.1" + +websocket-extensions@>=0.1.1: + version "0.1.4" + resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.4.tgz#7f8473bc839dfd87608adb95d7eb075211578a42" + integrity sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg== + which-boxed-primitive@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" @@ -5474,11 +6403,25 @@ word-wrap@~1.2.3: resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34" integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA== +wrap-ansi@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + xstate@^4.15.1: version "4.38.3" resolved "https://registry.yarnpkg.com/xstate/-/xstate-4.38.3.tgz#4e15e7ad3aa0ca1eea2010548a5379966d8f1075" integrity sha512-SH7nAaaPQx57dx6qvfcIgqKRXIh4L0A1iYEqim4s1u7c9VoCgzZc+63FY90AKU4ZzOC2cfJzTnpO4zK7fCUzzw== +y18n@^5.0.5: + version "5.0.8" + resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" + integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== + yallist@^3.0.2: version "3.1.1" resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" @@ -5494,6 +6437,24 @@ yaml@^1.10.0, yaml@^1.7.2: resolved "https://registry.yarnpkg.com/yaml/-/yaml-1.10.2.tgz#2301c5ffbf12b467de8da2333a459e29e7920e4b" integrity sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg== +yargs-parser@^21.1.1: + version "21.1.1" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35" + integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw== + +yargs@^17.7.2: + version "17.7.2" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" + integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== + dependencies: + cliui "^8.0.1" + escalade "^3.1.1" + get-caller-file "^2.0.5" + require-directory "^2.1.1" + string-width "^4.2.3" + y18n "^5.0.5" + yargs-parser "^21.1.1" + zustand@^4.3.3: version "4.4.7" resolved "https://registry.yarnpkg.com/zustand/-/zustand-4.4.7.tgz#355406be6b11ab335f59a66d2cf9815e8f24038c"