diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 00000000..72e9aa42 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,7 @@ +Dockerfile +.dockerignore +node_modules +npm-debug.log +README.md +.next +.git \ No newline at end of file diff --git a/.env.development b/.env.development new file mode 100644 index 00000000..1335bf8c --- /dev/null +++ b/.env.development @@ -0,0 +1,4 @@ +NEXT_PUBLIC_API_URL=https://dev.say-cheese.me +NEXT_PUBLIC_KAKAO_AUTH_URL=https://dev.say-cheese.me/oauth2/authorization/kakao +NEXT_PUBLIC_CLIENT_URL=http://localhost:3000 +NEXT_PUBLIC_KAKAO_JS_KEY=06fe9be27d60a133d4b9d25b9628f274 diff --git a/.env.production b/.env.production new file mode 100644 index 00000000..bc107e0c --- /dev/null +++ b/.env.production @@ -0,0 +1,5 @@ +NEXT_PUBLIC_API_URL=https://dev.say-cheese.me +# NEXT_PUBLIC_API_URL=https://api.say-cheese.me +NEXT_PUBLIC_KAKAO_AUTH_URL=https://dev.say-cheese.me/oauth2/authorization/kakao +NEXT_PUBLIC_CLIENT_URL=https://say-cheese.me +NEXT_PUBLIC_KAKAO_JS_KEY=06fe9be27d60a133d4b9d25b9628f274 diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 00000000..a131ed24 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,20 @@ +--- +name: Frontend Issue +about: 프론트엔드 관련 이슈 및 개발 요청 (UI/UX, 컴포넌트, 페이지 등) +title: '' +labels: '' +assignees: '' +--- + +## 🎯 이슈 유형 + +- [ ] 🐛 버그 수정 +- [ ] ✨ 새 기능 개발 +- [ ] 🎨 UI/UX 개선 +- [ ] 📱 반응형 대응 +- [ ] ⚡ 성능 최적화 +- [ ] 🔧 리팩토링 + +## 📝 상세 설명 + + diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md new file mode 100644 index 00000000..7cb246ff --- /dev/null +++ b/.github/pull_request_template.md @@ -0,0 +1,36 @@ +## 요약 + + + +## 구현 사항 + + + +- [ ] +- [ ] +- [ ] +- [ ] + +## 📸 스크린샷 + + + +## Need Review + +- ~ 부분 이렇게 구현했어요, 피드백 부탁해요! + + +## Reference + + + +### 📜 리뷰 규칙 + +Reviewer는 아래 **P5 Rule**을 참고하여 리뷰를 진행합니다. +P5 Rule을 통해 Reviewer는 Reviewee에게 리뷰의 의도를 보다 정확히 전달할 수 있습니다. + +- P1: 꼭 반영해주세요 (Comment) +- P2: 적극적으로 고려해주세요 (Comment) +- P3: 웬만하면 반영해 주세요 (Comment) +- P4: 반영해도 좋고 넘어가도 좋습니다 (Approve) +- P5: 그냥 사소한 의견입니다 (Approve) diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml new file mode 100644 index 00000000..aac174f9 --- /dev/null +++ b/.github/workflows/deploy.yml @@ -0,0 +1,59 @@ +name: deploy + +# on: +# push: +# branches: [main] + +jobs: + app-build: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + + - uses: docker/setup-buildx-action@v3 + + - uses: docker/login-action@v3 + with: + username: ${{ secrets.DOCKER_USERNAME }} + password: ${{ secrets.DOCKER_TOKEN }} + + - name: Build & Push + uses: docker/build-push-action@v6 + with: + context: . + file: ./Dockerfile + push: true + platforms: linux/amd64 + tags: | + ${{ secrets.DOCKER_USERNAME }}/${{ secrets.DOCKER_IMAGENAME }}:${{ github.run_id }} + ${{ secrets.DOCKER_USERNAME }}/${{ secrets.DOCKER_IMAGENAME }}:latest + cache-from: type=registry,ref=${{ secrets.DOCKER_USERNAME }}/${{ secrets.DOCKER_IMAGENAME }}:buildcache + cache-to: type=registry,ref=${{ secrets.DOCKER_USERNAME }}/${{ secrets.DOCKER_IMAGENAME }}:buildcache,mode=max + + - name: Deploy via SSH + uses: appleboy/ssh-action@v1.2.0 + with: + host: ${{ secrets.SSH_HOST }} + username: ${{ secrets.SSH_USERNAME }} + password: ${{ secrets.SSH_PASSWORD }} + port: ${{ secrets.SSH_PORT }} + script: | + set -e + APP=${{ secrets.DOCKER_IMAGENAME }} + IMAGE=${{ secrets.DOCKER_USERNAME }}/${{ secrets.DOCKER_IMAGENAME }}:${{ github.run_id }} + + docker login -u ${{ secrets.DOCKER_USERNAME }} -p ${{ secrets.DOCKER_TOKEN }} + docker pull $IMAGE + + # 기존 컨테이너 종료/삭제 + docker stop ${APP} || true + docker rm ${APP} || true + + # 새 컨테이너(임시 이름) 먼저 띄우고 헬스체크 + docker run -d --name ${APP} \ + --restart=always \ + -p 3000:3000 \ + $IMAGE + + # 오래된 이미지 정리(선택) + docker image prune -f || true diff --git a/.github/workflows/lint-checker.yml b/.github/workflows/lint-checker.yml new file mode 100644 index 00000000..d7b81c06 --- /dev/null +++ b/.github/workflows/lint-checker.yml @@ -0,0 +1,26 @@ +name: Lint Checker + +on: + pull_request: + types: [opened, synchronize, reopened] + +jobs: + lint: + runs-on: ubuntu-latest + + steps: + - name: Checkout + uses: actions/checkout@v3 + + - name: Setup pnpm + uses: pnpm/action-setup@v4 + with: + run_install: true + + - name: Setup Node + uses: actions/setup-node@v4 + with: + node-version: 22 + + - name: Run lint + run: pnpm lint diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 00000000..c57681fe --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,29 @@ +name: CI - Test + +on: + pull_request: + +jobs: + test: + runs-on: ubuntu-latest + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Install pnpm + uses: pnpm/action-setup@v4 + with: + run_install: false + + - name: Setup Node.js + uses: actions/setup-node@v4 + with: + node-version: '22' + cache: 'pnpm' + + - name: Install dependencies + run: pnpm install --frozen-lockfile + + - name: Run tests + run: pnpm test diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..5340dd9a --- /dev/null +++ b/.gitignore @@ -0,0 +1,38 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.* +.yarn/* +!.yarn/patches +!.yarn/plugins +!.yarn/releases +!.yarn/versions + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* +.pnpm-debug.log* + +# vercel +.vercel + +# typescript +*.tsbuildinfo +next-env.d.ts diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 00000000..4b7ffd9c --- /dev/null +++ b/.prettierignore @@ -0,0 +1,4 @@ +*.yaml +*.css +**/token.json +src/global/api/ep.ts diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 00000000..91cc9e17 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,10 @@ +{ + "semi": true, + "singleQuote": true, + "jsxSingleQuote": true, + "trailingComma": "all", + "tabWidth": 2, + "printWidth": 80, + "endOfLine": "lf", + "plugins": ["prettier-plugin-tailwindcss"] +} diff --git a/.vscode/extensions.json b/.vscode/extensions.json new file mode 100644 index 00000000..0b1c1e74 --- /dev/null +++ b/.vscode/extensions.json @@ -0,0 +1,6 @@ +{ + "recommendations": [ + "esbenp.prettier-vscode", // Prettier - 코드 포매터 + "bradlc.vscode-tailwindcss" // Tailwind CSS IntelliSense + ] +} diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 00000000..201cb3f8 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,11 @@ +{ + "files.eol": "\n", + "files.insertFinalNewline": true, + "files.trimTrailingWhitespace": true, + "prettier.endOfLine": "lf", + "editor.formatOnSave": true, + "editor.codeActionsOnSave": { + "source.organizeImports": "explicit", + "source.fixAll.eslint": "explicit" + } +} diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 00000000..02e5f051 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,66 @@ +# syntax=docker.io/docker/dockerfile:1 + +FROM node:22-alpine AS base + +# Install dependencies only when needed +FROM base AS deps +# Check https://github.com/nodejs/docker-node/tree/b4117f9333da4138b03a546ec926ef50a31506c3#nodealpine to understand why libc6-compat might be needed. +RUN apk add --no-cache libc6-compat +WORKDIR /app + +# Install dependencies based on the preferred package manager +COPY package.json yarn.lock* package-lock.json* pnpm-lock.yaml* .npmrc* ./ +RUN \ + if [ -f yarn.lock ]; then yarn --frozen-lockfile; \ + elif [ -f package-lock.json ]; then npm ci; \ + elif [ -f pnpm-lock.yaml ]; then corepack enable pnpm && pnpm i --frozen-lockfile; \ + else echo "Lockfile not found." && exit 1; \ + fi + + +# Rebuild the source code only when needed +FROM base AS builder +WORKDIR /app +COPY --from=deps /app/node_modules ./node_modules +COPY . . + +# Next.js collects completely anonymous telemetry data about general usage. +# Learn more here: https://nextjs.org/telemetry +# Uncomment the following line in case you want to disable telemetry during the build. +# ENV NEXT_TELEMETRY_DISABLED=1 + +RUN \ + if [ -f yarn.lock ]; then yarn run build; \ + elif [ -f package-lock.json ]; then npm run build; \ + elif [ -f pnpm-lock.yaml ]; then corepack enable pnpm && pnpm run build; \ + else echo "Lockfile not found." && exit 1; \ + fi + +# Production image, copy all the files and run next +FROM base AS runner +WORKDIR /app + +ENV NODE_ENV=production +# Uncomment the following line in case you want to disable telemetry during runtime. +# ENV NEXT_TELEMETRY_DISABLED=1 + +RUN addgroup --system --gid 1001 nodejs +RUN adduser --system --uid 1001 nextjs + +COPY --from=builder /app/public ./public + +# Automatically leverage output traces to reduce image size +# https://nextjs.org/docs/advanced-features/output-file-tracing +COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./ +COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static + +USER nextjs + +EXPOSE 3000 + +ENV PORT=3000 + +# server.js is created by next build from the standalone output +# https://nextjs.org/docs/pages/api-reference/config/next-config-js/output +ENV HOSTNAME="0.0.0.0" +CMD ["node", "server.js"] \ No newline at end of file diff --git a/README.md b/README.md index 99291317..e4d30f92 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,95 @@ -# FE -🧀 Kusitms 32nd Cheeeese Frontend Repository 🧀 +# 🧀 치이이즈 : 딱 7일만 열리는 특별한 공유 앨범 서비스 + +> 🔗 서비스 링크: [https://say-cheese.me](https://say-cheese.me) + +![웹 썸네일](https://github.com/user-attachments/assets/f5a6c97a-21b9-4dff-a7b7-8c12fe6e27db) + +## 🧑‍🤝‍🧑 Frontend Members + +| **김규태** | **김건우** | +| :-----------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------: | +| | | +| `frontend` | `frontend` | + +## 🏛️ System Architecture + +image + +## 🛠️ 기술 스택 + +- Language & Framework + - Next.js 15 + - React + - TypeScript + +- State & Data Management + - Zustand + - TanStack Query + +- UI Utilities + - Tailwind CSS + - shadcn/ui + - Framer Motion + - Lucide Icons + +- Testing + - Vitest + +- CI/CD + - GitHub Actions + +## 📂 폴더 구조 + +```csharp +src +├── app # 🌐 Next.js App Router (페이지 엔트리) +│ +├── components +│ └── ui # 🎨 공통 UI 컴포넌트 (shadcn 기반) +│ +├── feature # 📦 도메인 단위 기능 모듈 (components, hooks, utils, svg, constants 폴더로 구성) +│ ├── album # 📸 앨범 도메인 +│ │ ├── 4cut # 🎞️ 치즈네컷 생성 / 렌더링 +│ │ ├── detail # 🖼️ 앨범 상세 화면 +│ │ └── qrcode # 🔳 앨범 QR 코드 생성 +│ ├── album-entry # 🚪 앨범 입장 플로우 +│ ├── album-select # 📂 앨범 선택 화면 +│ ├── create-album # 📝 앨범 생성 +│ ├── login # 🔐 로그인 / 인증 플로우 +│ ├── main # 🏠 메인 화면 +│ ├── mypage # 🙋‍♂️ 마이페이지 +│ ├── onboarding # 🎉 온보딩 +│ ├── photo-detail # 🖼️ 사진 상세 +│ ├── photo-entry # 📥 사진 업로드 입장 +│ ├── root # 🌳 랜딩페이지 +│ ├── term # 📄 약관 +│ └── upload # ⬆️ 사진 업로드 +│ +├── global # 🌍 전역 설정 / 공통 모듈 +│ ├── api # 🔗 API 클라이언트 / 엔드포인트 +│ ├── components # 🧩 글로벌 공통 컴포넌트 +│ ├── constants # 🧱 상수 +│ ├── context # ⚛️ 컨텍스트 프로바이더 +│ ├── hooks # 🪝 공용 Hooks +│ ├── svg # 🖼️ SVG 에셋 +│ ├── types # 🧾 타입 정의 +│ └── utils # 🔧 공용 유틸리티 함수 +│ +└── store # 🗂️ Zustand 스토어 + +``` + +## 💬 Commit Convention + +### {이슈Type} : 커밋내용 + +ex. feat : 앨범나가기 버튼추가 + +| Type | 내용 | +| ---------- | ----------------------------------- | +| `feat` | 새로운 기능 구현 | +| `chore` | 부수적인 코드 수정 및 기타 변경사항 | +| `docs` | 문서 추가 및 수정, 삭제 | +| `fix` | 버그 수정 | +| `test` | 테스트 코드 추가 및 수정, 삭제 | +| `refactor` | 코드 리팩토링 | diff --git a/components.json b/components.json new file mode 100644 index 00000000..edcaef26 --- /dev/null +++ b/components.json @@ -0,0 +1,22 @@ +{ + "$schema": "https://ui.shadcn.com/schema.json", + "style": "new-york", + "rsc": true, + "tsx": true, + "tailwind": { + "config": "", + "css": "src/app/globals.css", + "baseColor": "neutral", + "cssVariables": true, + "prefix": "" + }, + "iconLibrary": "lucide", + "aliases": { + "components": "@/components", + "utils": "@/lib/utils", + "ui": "@/components/ui", + "lib": "@/lib", + "hooks": "@/hooks" + }, + "registries": {} +} diff --git a/eslint.config.mjs b/eslint.config.mjs new file mode 100644 index 00000000..06dcd91b --- /dev/null +++ b/eslint.config.mjs @@ -0,0 +1,35 @@ +import { FlatCompat } from '@eslint/eslintrc'; +import { dirname } from 'path'; +import { fileURLToPath } from 'url'; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = dirname(__filename); + +const compat = new FlatCompat({ + baseDirectory: __dirname, +}); + +const eslintConfig = [ + ...compat.extends( + 'next/core-web-vitals', + 'next/typescript', + 'plugin:prettier/recommended', + ), + { + ignores: [ + 'node_modules/**', + '.next/**', + 'out/**', + 'build/**', + 'next-env.d.ts', + 'scripts/**', + 'src/global/api/ep.ts', + ], + rules: { + '@typescript-eslint/no-empty-object-type': 'off', // 빈 타입 객체 허용 + '@next/next/no-img-element': 'off', + }, + }, +]; + +export default eslintConfig; diff --git a/middleware.ts b/middleware.ts new file mode 100644 index 00000000..dd971821 --- /dev/null +++ b/middleware.ts @@ -0,0 +1,13 @@ +import type { NextRequest } from 'next/server'; +import { NextResponse } from 'next/server'; + +export function middleware(req: NextRequest) { + const res = NextResponse.next(); + + // HTML 요청일 때만 붙임 + if (req.nextUrl.pathname === '/') { + res.headers.set('Cache-Control', 'public, max-age=86400'); + } + + return res; +} diff --git a/next.config.ts b/next.config.ts new file mode 100644 index 00000000..40a41346 --- /dev/null +++ b/next.config.ts @@ -0,0 +1,11 @@ +import type { NextConfig } from 'next'; + +const nextConfig: NextConfig = { + output: 'standalone', + images: { + domains: ['say-cheese-profile.edge.naverncp.com'], + deviceSizes: [320, 480, 590, 640, 750, 828, 1080, 1200, 1920], + }, +}; + +export default nextConfig; diff --git a/package.json b/package.json new file mode 100644 index 00000000..ab099f01 --- /dev/null +++ b/package.json @@ -0,0 +1,67 @@ +{ + "name": "cheeese_fe", + "version": "0.1.0", + "packageManager": "pnpm@9.12.3", + "engines": { + "node": "22.20.0" + }, + "private": true, + "scripts": { + "dev": "next dev --turbopack", + "build": "next build --turbopack", + "build:docker": "docker build -t cheese_frontend .", + "start": "next start", + "lint": "eslint", + "test": "vitest", + "token": "node scripts/generate-color-token.mjs", + "ep": "node scripts/generate-endpoint.ts" + }, + "dependencies": { + "@headlessui/react": "^2.2.9", + "@radix-ui/react-alert-dialog": "^1.1.15", + "@radix-ui/react-dialog": "^1.1.15", + "@radix-ui/react-popover": "^1.1.15", + "@radix-ui/react-slot": "^1.2.3", + "@tanstack/react-query": "^5.90.2", + "axios": "^1.12.2", + "class-variance-authority": "^0.7.1", + "clsx": "^2.1.1", + "date-fns": "^4.1.0", + "embla-carousel-react": "^8.6.0", + "emoji-picker-react": "^4.14.2", + "exifr": "^7.1.3", + "framer-motion": "^12.23.24", + "heic-to": "^1.3.0", + "html-to-image": "^1.11.13", + "jszip": "^3.10.1", + "lottie-react": "^2.4.1", + "lucide-react": "^0.546.0", + "next": "15.5.7", + "react": "19.1.2", + "react-day-picker": "^9.11.1", + "react-dom": "19.1.2", + "react-qr-code": "^2.0.18", + "swiper": "^12.0.3", + "tailwind-merge": "^3.3.1", + "use-sync-external-store": "^1.6.0", + "vaul": "^1.1.2", + "zustand": "^5.0.8" + }, + "devDependencies": { + "@eslint/eslintrc": "^3", + "@tailwindcss/postcss": "^4", + "@types/node": "^20", + "@types/react": "^19", + "@types/react-dom": "^19", + "eslint": "^9", + "eslint-config-next": "15.5.4", + "eslint-config-prettier": "^10.1.8", + "eslint-plugin-prettier": "^5.5.4", + "prettier": "^3.6.2", + "prettier-plugin-tailwindcss": "^0.7.1", + "tailwindcss": "^4", + "tw-animate-css": "^1.4.0", + "typescript": "^5", + "vitest": "^4.0.14" + } +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml new file mode 100644 index 00000000..8b73bdfb --- /dev/null +++ b/pnpm-lock.yaml @@ -0,0 +1,5857 @@ +lockfileVersion: '9.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +importers: + + .: + dependencies: + '@headlessui/react': + specifier: ^2.2.9 + version: 2.2.9(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-alert-dialog': + specifier: ^1.1.15 + version: 1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-dialog': + specifier: ^1.1.15 + version: 1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-popover': + specifier: ^1.1.15 + version: 1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-slot': + specifier: ^1.2.3 + version: 1.2.3(@types/react@19.2.2)(react@19.1.2) + '@tanstack/react-query': + specifier: ^5.90.2 + version: 5.90.5(react@19.1.2) + axios: + specifier: ^1.12.2 + version: 1.12.2 + class-variance-authority: + specifier: ^0.7.1 + version: 0.7.1 + clsx: + specifier: ^2.1.1 + version: 2.1.1 + date-fns: + specifier: ^4.1.0 + version: 4.1.0 + embla-carousel-react: + specifier: ^8.6.0 + version: 8.6.0(react@19.1.2) + emoji-picker-react: + specifier: ^4.14.2 + version: 4.14.2(react@19.1.2) + exifr: + specifier: ^7.1.3 + version: 7.1.3 + framer-motion: + specifier: ^12.23.24 + version: 12.23.24(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + heic-to: + specifier: ^1.3.0 + version: 1.3.0 + html-to-image: + specifier: ^1.11.13 + version: 1.11.13 + jszip: + specifier: ^3.10.1 + version: 3.10.1 + lottie-react: + specifier: ^2.4.1 + version: 2.4.1(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + lucide-react: + specifier: ^0.546.0 + version: 0.546.0(react@19.1.2) + next: + specifier: 15.5.7 + version: 15.5.7(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + react: + specifier: 19.1.2 + version: 19.1.2 + react-day-picker: + specifier: ^9.11.1 + version: 9.11.1(react@19.1.2) + react-dom: + specifier: 19.1.2 + version: 19.1.2(react@19.1.2) + react-qr-code: + specifier: ^2.0.18 + version: 2.0.18(react@19.1.2) + swiper: + specifier: ^12.0.3 + version: 12.0.3 + tailwind-merge: + specifier: ^3.3.1 + version: 3.3.1 + use-sync-external-store: + specifier: ^1.6.0 + version: 1.6.0(react@19.1.2) + vaul: + specifier: ^1.1.2 + version: 1.1.2(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + zustand: + specifier: ^5.0.8 + version: 5.0.8(@types/react@19.2.2)(react@19.1.2)(use-sync-external-store@1.6.0(react@19.1.2)) + devDependencies: + '@eslint/eslintrc': + specifier: ^3 + version: 3.3.1 + '@tailwindcss/postcss': + specifier: ^4 + version: 4.1.16 + '@types/node': + specifier: ^20 + version: 20.19.23 + '@types/react': + specifier: ^19 + version: 19.2.2 + '@types/react-dom': + specifier: ^19 + version: 19.2.2(@types/react@19.2.2) + eslint: + specifier: ^9 + version: 9.38.0(jiti@2.6.1) + eslint-config-next: + specifier: 15.5.4 + version: 15.5.4(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + eslint-config-prettier: + specifier: ^10.1.8 + version: 10.1.8(eslint@9.38.0(jiti@2.6.1)) + eslint-plugin-prettier: + specifier: ^5.5.4 + version: 5.5.4(eslint-config-prettier@10.1.8(eslint@9.38.0(jiti@2.6.1)))(eslint@9.38.0(jiti@2.6.1))(prettier@3.6.2) + prettier: + specifier: ^3.6.2 + version: 3.6.2 + prettier-plugin-tailwindcss: + specifier: ^0.7.1 + version: 0.7.1(prettier@3.6.2) + tailwindcss: + specifier: ^4 + version: 4.1.16 + tw-animate-css: + specifier: ^1.4.0 + version: 1.4.0 + typescript: + specifier: ^5 + version: 5.9.3 + vitest: + specifier: ^4.0.14 + version: 4.0.14(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2) + +packages: + + '@alloc/quick-lru@5.2.0': + resolution: {integrity: sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==} + engines: {node: '>=10'} + + '@date-fns/tz@1.4.1': + resolution: {integrity: sha512-P5LUNhtbj6YfI3iJjw5EL9eUAG6OitD0W3fWQcpQjDRc/QIsL0tRNuO1PcDvPccWL1fSTXXdE1ds+l95DV/OFA==} + + '@emnapi/core@1.6.0': + resolution: {integrity: sha512-zq/ay+9fNIJJtJiZxdTnXS20PllcYMX3OE23ESc4HK/bdYu3cOWYVhsOhVnXALfU/uqJIxn5NBPd9z4v+SfoSg==} + + '@emnapi/runtime@1.6.0': + resolution: {integrity: sha512-obtUmAHTMjll499P+D9A3axeJFlhdjOWdKUNs/U6QIGT7V5RjcUW1xToAzjvmgTSQhDbYn/NwfTRoJcQ2rNBxA==} + + '@emnapi/wasi-threads@1.1.0': + resolution: {integrity: sha512-WI0DdZ8xFSbgMjR1sFsKABJ/C5OnRrjT06JXbZKexJGrDuPTzZdDYfFlsgcCXCyf+suG5QU2e/y1Wo2V/OapLQ==} + + '@esbuild/aix-ppc64@0.25.12': + resolution: {integrity: sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==} + engines: {node: '>=18'} + cpu: [ppc64] + os: [aix] + + '@esbuild/android-arm64@0.25.12': + resolution: {integrity: sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [android] + + '@esbuild/android-arm@0.25.12': + resolution: {integrity: sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==} + engines: {node: '>=18'} + cpu: [arm] + os: [android] + + '@esbuild/android-x64@0.25.12': + resolution: {integrity: sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==} + engines: {node: '>=18'} + cpu: [x64] + os: [android] + + '@esbuild/darwin-arm64@0.25.12': + resolution: {integrity: sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [darwin] + + '@esbuild/darwin-x64@0.25.12': + resolution: {integrity: sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==} + engines: {node: '>=18'} + cpu: [x64] + os: [darwin] + + '@esbuild/freebsd-arm64@0.25.12': + resolution: {integrity: sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.25.12': + resolution: {integrity: sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [freebsd] + + '@esbuild/linux-arm64@0.25.12': + resolution: {integrity: sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==} + engines: {node: '>=18'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm@0.25.12': + resolution: {integrity: sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==} + engines: {node: '>=18'} + cpu: [arm] + os: [linux] + + '@esbuild/linux-ia32@0.25.12': + resolution: {integrity: sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==} + engines: {node: '>=18'} + cpu: [ia32] + os: [linux] + + '@esbuild/linux-loong64@0.25.12': + resolution: {integrity: sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==} + engines: {node: '>=18'} + cpu: [loong64] + os: [linux] + + '@esbuild/linux-mips64el@0.25.12': + resolution: {integrity: sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==} + engines: {node: '>=18'} + cpu: [mips64el] + os: [linux] + + '@esbuild/linux-ppc64@0.25.12': + resolution: {integrity: sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==} + engines: {node: '>=18'} + cpu: [ppc64] + os: [linux] + + '@esbuild/linux-riscv64@0.25.12': + resolution: {integrity: sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==} + engines: {node: '>=18'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-s390x@0.25.12': + resolution: {integrity: sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==} + engines: {node: '>=18'} + cpu: [s390x] + os: [linux] + + '@esbuild/linux-x64@0.25.12': + resolution: {integrity: sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==} + engines: {node: '>=18'} + cpu: [x64] + os: [linux] + + '@esbuild/netbsd-arm64@0.25.12': + resolution: {integrity: sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [netbsd] + + '@esbuild/netbsd-x64@0.25.12': + resolution: {integrity: sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [netbsd] + + '@esbuild/openbsd-arm64@0.25.12': + resolution: {integrity: sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==} + engines: {node: '>=18'} + cpu: [arm64] + os: [openbsd] + + '@esbuild/openbsd-x64@0.25.12': + resolution: {integrity: sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==} + engines: {node: '>=18'} + cpu: [x64] + os: [openbsd] + + '@esbuild/openharmony-arm64@0.25.12': + resolution: {integrity: sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [openharmony] + + '@esbuild/sunos-x64@0.25.12': + resolution: {integrity: sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==} + engines: {node: '>=18'} + cpu: [x64] + os: [sunos] + + '@esbuild/win32-arm64@0.25.12': + resolution: {integrity: sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-ia32@0.25.12': + resolution: {integrity: sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==} + engines: {node: '>=18'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-x64@0.25.12': + resolution: {integrity: sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==} + engines: {node: '>=18'} + cpu: [x64] + os: [win32] + + '@eslint-community/eslint-utils@4.9.0': + resolution: {integrity: sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 + + '@eslint-community/regexpp@4.12.2': + resolution: {integrity: sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==} + engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} + + '@eslint/config-array@0.21.1': + resolution: {integrity: sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/config-helpers@0.4.1': + resolution: {integrity: sha512-csZAzkNhsgwb0I/UAV6/RGFTbiakPCf0ZrGmrIxQpYvGZ00PhTkSnyKNolphgIvmnJeGw6rcGVEXfTzUnFuEvw==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/core@0.16.0': + resolution: {integrity: sha512-nmC8/totwobIiFcGkDza3GIKfAw1+hLiYVrh3I1nIomQ8PEr5cxg34jnkmGawul/ep52wGRAcyeDCNtWKSOj4Q==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/eslintrc@3.3.1': + resolution: {integrity: sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/js@9.38.0': + resolution: {integrity: sha512-UZ1VpFvXf9J06YG9xQBdnzU+kthors6KjhMAl6f4gH4usHyh31rUf2DLGInT8RFYIReYXNSydgPY0V2LuWgl7A==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/object-schema@2.1.7': + resolution: {integrity: sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@eslint/plugin-kit@0.4.0': + resolution: {integrity: sha512-sB5uyeq+dwCWyPi31B2gQlVlo+j5brPlWx4yZBrEaRo/nhdDE8Xke1gsGgtiBdaBTxuTkceLVuVt/pclrasb0A==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@floating-ui/core@1.7.3': + resolution: {integrity: sha512-sGnvb5dmrJaKEZ+LDIpguvdX3bDlEllmv4/ClQ9awcmCZrlx5jQyyMWFM5kBI+EyNOCDDiKk8il0zeuX3Zlg/w==} + + '@floating-ui/dom@1.7.4': + resolution: {integrity: sha512-OOchDgh4F2CchOX94cRVqhvy7b3AFb+/rQXyswmzmGakRfkMgoWVjfnLWkRirfLEfuD4ysVW16eXzwt3jHIzKA==} + + '@floating-ui/react-dom@2.1.6': + resolution: {integrity: sha512-4JX6rEatQEvlmgU80wZyq9RT96HZJa88q8hp0pBd+LrczeDI4o6uA2M+uvxngVHo4Ihr8uibXxH6+70zhAFrVw==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/react@0.26.28': + resolution: {integrity: sha512-yORQuuAtVpiRjpMhdc0wJj06b9JFjrYF4qp96j++v2NBpbi6SEGF7donUJ3TMieerQ6qVkAv1tgr7L4r5roTqw==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' + + '@floating-ui/utils@0.2.10': + resolution: {integrity: sha512-aGTxbpbg8/b5JfU1HXSrbH3wXZuLPJcNEcZQFMxLs3oSzgtVu6nFPkbbGGUvBcUjKV2YyB9Wxxabo+HEH9tcRQ==} + + '@headlessui/react@2.2.9': + resolution: {integrity: sha512-Mb+Un58gwBn0/yWZfyrCh0TJyurtT+dETj7YHleylHk5od3dv2XqETPGWMyQ5/7sYN7oWdyM1u9MvC0OC8UmzQ==} + engines: {node: '>=10'} + peerDependencies: + react: ^18 || ^19 || ^19.0.0-rc + react-dom: ^18 || ^19 || ^19.0.0-rc + + '@humanfs/core@0.19.1': + resolution: {integrity: sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==} + engines: {node: '>=18.18.0'} + + '@humanfs/node@0.16.7': + resolution: {integrity: sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==} + engines: {node: '>=18.18.0'} + + '@humanwhocodes/module-importer@1.0.1': + resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} + engines: {node: '>=12.22'} + + '@humanwhocodes/retry@0.4.3': + resolution: {integrity: sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==} + engines: {node: '>=18.18'} + + '@img/colour@1.0.0': + resolution: {integrity: sha512-A5P/LfWGFSl6nsckYtjw9da+19jB8hkJ6ACTGcDfEJ0aE+l2n2El7dsVM7UVHZQ9s2lmYMWlrS21YLy2IR1LUw==} + engines: {node: '>=18'} + + '@img/sharp-darwin-arm64@0.34.4': + resolution: {integrity: sha512-sitdlPzDVyvmINUdJle3TNHl+AG9QcwiAMsXmccqsCOMZNIdW2/7S26w0LyU8euiLVzFBL3dXPwVCq/ODnf2vA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [arm64] + os: [darwin] + + '@img/sharp-darwin-x64@0.34.4': + resolution: {integrity: sha512-rZheupWIoa3+SOdF/IcUe1ah4ZDpKBGWcsPX6MT0lYniH9micvIU7HQkYTfrx5Xi8u+YqwLtxC/3vl8TQN6rMg==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [x64] + os: [darwin] + + '@img/sharp-libvips-darwin-arm64@1.2.3': + resolution: {integrity: sha512-QzWAKo7kpHxbuHqUC28DZ9pIKpSi2ts2OJnoIGI26+HMgq92ZZ4vk8iJd4XsxN+tYfNJxzH6W62X5eTcsBymHw==} + cpu: [arm64] + os: [darwin] + + '@img/sharp-libvips-darwin-x64@1.2.3': + resolution: {integrity: sha512-Ju+g2xn1E2AKO6YBhxjj+ACcsPQRHT0bhpglxcEf+3uyPY+/gL8veniKoo96335ZaPo03bdDXMv0t+BBFAbmRA==} + cpu: [x64] + os: [darwin] + + '@img/sharp-libvips-linux-arm64@1.2.3': + resolution: {integrity: sha512-I4RxkXU90cpufazhGPyVujYwfIm9Nk1QDEmiIsaPwdnm013F7RIceaCc87kAH+oUB1ezqEvC6ga4m7MSlqsJvQ==} + cpu: [arm64] + os: [linux] + + '@img/sharp-libvips-linux-arm@1.2.3': + resolution: {integrity: sha512-x1uE93lyP6wEwGvgAIV0gP6zmaL/a0tGzJs/BIDDG0zeBhMnuUPm7ptxGhUbcGs4okDJrk4nxgrmxpib9g6HpA==} + cpu: [arm] + os: [linux] + + '@img/sharp-libvips-linux-ppc64@1.2.3': + resolution: {integrity: sha512-Y2T7IsQvJLMCBM+pmPbM3bKT/yYJvVtLJGfCs4Sp95SjvnFIjynbjzsa7dY1fRJX45FTSfDksbTp6AGWudiyCg==} + cpu: [ppc64] + os: [linux] + + '@img/sharp-libvips-linux-s390x@1.2.3': + resolution: {integrity: sha512-RgWrs/gVU7f+K7P+KeHFaBAJlNkD1nIZuVXdQv6S+fNA6syCcoboNjsV2Pou7zNlVdNQoQUpQTk8SWDHUA3y/w==} + cpu: [s390x] + os: [linux] + + '@img/sharp-libvips-linux-x64@1.2.3': + resolution: {integrity: sha512-3JU7LmR85K6bBiRzSUc/Ff9JBVIFVvq6bomKE0e63UXGeRw2HPVEjoJke1Yx+iU4rL7/7kUjES4dZ/81Qjhyxg==} + cpu: [x64] + os: [linux] + + '@img/sharp-libvips-linuxmusl-arm64@1.2.3': + resolution: {integrity: sha512-F9q83RZ8yaCwENw1GieztSfj5msz7GGykG/BA+MOUefvER69K/ubgFHNeSyUu64amHIYKGDs4sRCMzXVj8sEyw==} + cpu: [arm64] + os: [linux] + + '@img/sharp-libvips-linuxmusl-x64@1.2.3': + resolution: {integrity: sha512-U5PUY5jbc45ANM6tSJpsgqmBF/VsL6LnxJmIf11kB7J5DctHgqm0SkuXzVWtIY90GnJxKnC/JT251TDnk1fu/g==} + cpu: [x64] + os: [linux] + + '@img/sharp-linux-arm64@0.34.4': + resolution: {integrity: sha512-YXU1F/mN/Wu786tl72CyJjP/Ngl8mGHN1hST4BGl+hiW5jhCnV2uRVTNOcaYPs73NeT/H8Upm3y9582JVuZHrQ==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [arm64] + os: [linux] + + '@img/sharp-linux-arm@0.34.4': + resolution: {integrity: sha512-Xyam4mlqM0KkTHYVSuc6wXRmM7LGN0P12li03jAnZ3EJWZqj83+hi8Y9UxZUbxsgsK1qOEwg7O0Bc0LjqQVtxA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [arm] + os: [linux] + + '@img/sharp-linux-ppc64@0.34.4': + resolution: {integrity: sha512-F4PDtF4Cy8L8hXA2p3TO6s4aDt93v+LKmpcYFLAVdkkD3hSxZzee0rh6/+94FpAynsuMpLX5h+LRsSG3rIciUQ==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [ppc64] + os: [linux] + + '@img/sharp-linux-s390x@0.34.4': + resolution: {integrity: sha512-qVrZKE9Bsnzy+myf7lFKvng6bQzhNUAYcVORq2P7bDlvmF6u2sCmK2KyEQEBdYk+u3T01pVsPrkj943T1aJAsw==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [s390x] + os: [linux] + + '@img/sharp-linux-x64@0.34.4': + resolution: {integrity: sha512-ZfGtcp2xS51iG79c6Vhw9CWqQC8l2Ot8dygxoDoIQPTat/Ov3qAa8qpxSrtAEAJW+UjTXc4yxCjNfxm4h6Xm2A==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [x64] + os: [linux] + + '@img/sharp-linuxmusl-arm64@0.34.4': + resolution: {integrity: sha512-8hDVvW9eu4yHWnjaOOR8kHVrew1iIX+MUgwxSuH2XyYeNRtLUe4VNioSqbNkB7ZYQJj9rUTT4PyRscyk2PXFKA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [arm64] + os: [linux] + + '@img/sharp-linuxmusl-x64@0.34.4': + resolution: {integrity: sha512-lU0aA5L8QTlfKjpDCEFOZsTYGn3AEiO6db8W5aQDxj0nQkVrZWmN3ZP9sYKWJdtq3PWPhUNlqehWyXpYDcI9Sg==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [x64] + os: [linux] + + '@img/sharp-wasm32@0.34.4': + resolution: {integrity: sha512-33QL6ZO/qpRyG7woB/HUALz28WnTMI2W1jgX3Nu2bypqLIKx/QKMILLJzJjI+SIbvXdG9fUnmrxR7vbi1sTBeA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [wasm32] + + '@img/sharp-win32-arm64@0.34.4': + resolution: {integrity: sha512-2Q250do/5WXTwxW3zjsEuMSv5sUU4Tq9VThWKlU2EYLm4MB7ZeMwF+SFJutldYODXF6jzc6YEOC+VfX0SZQPqA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [arm64] + os: [win32] + + '@img/sharp-win32-ia32@0.34.4': + resolution: {integrity: sha512-3ZeLue5V82dT92CNL6rsal6I2weKw1cYu+rGKm8fOCCtJTR2gYeUfY3FqUnIJsMUPIH68oS5jmZ0NiJ508YpEw==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [ia32] + os: [win32] + + '@img/sharp-win32-x64@0.34.4': + resolution: {integrity: sha512-xIyj4wpYs8J18sVN3mSQjwrw7fKUqRw+Z5rnHNCy5fYTxigBz81u5mOMPmFumwjcn8+ld1ppptMBCLic1nz6ig==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + cpu: [x64] + os: [win32] + + '@jridgewell/gen-mapping@0.3.13': + resolution: {integrity: sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==} + + '@jridgewell/remapping@2.3.5': + resolution: {integrity: sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==} + + '@jridgewell/resolve-uri@3.1.2': + resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} + engines: {node: '>=6.0.0'} + + '@jridgewell/sourcemap-codec@1.5.5': + resolution: {integrity: sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==} + + '@jridgewell/trace-mapping@0.3.31': + resolution: {integrity: sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==} + + '@napi-rs/wasm-runtime@0.2.12': + resolution: {integrity: sha512-ZVWUcfwY4E/yPitQJl481FjFo3K22D6qF0DuFH6Y/nbnE11GY5uguDxZMGXPQ8WQ0128MXQD7TnfHyK4oWoIJQ==} + + '@next/env@15.5.7': + resolution: {integrity: sha512-4h6Y2NyEkIEN7Z8YxkA27pq6zTkS09bUSYC0xjd0NpwFxjnIKeZEeH591o5WECSmjpUhLn3H2QLJcDye3Uzcvg==} + + '@next/eslint-plugin-next@15.5.4': + resolution: {integrity: sha512-SR1vhXNNg16T4zffhJ4TS7Xn7eq4NfKfcOsRwea7RIAHrjRpI9ALYbamqIJqkAhowLlERffiwk0FMvTLNdnVtw==} + + '@next/swc-darwin-arm64@15.5.7': + resolution: {integrity: sha512-IZwtxCEpI91HVU/rAUOOobWSZv4P2DeTtNaCdHqLcTJU4wdNXgAySvKa/qJCgR5m6KI8UsKDXtO2B31jcaw1Yw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [darwin] + + '@next/swc-darwin-x64@15.5.7': + resolution: {integrity: sha512-UP6CaDBcqaCBuiq/gfCEJw7sPEoX1aIjZHnBWN9v9qYHQdMKvCKcAVs4OX1vIjeE+tC5EIuwDTVIoXpUes29lg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [darwin] + + '@next/swc-linux-arm64-gnu@15.5.7': + resolution: {integrity: sha512-NCslw3GrNIw7OgmRBxHtdWFQYhexoUCq+0oS2ccjyYLtcn1SzGzeM54jpTFonIMUjNbHmpKpziXnpxhSWLcmBA==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@next/swc-linux-arm64-musl@15.5.7': + resolution: {integrity: sha512-nfymt+SE5cvtTrG9u1wdoxBr9bVB7mtKTcj0ltRn6gkP/2Nu1zM5ei8rwP9qKQP0Y//umK+TtkKgNtfboBxRrw==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@next/swc-linux-x64-gnu@15.5.7': + resolution: {integrity: sha512-hvXcZvCaaEbCZcVzcY7E1uXN9xWZfFvkNHwbe/n4OkRhFWrs1J1QV+4U1BN06tXLdaS4DazEGXwgqnu/VMcmqw==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@next/swc-linux-x64-musl@15.5.7': + resolution: {integrity: sha512-4IUO539b8FmF0odY6/SqANJdgwn1xs1GkPO5doZugwZ3ETF6JUdckk7RGmsfSf7ws8Qb2YB5It33mvNL/0acqA==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@next/swc-win32-arm64-msvc@15.5.7': + resolution: {integrity: sha512-CpJVTkYI3ZajQkC5vajM7/ApKJUOlm6uP4BknM3XKvJ7VXAvCqSjSLmM0LKdYzn6nBJVSjdclx8nYJSa3xlTgQ==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [win32] + + '@next/swc-win32-x64-msvc@15.5.7': + resolution: {integrity: sha512-gMzgBX164I6DN+9/PGA+9dQiwmTkE4TloBNx8Kv9UiGARsr9Nba7IpcBRA1iTV9vwlYnrE3Uy6I7Aj6qLjQuqw==} + engines: {node: '>= 10'} + cpu: [x64] + os: [win32] + + '@nodelib/fs.scandir@2.1.5': + resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} + engines: {node: '>= 8'} + + '@nodelib/fs.stat@2.0.5': + resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} + engines: {node: '>= 8'} + + '@nodelib/fs.walk@1.2.8': + resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} + engines: {node: '>= 8'} + + '@nolyfill/is-core-module@1.0.39': + resolution: {integrity: sha512-nn5ozdjYQpUCZlWGuxcJY/KpxkWQs4DcbMCmKojjyrYDEAGy4Ce19NN4v5MduafTwJlbKc99UA8YhSVqq9yPZA==} + engines: {node: '>=12.4.0'} + + '@pkgr/core@0.2.9': + resolution: {integrity: sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA==} + engines: {node: ^12.20.0 || ^14.18.0 || >=16.0.0} + + '@radix-ui/primitive@1.1.3': + resolution: {integrity: sha512-JTF99U/6XIjCBo0wqkU5sK10glYe27MRRsfwoiq5zzOEZLHU3A3KCMa5X/azekYRCJ0HlwI0crAXS/5dEHTzDg==} + + '@radix-ui/react-alert-dialog@1.1.15': + resolution: {integrity: sha512-oTVLkEw5GpdRe29BqJ0LSDFWI3qu0vR1M0mUkOQWDIUnY/QIkLpgDMWuKxP94c2NAC2LGcgVhG1ImF3jkZ5wXw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-arrow@1.1.7': + resolution: {integrity: sha512-F+M1tLhO+mlQaOWspE8Wstg+z6PwxwRd8oQ8IXceWz92kfAmalTRf0EjrouQeo7QssEPfCn05B4Ihs1K9WQ/7w==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-compose-refs@1.1.2': + resolution: {integrity: sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-context@1.1.2': + resolution: {integrity: sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-dialog@1.1.15': + resolution: {integrity: sha512-TCglVRtzlffRNxRMEyR36DGBLJpeusFcgMVD9PZEzAKnUs1lKCgX5u9BmC2Yg+LL9MgZDugFFs1Vl+Jp4t/PGw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-dismissable-layer@1.1.11': + resolution: {integrity: sha512-Nqcp+t5cTB8BinFkZgXiMJniQH0PsUt2k51FUhbdfeKvc4ACcG2uQniY/8+h1Yv6Kza4Q7lD7PQV0z0oicE0Mg==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-focus-guards@1.1.3': + resolution: {integrity: sha512-0rFg/Rj2Q62NCm62jZw0QX7a3sz6QCQU0LpZdNrJX8byRGaGVTqbrW9jAoIAHyMQqsNpeZ81YgSizOt5WXq0Pw==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-focus-scope@1.1.7': + resolution: {integrity: sha512-t2ODlkXBQyn7jkl6TNaw/MtVEVvIGelJDCG41Okq/KwUsJBwQ4XVZsHAVUkK4mBv3ewiAS3PGuUWuY2BoK4ZUw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-id@1.1.1': + resolution: {integrity: sha512-kGkGegYIdQsOb4XjsfM97rXsiHaBwco+hFI66oO4s9LU+PLAC5oJ7khdOVFxkhsmlbpUqDAvXw11CluXP+jkHg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-popover@1.1.15': + resolution: {integrity: sha512-kr0X2+6Yy/vJzLYJUPCZEc8SfQcf+1COFoAqauJm74umQhta9M7lNJHP7QQS3vkvcGLQUbWpMzwrXYwrYztHKA==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-popper@1.2.8': + resolution: {integrity: sha512-0NJQ4LFFUuWkE7Oxf0htBKS6zLkkjBH+hM1uk7Ng705ReR8m/uelduy1DBo0PyBXPKVnBA6YBlU94MBGXrSBCw==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-portal@1.1.9': + resolution: {integrity: sha512-bpIxvq03if6UNwXZ+HTK71JLh4APvnXntDc6XOX8UVq4XQOVl7lwok0AvIl+b8zgCw3fSaVTZMpAPPagXbKmHQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-presence@1.1.5': + resolution: {integrity: sha512-/jfEwNDdQVBCNvjkGit4h6pMOzq8bHkopq458dPt2lMjx+eBQUohZNG9A7DtO/O5ukSbxuaNGXMjHicgwy6rQQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-primitive@2.1.3': + resolution: {integrity: sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/react-slot@1.2.3': + resolution: {integrity: sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-callback-ref@1.1.1': + resolution: {integrity: sha512-FkBMwD+qbGQeMu1cOHnuGB6x4yzPjho8ap5WtbEJ26umhgqVXbhekKUQO+hZEL1vU92a3wHwdp0HAcqAUF5iDg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-controllable-state@1.2.2': + resolution: {integrity: sha512-BjasUjixPFdS+NKkypcyyN5Pmg83Olst0+c6vGov0diwTEo6mgdqVR6hxcEgFuh4QrAs7Rc+9KuGJ9TVCj0Zzg==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-effect-event@0.0.2': + resolution: {integrity: sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-escape-keydown@1.1.1': + resolution: {integrity: sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-layout-effect@1.1.1': + resolution: {integrity: sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-rect@1.1.1': + resolution: {integrity: sha512-QTYuDesS0VtuHNNvMh+CjlKJ4LJickCMUAqjlE3+j8w+RlRpwyX3apEQKGFzbZGdo7XNG1tXa+bQqIE7HIXT2w==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-size@1.1.1': + resolution: {integrity: sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/rect@1.1.1': + resolution: {integrity: sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw==} + + '@react-aria/focus@3.21.2': + resolution: {integrity: sha512-JWaCR7wJVggj+ldmM/cb/DXFg47CXR55lznJhZBh4XVqJjMKwaOOqpT5vNN7kpC1wUpXicGNuDnJDN1S/+6dhQ==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + react-dom: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@react-aria/interactions@3.25.6': + resolution: {integrity: sha512-5UgwZmohpixwNMVkMvn9K1ceJe6TzlRlAfuYoQDUuOkk62/JVJNDLAPKIf5YMRc7d2B0rmfgaZLMtbREb0Zvkw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + react-dom: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@react-aria/ssr@3.9.10': + resolution: {integrity: sha512-hvTm77Pf+pMBhuBm760Li0BVIO38jv1IBws1xFm1NoL26PU+fe+FMW5+VZWyANR6nYL65joaJKZqOdTQMkO9IQ==} + engines: {node: '>= 12'} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@react-aria/utils@3.31.0': + resolution: {integrity: sha512-ABOzCsZrWzf78ysswmguJbx3McQUja7yeGj6/vZo4JVsZNlxAN+E9rs381ExBRI0KzVo6iBTeX5De8eMZPJXig==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + react-dom: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@react-stately/flags@3.1.2': + resolution: {integrity: sha512-2HjFcZx1MyQXoPqcBGALwWWmgFVUk2TuKVIQxCbRq7fPyWXIl6VHcakCLurdtYC2Iks7zizvz0Idv48MQ38DWg==} + + '@react-stately/utils@3.10.8': + resolution: {integrity: sha512-SN3/h7SzRsusVQjQ4v10LaVsDc81jyyR0DD5HnsQitm/I5WDpaSr2nRHtyloPFU48jlql1XX/S04T2DLQM7Y3g==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@react-types/shared@3.32.1': + resolution: {integrity: sha512-famxyD5emrGGpFuUlgOP6fVW2h/ZaF405G5KDi3zPHzyjAWys/8W6NAVJtNbkCkhedmvL0xOhvt8feGXyXaw5w==} + peerDependencies: + react: ^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0-rc.1 + + '@rollup/rollup-android-arm-eabi@4.53.3': + resolution: {integrity: sha512-mRSi+4cBjrRLoaal2PnqH82Wqyb+d3HsPUN/W+WslCXsZsyHa9ZeQQX/pQsZaVIWDkPcpV6jJ+3KLbTbgnwv8w==} + cpu: [arm] + os: [android] + + '@rollup/rollup-android-arm64@4.53.3': + resolution: {integrity: sha512-CbDGaMpdE9sh7sCmTrTUyllhrg65t6SwhjlMJsLr+J8YjFuPmCEjbBSx4Z/e4SmDyH3aB5hGaJUP2ltV/vcs4w==} + cpu: [arm64] + os: [android] + + '@rollup/rollup-darwin-arm64@4.53.3': + resolution: {integrity: sha512-Nr7SlQeqIBpOV6BHHGZgYBuSdanCXuw09hon14MGOLGmXAFYjx1wNvquVPmpZnl0tLjg25dEdr4IQ6GgyToCUA==} + cpu: [arm64] + os: [darwin] + + '@rollup/rollup-darwin-x64@4.53.3': + resolution: {integrity: sha512-DZ8N4CSNfl965CmPktJ8oBnfYr3F8dTTNBQkRlffnUarJ2ohudQD17sZBa097J8xhQ26AwhHJ5mvUyQW8ddTsQ==} + cpu: [x64] + os: [darwin] + + '@rollup/rollup-freebsd-arm64@4.53.3': + resolution: {integrity: sha512-yMTrCrK92aGyi7GuDNtGn2sNW+Gdb4vErx4t3Gv/Tr+1zRb8ax4z8GWVRfr3Jw8zJWvpGHNpss3vVlbF58DZ4w==} + cpu: [arm64] + os: [freebsd] + + '@rollup/rollup-freebsd-x64@4.53.3': + resolution: {integrity: sha512-lMfF8X7QhdQzseM6XaX0vbno2m3hlyZFhwcndRMw8fbAGUGL3WFMBdK0hbUBIUYcEcMhVLr1SIamDeuLBnXS+Q==} + cpu: [x64] + os: [freebsd] + + '@rollup/rollup-linux-arm-gnueabihf@4.53.3': + resolution: {integrity: sha512-k9oD15soC/Ln6d2Wv/JOFPzZXIAIFLp6B+i14KhxAfnq76ajt0EhYc5YPeX6W1xJkAdItcVT+JhKl1QZh44/qw==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm-musleabihf@4.53.3': + resolution: {integrity: sha512-vTNlKq+N6CK/8UktsrFuc+/7NlEYVxgaEgRXVUVK258Z5ymho29skzW1sutgYjqNnquGwVUObAaxae8rZ6YMhg==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm64-gnu@4.53.3': + resolution: {integrity: sha512-RGrFLWgMhSxRs/EWJMIFM1O5Mzuz3Xy3/mnxJp/5cVhZ2XoCAxJnmNsEyeMJtpK+wu0FJFWz+QF4mjCA7AUQ3w==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-arm64-musl@4.53.3': + resolution: {integrity: sha512-kASyvfBEWYPEwe0Qv4nfu6pNkITLTb32p4yTgzFCocHnJLAHs+9LjUu9ONIhvfT/5lv4YS5muBHyuV84epBo/A==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-loong64-gnu@4.53.3': + resolution: {integrity: sha512-JiuKcp2teLJwQ7vkJ95EwESWkNRFJD7TQgYmCnrPtlu50b4XvT5MOmurWNrCj3IFdyjBQ5p9vnrX4JM6I8OE7g==} + cpu: [loong64] + os: [linux] + + '@rollup/rollup-linux-ppc64-gnu@4.53.3': + resolution: {integrity: sha512-EoGSa8nd6d3T7zLuqdojxC20oBfNT8nexBbB/rkxgKj5T5vhpAQKKnD+h3UkoMuTyXkP5jTjK/ccNRmQrPNDuw==} + cpu: [ppc64] + os: [linux] + + '@rollup/rollup-linux-riscv64-gnu@4.53.3': + resolution: {integrity: sha512-4s+Wped2IHXHPnAEbIB0YWBv7SDohqxobiiPA1FIWZpX+w9o2i4LezzH/NkFUl8LRci/8udci6cLq+jJQlh+0g==} + cpu: [riscv64] + os: [linux] + + '@rollup/rollup-linux-riscv64-musl@4.53.3': + resolution: {integrity: sha512-68k2g7+0vs2u9CxDt5ktXTngsxOQkSEV/xBbwlqYcUrAVh6P9EgMZvFsnHy4SEiUl46Xf0IObWVbMvPrr2gw8A==} + cpu: [riscv64] + os: [linux] + + '@rollup/rollup-linux-s390x-gnu@4.53.3': + resolution: {integrity: sha512-VYsFMpULAz87ZW6BVYw3I6sWesGpsP9OPcyKe8ofdg9LHxSbRMd7zrVrr5xi/3kMZtpWL/wC+UIJWJYVX5uTKg==} + cpu: [s390x] + os: [linux] + + '@rollup/rollup-linux-x64-gnu@4.53.3': + resolution: {integrity: sha512-3EhFi1FU6YL8HTUJZ51imGJWEX//ajQPfqWLI3BQq4TlvHy4X0MOr5q3D2Zof/ka0d5FNdPwZXm3Yyib/UEd+w==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-linux-x64-musl@4.53.3': + resolution: {integrity: sha512-eoROhjcc6HbZCJr+tvVT8X4fW3/5g/WkGvvmwz/88sDtSJzO7r/blvoBDgISDiCjDRZmHpwud7h+6Q9JxFwq1Q==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-openharmony-arm64@4.53.3': + resolution: {integrity: sha512-OueLAWgrNSPGAdUdIjSWXw+u/02BRTcnfw9PN41D2vq/JSEPnJnVuBgw18VkN8wcd4fjUs+jFHVM4t9+kBSNLw==} + cpu: [arm64] + os: [openharmony] + + '@rollup/rollup-win32-arm64-msvc@4.53.3': + resolution: {integrity: sha512-GOFuKpsxR/whszbF/bzydebLiXIHSgsEUp6M0JI8dWvi+fFa1TD6YQa4aSZHtpmh2/uAlj/Dy+nmby3TJ3pkTw==} + cpu: [arm64] + os: [win32] + + '@rollup/rollup-win32-ia32-msvc@4.53.3': + resolution: {integrity: sha512-iah+THLcBJdpfZ1TstDFbKNznlzoxa8fmnFYK4V67HvmuNYkVdAywJSoteUszvBQ9/HqN2+9AZghbajMsFT+oA==} + cpu: [ia32] + os: [win32] + + '@rollup/rollup-win32-x64-gnu@4.53.3': + resolution: {integrity: sha512-J9QDiOIZlZLdcot5NXEepDkstocktoVjkaKUtqzgzpt2yWjGlbYiKyp05rWwk4nypbYUNoFAztEgixoLaSETkg==} + cpu: [x64] + os: [win32] + + '@rollup/rollup-win32-x64-msvc@4.53.3': + resolution: {integrity: sha512-UhTd8u31dXadv0MopwGgNOBpUVROFKWVQgAg5N1ESyCz8AuBcMqm4AuTjrwgQKGDfoFuz02EuMRHQIw/frmYKQ==} + cpu: [x64] + os: [win32] + + '@rtsao/scc@1.1.0': + resolution: {integrity: sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==} + + '@rushstack/eslint-patch@1.14.1': + resolution: {integrity: sha512-jGTk8UD/RdjsNZW8qq10r0RBvxL8OWtoT+kImlzPDFilmozzM+9QmIJsmze9UiSBrFU45ZxhTYBypn9q9z/VfQ==} + + '@standard-schema/spec@1.0.0': + resolution: {integrity: sha512-m2bOd0f2RT9k8QJx1JN85cZYyH1RqFBdlwtkSlf4tBDYLCiiZnv1fIIwacK6cqwXavOydf0NPToMQgpKq+dVlA==} + + '@swc/helpers@0.5.15': + resolution: {integrity: sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==} + + '@tailwindcss/node@4.1.16': + resolution: {integrity: sha512-BX5iaSsloNuvKNHRN3k2RcCuTEgASTo77mofW0vmeHkfrDWaoFAFvNHpEgtu0eqyypcyiBkDWzSMxJhp3AUVcw==} + + '@tailwindcss/oxide-android-arm64@4.1.16': + resolution: {integrity: sha512-8+ctzkjHgwDJ5caq9IqRSgsP70xhdhJvm+oueS/yhD5ixLhqTw9fSL1OurzMUhBwE5zK26FXLCz2f/RtkISqHA==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [android] + + '@tailwindcss/oxide-darwin-arm64@4.1.16': + resolution: {integrity: sha512-C3oZy5042v2FOALBZtY0JTDnGNdS6w7DxL/odvSny17ORUnaRKhyTse8xYi3yKGyfnTUOdavRCdmc8QqJYwFKA==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [darwin] + + '@tailwindcss/oxide-darwin-x64@4.1.16': + resolution: {integrity: sha512-vjrl/1Ub9+JwU6BP0emgipGjowzYZMjbWCDqwA2Z4vCa+HBSpP4v6U2ddejcHsolsYxwL5r4bPNoamlV0xDdLg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [darwin] + + '@tailwindcss/oxide-freebsd-x64@4.1.16': + resolution: {integrity: sha512-TSMpPYpQLm+aR1wW5rKuUuEruc/oOX3C7H0BTnPDn7W/eMw8W+MRMpiypKMkXZfwH8wqPIRKppuZoedTtNj2tg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [freebsd] + + '@tailwindcss/oxide-linux-arm-gnueabihf@4.1.16': + resolution: {integrity: sha512-p0GGfRg/w0sdsFKBjMYvvKIiKy/LNWLWgV/plR4lUgrsxFAoQBFrXkZ4C0w8IOXfslB9vHK/JGASWD2IefIpvw==} + engines: {node: '>= 10'} + cpu: [arm] + os: [linux] + + '@tailwindcss/oxide-linux-arm64-gnu@4.1.16': + resolution: {integrity: sha512-DoixyMmTNO19rwRPdqviTrG1rYzpxgyYJl8RgQvdAQUzxC1ToLRqtNJpU/ATURSKgIg6uerPw2feW0aS8SNr/w==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@tailwindcss/oxide-linux-arm64-musl@4.1.16': + resolution: {integrity: sha512-H81UXMa9hJhWhaAUca6bU2wm5RRFpuHImrwXBUvPbYb+3jo32I9VIwpOX6hms0fPmA6f2pGVlybO6qU8pF4fzQ==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [linux] + + '@tailwindcss/oxide-linux-x64-gnu@4.1.16': + resolution: {integrity: sha512-ZGHQxDtFC2/ruo7t99Qo2TTIvOERULPl5l0K1g0oK6b5PGqjYMga+FcY1wIUnrUxY56h28FxybtDEla+ICOyew==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@tailwindcss/oxide-linux-x64-musl@4.1.16': + resolution: {integrity: sha512-Oi1tAaa0rcKf1Og9MzKeINZzMLPbhxvm7rno5/zuP1WYmpiG0bEHq4AcRUiG2165/WUzvxkW4XDYCscZWbTLZw==} + engines: {node: '>= 10'} + cpu: [x64] + os: [linux] + + '@tailwindcss/oxide-wasm32-wasi@4.1.16': + resolution: {integrity: sha512-B01u/b8LteGRwucIBmCQ07FVXLzImWESAIMcUU6nvFt/tYsQ6IHz8DmZ5KtvmwxD+iTYBtM1xwoGXswnlu9v0Q==} + engines: {node: '>=14.0.0'} + cpu: [wasm32] + bundledDependencies: + - '@napi-rs/wasm-runtime' + - '@emnapi/core' + - '@emnapi/runtime' + - '@tybys/wasm-util' + - '@emnapi/wasi-threads' + - tslib + + '@tailwindcss/oxide-win32-arm64-msvc@4.1.16': + resolution: {integrity: sha512-zX+Q8sSkGj6HKRTMJXuPvOcP8XfYON24zJBRPlszcH1Np7xuHXhWn8qfFjIujVzvH3BHU+16jBXwgpl20i+v9A==} + engines: {node: '>= 10'} + cpu: [arm64] + os: [win32] + + '@tailwindcss/oxide-win32-x64-msvc@4.1.16': + resolution: {integrity: sha512-m5dDFJUEejbFqP+UXVstd4W/wnxA4F61q8SoL+mqTypId2T2ZpuxosNSgowiCnLp2+Z+rivdU0AqpfgiD7yCBg==} + engines: {node: '>= 10'} + cpu: [x64] + os: [win32] + + '@tailwindcss/oxide@4.1.16': + resolution: {integrity: sha512-2OSv52FRuhdlgyOQqgtQHuCgXnS8nFSYRp2tJ+4WZXKgTxqPy7SMSls8c3mPT5pkZ17SBToGM5LHEJBO7miEdg==} + engines: {node: '>= 10'} + + '@tailwindcss/postcss@4.1.16': + resolution: {integrity: sha512-Qn3SFGPXYQMKR/UtqS+dqvPrzEeBZHrFA92maT4zijCVggdsXnDBMsPFJo1eArX3J+O+Gi+8pV4PkqjLCNBk3A==} + + '@tanstack/query-core@5.90.5': + resolution: {integrity: sha512-wLamYp7FaDq6ZnNehypKI5fNvxHPfTYylE0m/ZpuuzJfJqhR5Pxg9gvGBHZx4n7J+V5Rg5mZxHHTlv25Zt5u+w==} + + '@tanstack/react-query@5.90.5': + resolution: {integrity: sha512-pN+8UWpxZkEJ/Rnnj2v2Sxpx1WFlaa9L6a4UO89p6tTQbeo+m0MS8oYDjbggrR8QcTyjKoYWKS3xJQGr3ExT8Q==} + peerDependencies: + react: ^18 || ^19 + + '@tanstack/react-virtual@3.13.12': + resolution: {integrity: sha512-Gd13QdxPSukP8ZrkbgS2RwoZseTTbQPLnQEn7HY/rqtM+8Zt95f7xKC7N0EsKs7aoz0WzZ+fditZux+F8EzYxA==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + + '@tanstack/virtual-core@3.13.12': + resolution: {integrity: sha512-1YBOJfRHV4sXUmWsFSf5rQor4Ss82G8dQWLRbnk3GA4jeP8hQt1hxXh0tmflpC0dz3VgEv/1+qwPyLeWkQuPFA==} + + '@tybys/wasm-util@0.10.1': + resolution: {integrity: sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg==} + + '@types/chai@5.2.3': + resolution: {integrity: sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==} + + '@types/deep-eql@4.0.2': + resolution: {integrity: sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==} + + '@types/estree@1.0.8': + resolution: {integrity: sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==} + + '@types/json-schema@7.0.15': + resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} + + '@types/json5@0.0.29': + resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==} + + '@types/node@20.19.23': + resolution: {integrity: sha512-yIdlVVVHXpmqRhtyovZAcSy0MiPcYWGkoO4CGe/+jpP0hmNuihm4XhHbADpK++MsiLHP5MVlv+bcgdF99kSiFQ==} + + '@types/react-dom@19.2.2': + resolution: {integrity: sha512-9KQPoO6mZCi7jcIStSnlOWn2nEF3mNmyr3rIAsGnAbQKYbRLyqmeSc39EVgtxXVia+LMT8j3knZLAZAh+xLmrw==} + peerDependencies: + '@types/react': ^19.2.0 + + '@types/react@19.2.2': + resolution: {integrity: sha512-6mDvHUFSjyT2B2yeNx2nUgMxh9LtOWvkhIU3uePn2I2oyNymUAX1NIsdgviM4CH+JSrp2D2hsMvJOkxY+0wNRA==} + + '@typescript-eslint/eslint-plugin@8.46.2': + resolution: {integrity: sha512-ZGBMToy857/NIPaaCucIUQgqueOiq7HeAKkhlvqVV4lm089zUFW6ikRySx2v+cAhKeUCPuWVHeimyk6Dw1iY3w==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + '@typescript-eslint/parser': ^8.46.2 + eslint: ^8.57.0 || ^9.0.0 + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/parser@8.46.2': + resolution: {integrity: sha512-BnOroVl1SgrPLywqxyqdJ4l3S2MsKVLDVxZvjI1Eoe8ev2r3kGDo+PcMihNmDE+6/KjkTubSJnmqGZZjQSBq/g==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + eslint: ^8.57.0 || ^9.0.0 + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/project-service@8.46.2': + resolution: {integrity: sha512-PULOLZ9iqwI7hXcmL4fVfIsBi6AN9YxRc0frbvmg8f+4hQAjQ5GYNKK0DIArNo+rOKmR/iBYwkpBmnIwin4wBg==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/scope-manager@8.46.2': + resolution: {integrity: sha512-LF4b/NmGvdWEHD2H4MsHD8ny6JpiVNDzrSZr3CsckEgCbAGZbYM4Cqxvi9L+WqDMT+51Ozy7lt2M+d0JLEuBqA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@typescript-eslint/tsconfig-utils@8.46.2': + resolution: {integrity: sha512-a7QH6fw4S57+F5y2FIxxSDyi5M4UfGF+Jl1bCGd7+L4KsaUY80GsiF/t0UoRFDHAguKlBaACWJRmdrc6Xfkkag==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/type-utils@8.46.2': + resolution: {integrity: sha512-HbPM4LbaAAt/DjxXaG9yiS9brOOz6fabal4uvUmaUYe6l3K1phQDMQKBRUrr06BQkxkvIZVVHttqiybM9nJsLA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + eslint: ^8.57.0 || ^9.0.0 + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/types@8.46.2': + resolution: {integrity: sha512-lNCWCbq7rpg7qDsQrd3D6NyWYu+gkTENkG5IKYhUIcxSb59SQC/hEQ+MrG4sTgBVghTonNWq42bA/d4yYumldQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@typescript-eslint/typescript-estree@8.46.2': + resolution: {integrity: sha512-f7rW7LJ2b7Uh2EiQ+7sza6RDZnajbNbemn54Ob6fRwQbgcIn+GWfyuHDHRYgRoZu1P4AayVScrRW+YfbTvPQoQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/utils@8.46.2': + resolution: {integrity: sha512-sExxzucx0Tud5tE0XqR0lT0psBQvEpnpiul9XbGUB1QwpWJJAps1O/Z7hJxLGiZLBKMCutjTzDgmd1muEhBnVg==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + peerDependencies: + eslint: ^8.57.0 || ^9.0.0 + typescript: '>=4.8.4 <6.0.0' + + '@typescript-eslint/visitor-keys@8.46.2': + resolution: {integrity: sha512-tUFMXI4gxzzMXt4xpGJEsBsTox0XbNQ1y94EwlD/CuZwFcQP79xfQqMhau9HsRc/J0cAPA/HZt1dZPtGn9V/7w==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + '@unrs/resolver-binding-android-arm-eabi@1.11.1': + resolution: {integrity: sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw==} + cpu: [arm] + os: [android] + + '@unrs/resolver-binding-android-arm64@1.11.1': + resolution: {integrity: sha512-lCxkVtb4wp1v+EoN+HjIG9cIIzPkX5OtM03pQYkG+U5O/wL53LC4QbIeazgiKqluGeVEeBlZahHalCaBvU1a2g==} + cpu: [arm64] + os: [android] + + '@unrs/resolver-binding-darwin-arm64@1.11.1': + resolution: {integrity: sha512-gPVA1UjRu1Y/IsB/dQEsp2V1pm44Of6+LWvbLc9SDk1c2KhhDRDBUkQCYVWe6f26uJb3fOK8saWMgtX8IrMk3g==} + cpu: [arm64] + os: [darwin] + + '@unrs/resolver-binding-darwin-x64@1.11.1': + resolution: {integrity: sha512-cFzP7rWKd3lZaCsDze07QX1SC24lO8mPty9vdP+YVa3MGdVgPmFc59317b2ioXtgCMKGiCLxJ4HQs62oz6GfRQ==} + cpu: [x64] + os: [darwin] + + '@unrs/resolver-binding-freebsd-x64@1.11.1': + resolution: {integrity: sha512-fqtGgak3zX4DCB6PFpsH5+Kmt/8CIi4Bry4rb1ho6Av2QHTREM+47y282Uqiu3ZRF5IQioJQ5qWRV6jduA+iGw==} + cpu: [x64] + os: [freebsd] + + '@unrs/resolver-binding-linux-arm-gnueabihf@1.11.1': + resolution: {integrity: sha512-u92mvlcYtp9MRKmP+ZvMmtPN34+/3lMHlyMj7wXJDeXxuM0Vgzz0+PPJNsro1m3IZPYChIkn944wW8TYgGKFHw==} + cpu: [arm] + os: [linux] + + '@unrs/resolver-binding-linux-arm-musleabihf@1.11.1': + resolution: {integrity: sha512-cINaoY2z7LVCrfHkIcmvj7osTOtm6VVT16b5oQdS4beibX2SYBwgYLmqhBjA1t51CarSaBuX5YNsWLjsqfW5Cw==} + cpu: [arm] + os: [linux] + + '@unrs/resolver-binding-linux-arm64-gnu@1.11.1': + resolution: {integrity: sha512-34gw7PjDGB9JgePJEmhEqBhWvCiiWCuXsL9hYphDF7crW7UgI05gyBAi6MF58uGcMOiOqSJ2ybEeCvHcq0BCmQ==} + cpu: [arm64] + os: [linux] + + '@unrs/resolver-binding-linux-arm64-musl@1.11.1': + resolution: {integrity: sha512-RyMIx6Uf53hhOtJDIamSbTskA99sPHS96wxVE/bJtePJJtpdKGXO1wY90oRdXuYOGOTuqjT8ACccMc4K6QmT3w==} + cpu: [arm64] + os: [linux] + + '@unrs/resolver-binding-linux-ppc64-gnu@1.11.1': + resolution: {integrity: sha512-D8Vae74A4/a+mZH0FbOkFJL9DSK2R6TFPC9M+jCWYia/q2einCubX10pecpDiTmkJVUH+y8K3BZClycD8nCShA==} + cpu: [ppc64] + os: [linux] + + '@unrs/resolver-binding-linux-riscv64-gnu@1.11.1': + resolution: {integrity: sha512-frxL4OrzOWVVsOc96+V3aqTIQl1O2TjgExV4EKgRY09AJ9leZpEg8Ak9phadbuX0BA4k8U5qtvMSQQGGmaJqcQ==} + cpu: [riscv64] + os: [linux] + + '@unrs/resolver-binding-linux-riscv64-musl@1.11.1': + resolution: {integrity: sha512-mJ5vuDaIZ+l/acv01sHoXfpnyrNKOk/3aDoEdLO/Xtn9HuZlDD6jKxHlkN8ZhWyLJsRBxfv9GYM2utQ1SChKew==} + cpu: [riscv64] + os: [linux] + + '@unrs/resolver-binding-linux-s390x-gnu@1.11.1': + resolution: {integrity: sha512-kELo8ebBVtb9sA7rMe1Cph4QHreByhaZ2QEADd9NzIQsYNQpt9UkM9iqr2lhGr5afh885d/cB5QeTXSbZHTYPg==} + cpu: [s390x] + os: [linux] + + '@unrs/resolver-binding-linux-x64-gnu@1.11.1': + resolution: {integrity: sha512-C3ZAHugKgovV5YvAMsxhq0gtXuwESUKc5MhEtjBpLoHPLYM+iuwSj3lflFwK3DPm68660rZ7G8BMcwSro7hD5w==} + cpu: [x64] + os: [linux] + + '@unrs/resolver-binding-linux-x64-musl@1.11.1': + resolution: {integrity: sha512-rV0YSoyhK2nZ4vEswT/QwqzqQXw5I6CjoaYMOX0TqBlWhojUf8P94mvI7nuJTeaCkkds3QE4+zS8Ko+GdXuZtA==} + cpu: [x64] + os: [linux] + + '@unrs/resolver-binding-wasm32-wasi@1.11.1': + resolution: {integrity: sha512-5u4RkfxJm+Ng7IWgkzi3qrFOvLvQYnPBmjmZQ8+szTK/b31fQCnleNl1GgEt7nIsZRIf5PLhPwT0WM+q45x/UQ==} + engines: {node: '>=14.0.0'} + cpu: [wasm32] + + '@unrs/resolver-binding-win32-arm64-msvc@1.11.1': + resolution: {integrity: sha512-nRcz5Il4ln0kMhfL8S3hLkxI85BXs3o8EYoattsJNdsX4YUU89iOkVn7g0VHSRxFuVMdM4Q1jEpIId1Ihim/Uw==} + cpu: [arm64] + os: [win32] + + '@unrs/resolver-binding-win32-ia32-msvc@1.11.1': + resolution: {integrity: sha512-DCEI6t5i1NmAZp6pFonpD5m7i6aFrpofcp4LA2i8IIq60Jyo28hamKBxNrZcyOwVOZkgsRp9O2sXWBWP8MnvIQ==} + cpu: [ia32] + os: [win32] + + '@unrs/resolver-binding-win32-x64-msvc@1.11.1': + resolution: {integrity: sha512-lrW200hZdbfRtztbygyaq/6jP6AKE8qQN2KvPcJ+x7wiD038YtnYtZ82IMNJ69GJibV7bwL3y9FgK+5w/pYt6g==} + cpu: [x64] + os: [win32] + + '@vitest/expect@4.0.14': + resolution: {integrity: sha512-RHk63V3zvRiYOWAV0rGEBRO820ce17hz7cI2kDmEdfQsBjT2luEKB5tCOc91u1oSQoUOZkSv3ZyzkdkSLD7lKw==} + + '@vitest/mocker@4.0.14': + resolution: {integrity: sha512-RzS5NujlCzeRPF1MK7MXLiEFpkIXeMdQ+rN3Kk3tDI9j0mtbr7Nmuq67tpkOJQpgyClbOltCXMjLZicJHsH5Cg==} + peerDependencies: + msw: ^2.4.9 + vite: ^6.0.0 || ^7.0.0-0 + peerDependenciesMeta: + msw: + optional: true + vite: + optional: true + + '@vitest/pretty-format@4.0.14': + resolution: {integrity: sha512-SOYPgujB6TITcJxgd3wmsLl+wZv+fy3av2PpiPpsWPZ6J1ySUYfScfpIt2Yv56ShJXR2MOA6q2KjKHN4EpdyRQ==} + + '@vitest/runner@4.0.14': + resolution: {integrity: sha512-BsAIk3FAqxICqREbX8SetIteT8PiaUL/tgJjmhxJhCsigmzzH8xeadtp7LRnTpCVzvf0ib9BgAfKJHuhNllKLw==} + + '@vitest/snapshot@4.0.14': + resolution: {integrity: sha512-aQVBfT1PMzDSA16Y3Fp45a0q8nKexx6N5Amw3MX55BeTeZpoC08fGqEZqVmPcqN0ueZsuUQ9rriPMhZ3Mu19Ag==} + + '@vitest/spy@4.0.14': + resolution: {integrity: sha512-JmAZT1UtZooO0tpY3GRyiC/8W7dCs05UOq9rfsUUgEZEdq+DuHLmWhPsrTt0TiW7WYeL/hXpaE07AZ2RCk44hg==} + + '@vitest/utils@4.0.14': + resolution: {integrity: sha512-hLqXZKAWNg8pI+SQXyXxWCTOpA3MvsqcbVeNgSi8x/CSN2wi26dSzn1wrOhmCmFjEvN9p8/kLFRHa6PI8jHazw==} + + acorn-jsx@5.3.2: + resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} + peerDependencies: + acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 + + acorn@8.15.0: + resolution: {integrity: sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==} + engines: {node: '>=0.4.0'} + hasBin: true + + ajv@6.12.6: + resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} + + ansi-styles@4.3.0: + resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} + engines: {node: '>=8'} + + argparse@2.0.1: + resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} + + aria-hidden@1.2.6: + resolution: {integrity: sha512-ik3ZgC9dY/lYVVM++OISsaYDeg1tb0VtP5uL3ouh1koGOaUMDPpbFIei4JkFimWUFPn90sbMNMXQAIVOlnYKJA==} + engines: {node: '>=10'} + + aria-query@5.3.2: + resolution: {integrity: sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==} + engines: {node: '>= 0.4'} + + array-buffer-byte-length@1.0.2: + resolution: {integrity: sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==} + engines: {node: '>= 0.4'} + + array-includes@3.1.9: + resolution: {integrity: sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==} + engines: {node: '>= 0.4'} + + array.prototype.findlast@1.2.5: + resolution: {integrity: sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==} + engines: {node: '>= 0.4'} + + array.prototype.findlastindex@1.2.6: + resolution: {integrity: sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ==} + engines: {node: '>= 0.4'} + + array.prototype.flat@1.3.3: + resolution: {integrity: sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==} + engines: {node: '>= 0.4'} + + array.prototype.flatmap@1.3.3: + resolution: {integrity: sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==} + engines: {node: '>= 0.4'} + + array.prototype.tosorted@1.1.4: + resolution: {integrity: sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==} + engines: {node: '>= 0.4'} + + arraybuffer.prototype.slice@1.0.4: + resolution: {integrity: sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==} + engines: {node: '>= 0.4'} + + assertion-error@2.0.1: + resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} + engines: {node: '>=12'} + + ast-types-flow@0.0.8: + resolution: {integrity: sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==} + + async-function@1.0.0: + resolution: {integrity: sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==} + engines: {node: '>= 0.4'} + + asynckit@0.4.0: + resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} + + available-typed-arrays@1.0.7: + resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} + engines: {node: '>= 0.4'} + + axe-core@4.11.0: + resolution: {integrity: sha512-ilYanEU8vxxBexpJd8cWM4ElSQq4QctCLKih0TSfjIfCQTeyH/6zVrmIJfLPrKTKJRbiG+cfnZbQIjAlJmF1jQ==} + engines: {node: '>=4'} + + axios@1.12.2: + resolution: {integrity: sha512-vMJzPewAlRyOgxV2dU0Cuz2O8zzzx9VYtbJOaBgXFeLc4IV/Eg50n4LowmehOOR61S8ZMpc2K5Sa7g6A4jfkUw==} + + axobject-query@4.1.0: + resolution: {integrity: sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==} + engines: {node: '>= 0.4'} + + balanced-match@1.0.2: + resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + + brace-expansion@1.1.12: + resolution: {integrity: sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==} + + brace-expansion@2.0.2: + resolution: {integrity: sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==} + + braces@3.0.3: + resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} + engines: {node: '>=8'} + + call-bind-apply-helpers@1.0.2: + resolution: {integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==} + engines: {node: '>= 0.4'} + + call-bind@1.0.8: + resolution: {integrity: sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==} + engines: {node: '>= 0.4'} + + call-bound@1.0.4: + resolution: {integrity: sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==} + engines: {node: '>= 0.4'} + + callsites@3.1.0: + resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} + engines: {node: '>=6'} + + caniuse-lite@1.0.30001751: + resolution: {integrity: sha512-A0QJhug0Ly64Ii3eIqHu5X51ebln3k4yTUkY1j8drqpWHVreg/VLijN48cZ1bYPiqOQuqpkIKnzr/Ul8V+p6Cw==} + + chai@6.2.1: + resolution: {integrity: sha512-p4Z49OGG5W/WBCPSS/dH3jQ73kD6tiMmUM+bckNK6Jr5JHMG3k9bg/BvKR8lKmtVBKmOiuVaV2ws8s9oSbwysg==} + engines: {node: '>=18'} + + chalk@4.1.2: + resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} + engines: {node: '>=10'} + + class-variance-authority@0.7.1: + resolution: {integrity: sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg==} + + client-only@0.0.1: + resolution: {integrity: sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==} + + clsx@2.1.1: + resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} + engines: {node: '>=6'} + + color-convert@2.0.1: + resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} + engines: {node: '>=7.0.0'} + + color-name@1.1.4: + resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + + combined-stream@1.0.8: + resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} + engines: {node: '>= 0.8'} + + concat-map@0.0.1: + resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + + core-util-is@1.0.3: + resolution: {integrity: sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==} + + cross-spawn@7.0.6: + resolution: {integrity: sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==} + engines: {node: '>= 8'} + + csstype@3.1.3: + resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} + + damerau-levenshtein@1.0.8: + resolution: {integrity: sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==} + + data-view-buffer@1.0.2: + resolution: {integrity: sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==} + engines: {node: '>= 0.4'} + + data-view-byte-length@1.0.2: + resolution: {integrity: sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==} + engines: {node: '>= 0.4'} + + data-view-byte-offset@1.0.1: + resolution: {integrity: sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==} + engines: {node: '>= 0.4'} + + date-fns-jalali@4.1.0-0: + resolution: {integrity: sha512-hTIP/z+t+qKwBDcmmsnmjWTduxCg+5KfdqWQvb2X/8C9+knYY6epN/pfxdDuyVlSVeFz0sM5eEfwIUQ70U4ckg==} + + date-fns@4.1.0: + resolution: {integrity: sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==} + + debug@3.2.7: + resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + + debug@4.4.3: + resolution: {integrity: sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + + deep-is@0.1.4: + resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} + + define-data-property@1.1.4: + resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} + engines: {node: '>= 0.4'} + + define-properties@1.2.1: + resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} + engines: {node: '>= 0.4'} + + delayed-stream@1.0.0: + resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} + engines: {node: '>=0.4.0'} + + detect-libc@2.1.2: + resolution: {integrity: sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==} + engines: {node: '>=8'} + + detect-node-es@1.1.0: + resolution: {integrity: sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==} + + doctrine@2.1.0: + resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==} + engines: {node: '>=0.10.0'} + + dunder-proto@1.0.1: + resolution: {integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==} + engines: {node: '>= 0.4'} + + embla-carousel-react@8.6.0: + resolution: {integrity: sha512-0/PjqU7geVmo6F734pmPqpyHqiM99olvyecY7zdweCw+6tKEXnrE90pBiBbMMU8s5tICemzpQ3hi5EpxzGW+JA==} + peerDependencies: + react: ^16.8.0 || ^17.0.1 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + + embla-carousel-reactive-utils@8.6.0: + resolution: {integrity: sha512-fMVUDUEx0/uIEDM0Mz3dHznDhfX+znCCDCeIophYb1QGVM7YThSWX+wz11zlYwWFOr74b4QLGg0hrGPJeG2s4A==} + peerDependencies: + embla-carousel: 8.6.0 + + embla-carousel@8.6.0: + resolution: {integrity: sha512-SjWyZBHJPbqxHOzckOfo8lHisEaJWmwd23XppYFYVh10bU66/Pn5tkVkbkCMZVdbUE5eTCI2nD8OyIP4Z+uwkA==} + + emoji-picker-react@4.14.2: + resolution: {integrity: sha512-tQfuZ5pqZnaqh/aGtgjp5iFqEKC2SoFOSs7YxbnSignstlIwNRS/Peh/Y1pFSMoODwS1DH4i0tdnod5/B/LwBw==} + engines: {node: '>=10'} + peerDependencies: + react: '>=16' + + emoji-regex@9.2.2: + resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} + + enhanced-resolve@5.18.3: + resolution: {integrity: sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==} + engines: {node: '>=10.13.0'} + + es-abstract@1.24.0: + resolution: {integrity: sha512-WSzPgsdLtTcQwm4CROfS5ju2Wa1QQcVeT37jFjYzdFz1r9ahadC8B8/a4qxJxM+09F18iumCdRmlr96ZYkQvEg==} + engines: {node: '>= 0.4'} + + es-define-property@1.0.1: + resolution: {integrity: sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==} + engines: {node: '>= 0.4'} + + es-errors@1.3.0: + resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} + engines: {node: '>= 0.4'} + + es-iterator-helpers@1.2.1: + resolution: {integrity: sha512-uDn+FE1yrDzyC0pCo961B2IHbdM8y/ACZsKD4dG6WqrjV53BADjwa7D+1aom2rsNVfLyDgU/eigvlJGJ08OQ4w==} + engines: {node: '>= 0.4'} + + es-module-lexer@1.7.0: + resolution: {integrity: sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==} + + es-object-atoms@1.1.1: + resolution: {integrity: sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==} + engines: {node: '>= 0.4'} + + es-set-tostringtag@2.1.0: + resolution: {integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==} + engines: {node: '>= 0.4'} + + es-shim-unscopables@1.1.0: + resolution: {integrity: sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==} + engines: {node: '>= 0.4'} + + es-to-primitive@1.3.0: + resolution: {integrity: sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==} + engines: {node: '>= 0.4'} + + esbuild@0.25.12: + resolution: {integrity: sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==} + engines: {node: '>=18'} + hasBin: true + + escape-string-regexp@4.0.0: + resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} + engines: {node: '>=10'} + + eslint-config-next@15.5.4: + resolution: {integrity: sha512-BzgVVuT3kfJes8i2GHenC1SRJ+W3BTML11lAOYFOOPzrk2xp66jBOAGEFRw+3LkYCln5UzvFsLhojrshb5Zfaw==} + peerDependencies: + eslint: ^7.23.0 || ^8.0.0 || ^9.0.0 + typescript: '>=3.3.1' + peerDependenciesMeta: + typescript: + optional: true + + eslint-config-prettier@10.1.8: + resolution: {integrity: sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==} + hasBin: true + peerDependencies: + eslint: '>=7.0.0' + + eslint-import-resolver-node@0.3.9: + resolution: {integrity: sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==} + + eslint-import-resolver-typescript@3.10.1: + resolution: {integrity: sha512-A1rHYb06zjMGAxdLSkN2fXPBwuSaQ0iO5M/hdyS0Ajj1VBaRp0sPD3dn1FhME3c/JluGFbwSxyCfqdSbtQLAHQ==} + engines: {node: ^14.18.0 || >=16.0.0} + peerDependencies: + eslint: '*' + eslint-plugin-import: '*' + eslint-plugin-import-x: '*' + peerDependenciesMeta: + eslint-plugin-import: + optional: true + eslint-plugin-import-x: + optional: true + + eslint-module-utils@2.12.1: + resolution: {integrity: sha512-L8jSWTze7K2mTg0vos/RuLRS5soomksDPoJLXIslC7c8Wmut3bx7CPpJijDcBZtxQ5lrbUdM+s0OlNbz0DCDNw==} + engines: {node: '>=4'} + peerDependencies: + '@typescript-eslint/parser': '*' + eslint: '*' + eslint-import-resolver-node: '*' + eslint-import-resolver-typescript: '*' + eslint-import-resolver-webpack: '*' + peerDependenciesMeta: + '@typescript-eslint/parser': + optional: true + eslint: + optional: true + eslint-import-resolver-node: + optional: true + eslint-import-resolver-typescript: + optional: true + eslint-import-resolver-webpack: + optional: true + + eslint-plugin-import@2.32.0: + resolution: {integrity: sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA==} + engines: {node: '>=4'} + peerDependencies: + '@typescript-eslint/parser': '*' + eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9 + peerDependenciesMeta: + '@typescript-eslint/parser': + optional: true + + eslint-plugin-jsx-a11y@6.10.2: + resolution: {integrity: sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q==} + engines: {node: '>=4.0'} + peerDependencies: + eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9 + + eslint-plugin-prettier@5.5.4: + resolution: {integrity: sha512-swNtI95SToIz05YINMA6Ox5R057IMAmWZ26GqPxusAp1TZzj+IdY9tXNWWD3vkF/wEqydCONcwjTFpxybBqZsg==} + engines: {node: ^14.18.0 || >=16.0.0} + peerDependencies: + '@types/eslint': '>=8.0.0' + eslint: '>=8.0.0' + eslint-config-prettier: '>= 7.0.0 <10.0.0 || >=10.1.0' + prettier: '>=3.0.0' + peerDependenciesMeta: + '@types/eslint': + optional: true + eslint-config-prettier: + optional: true + + eslint-plugin-react-hooks@5.2.0: + resolution: {integrity: sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg==} + engines: {node: '>=10'} + peerDependencies: + eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 + + eslint-plugin-react@7.37.5: + resolution: {integrity: sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA==} + engines: {node: '>=4'} + peerDependencies: + eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7 + + eslint-scope@8.4.0: + resolution: {integrity: sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + eslint-visitor-keys@3.4.3: + resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + + eslint-visitor-keys@4.2.1: + resolution: {integrity: sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + eslint@9.38.0: + resolution: {integrity: sha512-t5aPOpmtJcZcz5UJyY2GbvpDlsK5E8JqRqoKtfiKE3cNh437KIqfJr3A3AKf5k64NPx6d0G3dno6XDY05PqPtw==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + hasBin: true + peerDependencies: + jiti: '*' + peerDependenciesMeta: + jiti: + optional: true + + espree@10.4.0: + resolution: {integrity: sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + + esquery@1.6.0: + resolution: {integrity: sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==} + engines: {node: '>=0.10'} + + esrecurse@4.3.0: + resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} + engines: {node: '>=4.0'} + + estraverse@5.3.0: + resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} + engines: {node: '>=4.0'} + + estree-walker@3.0.3: + resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} + + esutils@2.0.3: + resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} + engines: {node: '>=0.10.0'} + + exifr@7.1.3: + resolution: {integrity: sha512-g/aje2noHivrRSLbAUtBPWFbxKdKhgj/xr1vATDdUXPOFYJlQ62Ft0oy+72V6XLIpDJfHs6gXLbBLAolqOXYRw==} + + expect-type@1.2.2: + resolution: {integrity: sha512-JhFGDVJ7tmDJItKhYgJCGLOWjuK9vPxiXoUFLwLDc99NlmklilbiQJwoctZtt13+xMw91MCk/REan6MWHqDjyA==} + engines: {node: '>=12.0.0'} + + fast-deep-equal@3.1.3: + resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} + + fast-diff@1.3.0: + resolution: {integrity: sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==} + + fast-glob@3.3.1: + resolution: {integrity: sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==} + engines: {node: '>=8.6.0'} + + fast-glob@3.3.3: + resolution: {integrity: sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==} + engines: {node: '>=8.6.0'} + + fast-json-stable-stringify@2.1.0: + resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} + + fast-levenshtein@2.0.6: + resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} + + fastq@1.19.1: + resolution: {integrity: sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==} + + fdir@6.5.0: + resolution: {integrity: sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==} + engines: {node: '>=12.0.0'} + peerDependencies: + picomatch: ^3 || ^4 + peerDependenciesMeta: + picomatch: + optional: true + + file-entry-cache@8.0.0: + resolution: {integrity: sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==} + engines: {node: '>=16.0.0'} + + fill-range@7.1.1: + resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} + engines: {node: '>=8'} + + find-up@5.0.0: + resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} + engines: {node: '>=10'} + + flairup@1.0.0: + resolution: {integrity: sha512-IKlE+pNvL2R+kVL1kEhUYqRxVqeFnjiIvHWDMLFXNaqyUdFXQM2wte44EfMYJNHkW16X991t2Zg8apKkhv7OBA==} + + flat-cache@4.0.1: + resolution: {integrity: sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==} + engines: {node: '>=16'} + + flatted@3.3.3: + resolution: {integrity: sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==} + + follow-redirects@1.15.11: + resolution: {integrity: sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ==} + engines: {node: '>=4.0'} + peerDependencies: + debug: '*' + peerDependenciesMeta: + debug: + optional: true + + for-each@0.3.5: + resolution: {integrity: sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==} + engines: {node: '>= 0.4'} + + form-data@4.0.4: + resolution: {integrity: sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==} + engines: {node: '>= 6'} + + framer-motion@12.23.24: + resolution: {integrity: sha512-HMi5HRoRCTou+3fb3h9oTLyJGBxHfW+HnNE25tAXOvVx/IvwMHK0cx7IR4a2ZU6sh3IX1Z+4ts32PcYBOqka8w==} + peerDependencies: + '@emotion/is-prop-valid': '*' + react: ^18.0.0 || ^19.0.0 + react-dom: ^18.0.0 || ^19.0.0 + peerDependenciesMeta: + '@emotion/is-prop-valid': + optional: true + react: + optional: true + react-dom: + optional: true + + fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + + function-bind@1.1.2: + resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + + function.prototype.name@1.1.8: + resolution: {integrity: sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==} + engines: {node: '>= 0.4'} + + functions-have-names@1.2.3: + resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==} + + generator-function@2.0.1: + resolution: {integrity: sha512-SFdFmIJi+ybC0vjlHN0ZGVGHc3lgE0DxPAT0djjVg+kjOnSqclqmj0KQ7ykTOLP6YxoqOvuAODGdcHJn+43q3g==} + engines: {node: '>= 0.4'} + + get-intrinsic@1.3.0: + resolution: {integrity: sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==} + engines: {node: '>= 0.4'} + + get-nonce@1.0.1: + resolution: {integrity: sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==} + engines: {node: '>=6'} + + get-proto@1.0.1: + resolution: {integrity: sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==} + engines: {node: '>= 0.4'} + + get-symbol-description@1.1.0: + resolution: {integrity: sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==} + engines: {node: '>= 0.4'} + + get-tsconfig@4.13.0: + resolution: {integrity: sha512-1VKTZJCwBrvbd+Wn3AOgQP/2Av+TfTCOlE4AcRJE72W1ksZXbAx8PPBR9RzgTeSPzlPMHrbANMH3LbltH73wxQ==} + + glob-parent@5.1.2: + resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} + engines: {node: '>= 6'} + + glob-parent@6.0.2: + resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} + engines: {node: '>=10.13.0'} + + globals@14.0.0: + resolution: {integrity: sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==} + engines: {node: '>=18'} + + globalthis@1.0.4: + resolution: {integrity: sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==} + engines: {node: '>= 0.4'} + + gopd@1.2.0: + resolution: {integrity: sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==} + engines: {node: '>= 0.4'} + + graceful-fs@4.2.11: + resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} + + graphemer@1.4.0: + resolution: {integrity: sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==} + + has-bigints@1.1.0: + resolution: {integrity: sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==} + engines: {node: '>= 0.4'} + + has-flag@4.0.0: + resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} + engines: {node: '>=8'} + + has-property-descriptors@1.0.2: + resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} + + has-proto@1.2.0: + resolution: {integrity: sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==} + engines: {node: '>= 0.4'} + + has-symbols@1.1.0: + resolution: {integrity: sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==} + engines: {node: '>= 0.4'} + + has-tostringtag@1.0.2: + resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} + engines: {node: '>= 0.4'} + + hasown@2.0.2: + resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} + engines: {node: '>= 0.4'} + + heic-to@1.3.0: + resolution: {integrity: sha512-ptlhtnvVjhUu0aXhmp1Ic0iD7dFRGVGplwXu17odLogd1O+zFSvmMRMKKzAobYqMpkVXwbA1AlY+ZN49O3FJkQ==} + + html-to-image@1.11.13: + resolution: {integrity: sha512-cuOPoI7WApyhBElTTb9oqsawRvZ0rHhaHwghRLlTuffoD1B2aDemlCruLeZrUIIdvG7gs9xeELEPm6PhuASqrg==} + + ignore@5.3.2: + resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} + engines: {node: '>= 4'} + + ignore@7.0.5: + resolution: {integrity: sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==} + engines: {node: '>= 4'} + + immediate@3.0.6: + resolution: {integrity: sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ==} + + import-fresh@3.3.1: + resolution: {integrity: sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==} + engines: {node: '>=6'} + + imurmurhash@0.1.4: + resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} + engines: {node: '>=0.8.19'} + + inherits@2.0.4: + resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + + internal-slot@1.1.0: + resolution: {integrity: sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==} + engines: {node: '>= 0.4'} + + is-array-buffer@3.0.5: + resolution: {integrity: sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==} + engines: {node: '>= 0.4'} + + is-async-function@2.1.1: + resolution: {integrity: sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==} + engines: {node: '>= 0.4'} + + is-bigint@1.1.0: + resolution: {integrity: sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==} + engines: {node: '>= 0.4'} + + is-boolean-object@1.2.2: + resolution: {integrity: sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==} + engines: {node: '>= 0.4'} + + is-bun-module@2.0.0: + resolution: {integrity: sha512-gNCGbnnnnFAUGKeZ9PdbyeGYJqewpmc2aKHUEMO5nQPWU9lOmv7jcmQIv+qHD8fXW6W7qfuCwX4rY9LNRjXrkQ==} + + is-callable@1.2.7: + resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} + engines: {node: '>= 0.4'} + + is-core-module@2.16.1: + resolution: {integrity: sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==} + engines: {node: '>= 0.4'} + + is-data-view@1.0.2: + resolution: {integrity: sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==} + engines: {node: '>= 0.4'} + + is-date-object@1.1.0: + resolution: {integrity: sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==} + engines: {node: '>= 0.4'} + + is-extglob@2.1.1: + resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} + engines: {node: '>=0.10.0'} + + is-finalizationregistry@1.1.1: + resolution: {integrity: sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==} + engines: {node: '>= 0.4'} + + is-generator-function@1.1.2: + resolution: {integrity: sha512-upqt1SkGkODW9tsGNG5mtXTXtECizwtS2kA161M+gJPc1xdb/Ax629af6YrTwcOeQHbewrPNlE5Dx7kzvXTizA==} + engines: {node: '>= 0.4'} + + is-glob@4.0.3: + resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} + engines: {node: '>=0.10.0'} + + is-map@2.0.3: + resolution: {integrity: sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==} + engines: {node: '>= 0.4'} + + is-negative-zero@2.0.3: + resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} + engines: {node: '>= 0.4'} + + is-number-object@1.1.1: + resolution: {integrity: sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==} + engines: {node: '>= 0.4'} + + is-number@7.0.0: + resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} + engines: {node: '>=0.12.0'} + + is-regex@1.2.1: + resolution: {integrity: sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==} + engines: {node: '>= 0.4'} + + is-set@2.0.3: + resolution: {integrity: sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==} + engines: {node: '>= 0.4'} + + is-shared-array-buffer@1.0.4: + resolution: {integrity: sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==} + engines: {node: '>= 0.4'} + + is-string@1.1.1: + resolution: {integrity: sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==} + engines: {node: '>= 0.4'} + + is-symbol@1.1.1: + resolution: {integrity: sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==} + engines: {node: '>= 0.4'} + + is-typed-array@1.1.15: + resolution: {integrity: sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==} + engines: {node: '>= 0.4'} + + is-weakmap@2.0.2: + resolution: {integrity: sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==} + engines: {node: '>= 0.4'} + + is-weakref@1.1.1: + resolution: {integrity: sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==} + engines: {node: '>= 0.4'} + + is-weakset@2.0.4: + resolution: {integrity: sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==} + engines: {node: '>= 0.4'} + + isarray@1.0.0: + resolution: {integrity: sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==} + + isarray@2.0.5: + resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + + isexe@2.0.0: + resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} + + iterator.prototype@1.1.5: + resolution: {integrity: sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g==} + engines: {node: '>= 0.4'} + + jiti@2.6.1: + resolution: {integrity: sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==} + hasBin: true + + js-tokens@4.0.0: + resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} + + js-yaml@4.1.0: + resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} + hasBin: true + + json-buffer@3.0.1: + resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} + + json-schema-traverse@0.4.1: + resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} + + json-stable-stringify-without-jsonify@1.0.1: + resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} + + json5@1.0.2: + resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==} + hasBin: true + + jsx-ast-utils@3.3.5: + resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==} + engines: {node: '>=4.0'} + + jszip@3.10.1: + resolution: {integrity: sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g==} + + keyv@4.5.4: + resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} + + language-subtag-registry@0.3.23: + resolution: {integrity: sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ==} + + language-tags@1.0.9: + resolution: {integrity: sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA==} + engines: {node: '>=0.10'} + + levn@0.4.1: + resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} + engines: {node: '>= 0.8.0'} + + lie@3.3.0: + resolution: {integrity: sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==} + + lightningcss-android-arm64@1.30.2: + resolution: {integrity: sha512-BH9sEdOCahSgmkVhBLeU7Hc9DWeZ1Eb6wNS6Da8igvUwAe0sqROHddIlvU06q3WyXVEOYDZ6ykBZQnjTbmo4+A==} + engines: {node: '>= 12.0.0'} + cpu: [arm64] + os: [android] + + lightningcss-darwin-arm64@1.30.2: + resolution: {integrity: sha512-ylTcDJBN3Hp21TdhRT5zBOIi73P6/W0qwvlFEk22fkdXchtNTOU4Qc37SkzV+EKYxLouZ6M4LG9NfZ1qkhhBWA==} + engines: {node: '>= 12.0.0'} + cpu: [arm64] + os: [darwin] + + lightningcss-darwin-x64@1.30.2: + resolution: {integrity: sha512-oBZgKchomuDYxr7ilwLcyms6BCyLn0z8J0+ZZmfpjwg9fRVZIR5/GMXd7r9RH94iDhld3UmSjBM6nXWM2TfZTQ==} + engines: {node: '>= 12.0.0'} + cpu: [x64] + os: [darwin] + + lightningcss-freebsd-x64@1.30.2: + resolution: {integrity: sha512-c2bH6xTrf4BDpK8MoGG4Bd6zAMZDAXS569UxCAGcA7IKbHNMlhGQ89eRmvpIUGfKWNVdbhSbkQaWhEoMGmGslA==} + engines: {node: '>= 12.0.0'} + cpu: [x64] + os: [freebsd] + + lightningcss-linux-arm-gnueabihf@1.30.2: + resolution: {integrity: sha512-eVdpxh4wYcm0PofJIZVuYuLiqBIakQ9uFZmipf6LF/HRj5Bgm0eb3qL/mr1smyXIS1twwOxNWndd8z0E374hiA==} + engines: {node: '>= 12.0.0'} + cpu: [arm] + os: [linux] + + lightningcss-linux-arm64-gnu@1.30.2: + resolution: {integrity: sha512-UK65WJAbwIJbiBFXpxrbTNArtfuznvxAJw4Q2ZGlU8kPeDIWEX1dg3rn2veBVUylA2Ezg89ktszWbaQnxD/e3A==} + engines: {node: '>= 12.0.0'} + cpu: [arm64] + os: [linux] + + lightningcss-linux-arm64-musl@1.30.2: + resolution: {integrity: sha512-5Vh9dGeblpTxWHpOx8iauV02popZDsCYMPIgiuw97OJ5uaDsL86cnqSFs5LZkG3ghHoX5isLgWzMs+eD1YzrnA==} + engines: {node: '>= 12.0.0'} + cpu: [arm64] + os: [linux] + + lightningcss-linux-x64-gnu@1.30.2: + resolution: {integrity: sha512-Cfd46gdmj1vQ+lR6VRTTadNHu6ALuw2pKR9lYq4FnhvgBc4zWY1EtZcAc6EffShbb1MFrIPfLDXD6Xprbnni4w==} + engines: {node: '>= 12.0.0'} + cpu: [x64] + os: [linux] + + lightningcss-linux-x64-musl@1.30.2: + resolution: {integrity: sha512-XJaLUUFXb6/QG2lGIW6aIk6jKdtjtcffUT0NKvIqhSBY3hh9Ch+1LCeH80dR9q9LBjG3ewbDjnumefsLsP6aiA==} + engines: {node: '>= 12.0.0'} + cpu: [x64] + os: [linux] + + lightningcss-win32-arm64-msvc@1.30.2: + resolution: {integrity: sha512-FZn+vaj7zLv//D/192WFFVA0RgHawIcHqLX9xuWiQt7P0PtdFEVaxgF9rjM/IRYHQXNnk61/H/gb2Ei+kUQ4xQ==} + engines: {node: '>= 12.0.0'} + cpu: [arm64] + os: [win32] + + lightningcss-win32-x64-msvc@1.30.2: + resolution: {integrity: sha512-5g1yc73p+iAkid5phb4oVFMB45417DkRevRbt/El/gKXJk4jid+vPFF/AXbxn05Aky8PapwzZrdJShv5C0avjw==} + engines: {node: '>= 12.0.0'} + cpu: [x64] + os: [win32] + + lightningcss@1.30.2: + resolution: {integrity: sha512-utfs7Pr5uJyyvDETitgsaqSyjCb2qNRAtuqUeWIAKztsOYdcACf2KtARYXg2pSvhkt+9NfoaNY7fxjl6nuMjIQ==} + engines: {node: '>= 12.0.0'} + + locate-path@6.0.0: + resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} + engines: {node: '>=10'} + + lodash.merge@4.6.2: + resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} + + loose-envify@1.4.0: + resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} + hasBin: true + + lottie-react@2.4.1: + resolution: {integrity: sha512-LQrH7jlkigIIv++wIyrOYFLHSKQpEY4zehPicL9bQsrt1rnoKRYCYgpCUe5maqylNtacy58/sQDZTkwMcTRxZw==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + + lottie-web@5.13.0: + resolution: {integrity: sha512-+gfBXl6sxXMPe8tKQm7qzLnUy5DUPJPKIyRHwtpCpyUEYjHYRJC/5gjUvdkuO2c3JllrPtHXH5UJJK8LRYl5yQ==} + + lucide-react@0.546.0: + resolution: {integrity: sha512-Z94u6fKT43lKeYHiVyvyR8fT7pwCzDu7RyMPpTvh054+xahSgj4HFQ+NmflvzdXsoAjYGdCguGaFKYuvq0ThCQ==} + peerDependencies: + react: ^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0 + + magic-string@0.30.21: + resolution: {integrity: sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==} + + math-intrinsics@1.1.0: + resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==} + engines: {node: '>= 0.4'} + + merge2@1.4.1: + resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} + engines: {node: '>= 8'} + + micromatch@4.0.8: + resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} + engines: {node: '>=8.6'} + + mime-db@1.52.0: + resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} + engines: {node: '>= 0.6'} + + mime-types@2.1.35: + resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} + engines: {node: '>= 0.6'} + + minimatch@3.1.2: + resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + + minimatch@9.0.5: + resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} + engines: {node: '>=16 || 14 >=14.17'} + + minimist@1.2.8: + resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + + motion-dom@12.23.23: + resolution: {integrity: sha512-n5yolOs0TQQBRUFImrRfs/+6X4p3Q4n1dUEqt/H58Vx7OW6RF+foWEgmTVDhIWJIMXOuNNL0apKH2S16en9eiA==} + + motion-utils@12.23.6: + resolution: {integrity: sha512-eAWoPgr4eFEOFfg2WjIsMoqJTW6Z8MTUCgn/GZ3VRpClWBdnbjryiA3ZSNLyxCTmCQx4RmYX6jX1iWHbenUPNQ==} + + ms@2.1.3: + resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + + nanoid@3.3.11: + resolution: {integrity: sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==} + engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + hasBin: true + + napi-postinstall@0.3.4: + resolution: {integrity: sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ==} + engines: {node: ^12.20.0 || ^14.18.0 || >=16.0.0} + hasBin: true + + natural-compare@1.4.0: + resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} + + next@15.5.7: + resolution: {integrity: sha512-+t2/0jIJ48kUpGKkdlhgkv+zPTEOoXyr60qXe68eB/pl3CMJaLeIGjzp5D6Oqt25hCBiBTt8wEeeAzfJvUKnPQ==} + engines: {node: ^18.18.0 || ^19.8.0 || >= 20.0.0} + hasBin: true + peerDependencies: + '@opentelemetry/api': ^1.1.0 + '@playwright/test': ^1.51.1 + babel-plugin-react-compiler: '*' + react: ^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0 + react-dom: ^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0 + sass: ^1.3.0 + peerDependenciesMeta: + '@opentelemetry/api': + optional: true + '@playwright/test': + optional: true + babel-plugin-react-compiler: + optional: true + sass: + optional: true + + object-assign@4.1.1: + resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} + engines: {node: '>=0.10.0'} + + object-inspect@1.13.4: + resolution: {integrity: sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==} + engines: {node: '>= 0.4'} + + object-keys@1.1.1: + resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==} + engines: {node: '>= 0.4'} + + object.assign@4.1.7: + resolution: {integrity: sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==} + engines: {node: '>= 0.4'} + + object.entries@1.1.9: + resolution: {integrity: sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw==} + engines: {node: '>= 0.4'} + + object.fromentries@2.0.8: + resolution: {integrity: sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==} + engines: {node: '>= 0.4'} + + object.groupby@1.0.3: + resolution: {integrity: sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==} + engines: {node: '>= 0.4'} + + object.values@1.2.1: + resolution: {integrity: sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==} + engines: {node: '>= 0.4'} + + obug@2.1.1: + resolution: {integrity: sha512-uTqF9MuPraAQ+IsnPf366RG4cP9RtUi7MLO1N3KEc+wb0a6yKpeL0lmk2IB1jY5KHPAlTc6T/JRdC/YqxHNwkQ==} + + optionator@0.9.4: + resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} + engines: {node: '>= 0.8.0'} + + own-keys@1.0.1: + resolution: {integrity: sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==} + engines: {node: '>= 0.4'} + + p-limit@3.1.0: + resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} + engines: {node: '>=10'} + + p-locate@5.0.0: + resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} + engines: {node: '>=10'} + + pako@1.0.11: + resolution: {integrity: sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==} + + parent-module@1.0.1: + resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} + engines: {node: '>=6'} + + path-exists@4.0.0: + resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} + engines: {node: '>=8'} + + path-key@3.1.1: + resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} + engines: {node: '>=8'} + + path-parse@1.0.7: + resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} + + pathe@2.0.3: + resolution: {integrity: sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==} + + picocolors@1.1.1: + resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==} + + picomatch@2.3.1: + resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} + engines: {node: '>=8.6'} + + picomatch@4.0.3: + resolution: {integrity: sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==} + engines: {node: '>=12'} + + possible-typed-array-names@1.1.0: + resolution: {integrity: sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==} + engines: {node: '>= 0.4'} + + postcss@8.4.31: + resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==} + engines: {node: ^10 || ^12 || >=14} + + postcss@8.5.6: + resolution: {integrity: sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==} + engines: {node: ^10 || ^12 || >=14} + + prelude-ls@1.2.1: + resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} + engines: {node: '>= 0.8.0'} + + prettier-linter-helpers@1.0.0: + resolution: {integrity: sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==} + engines: {node: '>=6.0.0'} + + prettier-plugin-tailwindcss@0.7.1: + resolution: {integrity: sha512-Bzv1LZcuiR1Sk02iJTS1QzlFNp/o5l2p3xkopwOrbPmtMeh3fK9rVW5M3neBQzHq+kGKj/4LGQMTNcTH4NGPtQ==} + engines: {node: '>=20.19'} + peerDependencies: + '@ianvs/prettier-plugin-sort-imports': '*' + '@prettier/plugin-hermes': '*' + '@prettier/plugin-oxc': '*' + '@prettier/plugin-pug': '*' + '@shopify/prettier-plugin-liquid': '*' + '@trivago/prettier-plugin-sort-imports': '*' + '@zackad/prettier-plugin-twig': '*' + prettier: ^3.0 + prettier-plugin-astro: '*' + prettier-plugin-css-order: '*' + prettier-plugin-jsdoc: '*' + prettier-plugin-marko: '*' + prettier-plugin-multiline-arrays: '*' + prettier-plugin-organize-attributes: '*' + prettier-plugin-organize-imports: '*' + prettier-plugin-sort-imports: '*' + prettier-plugin-svelte: '*' + peerDependenciesMeta: + '@ianvs/prettier-plugin-sort-imports': + optional: true + '@prettier/plugin-hermes': + optional: true + '@prettier/plugin-oxc': + optional: true + '@prettier/plugin-pug': + optional: true + '@shopify/prettier-plugin-liquid': + optional: true + '@trivago/prettier-plugin-sort-imports': + optional: true + '@zackad/prettier-plugin-twig': + optional: true + prettier-plugin-astro: + optional: true + prettier-plugin-css-order: + optional: true + prettier-plugin-jsdoc: + optional: true + prettier-plugin-marko: + optional: true + prettier-plugin-multiline-arrays: + optional: true + prettier-plugin-organize-attributes: + optional: true + prettier-plugin-organize-imports: + optional: true + prettier-plugin-sort-imports: + optional: true + prettier-plugin-svelte: + optional: true + + prettier@3.6.2: + resolution: {integrity: sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ==} + engines: {node: '>=14'} + hasBin: true + + process-nextick-args@2.0.1: + resolution: {integrity: sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==} + + prop-types@15.8.1: + resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} + + proxy-from-env@1.1.0: + resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==} + + punycode@2.3.1: + resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} + engines: {node: '>=6'} + + qr.js@0.0.0: + resolution: {integrity: sha512-c4iYnWb+k2E+vYpRimHqSu575b1/wKl4XFeJGpFmrJQz5I88v9aY2czh7s0w36srfCM1sXgC/xpoJz5dJfq+OQ==} + + queue-microtask@1.2.3: + resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} + + react-day-picker@9.11.1: + resolution: {integrity: sha512-l3ub6o8NlchqIjPKrRFUCkTUEq6KwemQlfv3XZzzwpUeGwmDJ+0u0Upmt38hJyd7D/vn2dQoOoLV/qAp0o3uUw==} + engines: {node: '>=18'} + peerDependencies: + react: '>=16.8.0' + + react-dom@19.1.2: + resolution: {integrity: sha512-dEoydsCp50i7kS1xHOmPXq4zQYoGWedUsvqv9H6zdif2r7yLHygyfP9qou71TulRN0d6ng9EbRVsQhSqfUc19g==} + peerDependencies: + react: ^19.1.2 + + react-is@16.13.1: + resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} + + react-qr-code@2.0.18: + resolution: {integrity: sha512-v1Jqz7urLMhkO6jkgJuBYhnqvXagzceg3qJUWayuCK/c6LTIonpWbwxR1f1APGd4xrW/QcQEovNrAojbUz65Tg==} + peerDependencies: + react: '*' + + react-remove-scroll-bar@2.3.8: + resolution: {integrity: sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': '*' + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + peerDependenciesMeta: + '@types/react': + optional: true + + react-remove-scroll@2.7.1: + resolution: {integrity: sha512-HpMh8+oahmIdOuS5aFKKY6Pyog+FNaZV/XyJOq7b4YFwsFHe5yYfdbIalI4k3vU2nSDql7YskmUseHsRrJqIPA==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': '*' + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + react-style-singleton@2.2.3: + resolution: {integrity: sha512-b6jSvxvVnyptAiLjbkWLE/lOnR4lfTtDAl+eUC7RZy+QQWc6wRzIV2CE6xBuMmDxc2qIihtDCZD5NPOFl7fRBQ==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': '*' + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + react@19.1.2: + resolution: {integrity: sha512-MdWVitvLbQULD+4DP8GYjZUrepGW7d+GQkNVqJEzNxE+e9WIa4egVFE/RDfVb1u9u/Jw7dNMmPB4IqxzbFYJ0w==} + engines: {node: '>=0.10.0'} + + readable-stream@2.3.8: + resolution: {integrity: sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==} + + reflect.getprototypeof@1.0.10: + resolution: {integrity: sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==} + engines: {node: '>= 0.4'} + + regexp.prototype.flags@1.5.4: + resolution: {integrity: sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==} + engines: {node: '>= 0.4'} + + resolve-from@4.0.0: + resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} + engines: {node: '>=4'} + + resolve-pkg-maps@1.0.0: + resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} + + resolve@1.22.11: + resolution: {integrity: sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==} + engines: {node: '>= 0.4'} + hasBin: true + + resolve@2.0.0-next.5: + resolution: {integrity: sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==} + hasBin: true + + reusify@1.1.0: + resolution: {integrity: sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==} + engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + + rollup@4.53.3: + resolution: {integrity: sha512-w8GmOxZfBmKknvdXU1sdM9NHcoQejwF/4mNgj2JuEEdRaHwwF12K7e9eXn1nLZ07ad+du76mkVsyeb2rKGllsA==} + engines: {node: '>=18.0.0', npm: '>=8.0.0'} + hasBin: true + + run-parallel@1.2.0: + resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + + safe-array-concat@1.1.3: + resolution: {integrity: sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==} + engines: {node: '>=0.4'} + + safe-buffer@5.1.2: + resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==} + + safe-push-apply@1.0.0: + resolution: {integrity: sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==} + engines: {node: '>= 0.4'} + + safe-regex-test@1.1.0: + resolution: {integrity: sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==} + engines: {node: '>= 0.4'} + + scheduler@0.26.0: + resolution: {integrity: sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA==} + + semver@6.3.1: + resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} + hasBin: true + + semver@7.7.3: + resolution: {integrity: sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==} + engines: {node: '>=10'} + hasBin: true + + set-function-length@1.2.2: + resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} + engines: {node: '>= 0.4'} + + set-function-name@2.0.2: + resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==} + engines: {node: '>= 0.4'} + + set-proto@1.0.0: + resolution: {integrity: sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==} + engines: {node: '>= 0.4'} + + setimmediate@1.0.5: + resolution: {integrity: sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==} + + sharp@0.34.4: + resolution: {integrity: sha512-FUH39xp3SBPnxWvd5iib1X8XY7J0K0X7d93sie9CJg2PO8/7gmg89Nve6OjItK53/MlAushNNxteBYfM6DEuoA==} + engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + + shebang-command@2.0.0: + resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} + engines: {node: '>=8'} + + shebang-regex@3.0.0: + resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} + engines: {node: '>=8'} + + side-channel-list@1.0.0: + resolution: {integrity: sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==} + engines: {node: '>= 0.4'} + + side-channel-map@1.0.1: + resolution: {integrity: sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==} + engines: {node: '>= 0.4'} + + side-channel-weakmap@1.0.2: + resolution: {integrity: sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==} + engines: {node: '>= 0.4'} + + side-channel@1.1.0: + resolution: {integrity: sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==} + engines: {node: '>= 0.4'} + + siginfo@2.0.0: + resolution: {integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==} + + source-map-js@1.2.1: + resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} + engines: {node: '>=0.10.0'} + + stable-hash@0.0.5: + resolution: {integrity: sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA==} + + stackback@0.0.2: + resolution: {integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==} + + std-env@3.10.0: + resolution: {integrity: sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==} + + stop-iteration-iterator@1.1.0: + resolution: {integrity: sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==} + engines: {node: '>= 0.4'} + + string.prototype.includes@2.0.1: + resolution: {integrity: sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg==} + engines: {node: '>= 0.4'} + + string.prototype.matchall@4.0.12: + resolution: {integrity: sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA==} + engines: {node: '>= 0.4'} + + string.prototype.repeat@1.0.0: + resolution: {integrity: sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==} + + string.prototype.trim@1.2.10: + resolution: {integrity: sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==} + engines: {node: '>= 0.4'} + + string.prototype.trimend@1.0.9: + resolution: {integrity: sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==} + engines: {node: '>= 0.4'} + + string.prototype.trimstart@1.0.8: + resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==} + engines: {node: '>= 0.4'} + + string_decoder@1.1.1: + resolution: {integrity: sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==} + + strip-bom@3.0.0: + resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==} + engines: {node: '>=4'} + + strip-json-comments@3.1.1: + resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} + engines: {node: '>=8'} + + styled-jsx@5.1.6: + resolution: {integrity: sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA==} + engines: {node: '>= 12.0.0'} + peerDependencies: + '@babel/core': '*' + babel-plugin-macros: '*' + react: '>= 16.8.0 || 17.x.x || ^18.0.0-0 || ^19.0.0-0' + peerDependenciesMeta: + '@babel/core': + optional: true + babel-plugin-macros: + optional: true + + supports-color@7.2.0: + resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} + engines: {node: '>=8'} + + supports-preserve-symlinks-flag@1.0.0: + resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} + engines: {node: '>= 0.4'} + + swiper@12.0.3: + resolution: {integrity: sha512-BHd6U1VPEIksrXlyXjMmRWO0onmdNPaTAFduzqR3pgjvi7KfmUCAm/0cj49u2D7B0zNjMw02TSeXfinC1hDCXg==} + engines: {node: '>= 4.7.0'} + + synckit@0.11.11: + resolution: {integrity: sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==} + engines: {node: ^14.18.0 || >=16.0.0} + + tabbable@6.3.0: + resolution: {integrity: sha512-EIHvdY5bPLuWForiR/AN2Bxngzpuwn1is4asboytXtpTgsArc+WmSJKVLlhdh71u7jFcryDqB2A8lQvj78MkyQ==} + + tailwind-merge@3.3.1: + resolution: {integrity: sha512-gBXpgUm/3rp1lMZZrM/w7D8GKqshif0zAymAhbCyIt8KMe+0v9DQ7cdYLR4FHH/cKpdTXb+A/tKKU3eolfsI+g==} + + tailwindcss@4.1.16: + resolution: {integrity: sha512-pONL5awpaQX4LN5eiv7moSiSPd/DLDzKVRJz8Q9PgzmAdd1R4307GQS2ZpfiN7ZmekdQrfhZZiSE5jkLR4WNaA==} + + tapable@2.3.0: + resolution: {integrity: sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg==} + engines: {node: '>=6'} + + tinybench@2.9.0: + resolution: {integrity: sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==} + + tinyexec@0.3.2: + resolution: {integrity: sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==} + + tinyglobby@0.2.15: + resolution: {integrity: sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==} + engines: {node: '>=12.0.0'} + + tinyrainbow@3.0.3: + resolution: {integrity: sha512-PSkbLUoxOFRzJYjjxHJt9xro7D+iilgMX/C9lawzVuYiIdcihh9DXmVibBe8lmcFrRi/VzlPjBxbN7rH24q8/Q==} + engines: {node: '>=14.0.0'} + + to-regex-range@5.0.1: + resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} + engines: {node: '>=8.0'} + + ts-api-utils@2.1.0: + resolution: {integrity: sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==} + engines: {node: '>=18.12'} + peerDependencies: + typescript: '>=4.8.4' + + tsconfig-paths@3.15.0: + resolution: {integrity: sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==} + + tslib@2.8.1: + resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} + + tw-animate-css@1.4.0: + resolution: {integrity: sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ==} + + type-check@0.4.0: + resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} + engines: {node: '>= 0.8.0'} + + typed-array-buffer@1.0.3: + resolution: {integrity: sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==} + engines: {node: '>= 0.4'} + + typed-array-byte-length@1.0.3: + resolution: {integrity: sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==} + engines: {node: '>= 0.4'} + + typed-array-byte-offset@1.0.4: + resolution: {integrity: sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==} + engines: {node: '>= 0.4'} + + typed-array-length@1.0.7: + resolution: {integrity: sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==} + engines: {node: '>= 0.4'} + + typescript@5.9.3: + resolution: {integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==} + engines: {node: '>=14.17'} + hasBin: true + + unbox-primitive@1.1.0: + resolution: {integrity: sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==} + engines: {node: '>= 0.4'} + + undici-types@6.21.0: + resolution: {integrity: sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==} + + unrs-resolver@1.11.1: + resolution: {integrity: sha512-bSjt9pjaEBnNiGgc9rUiHGKv5l4/TGzDmYw3RhnkJGtLhbnnA/5qJj7x3dNDCRx/PJxu774LlH8lCOlB4hEfKg==} + + uri-js@4.4.1: + resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} + + use-callback-ref@1.3.3: + resolution: {integrity: sha512-jQL3lRnocaFtu3V00JToYz/4QkNWswxijDaCVNZRiRTO3HQDLsdu1ZtmIUvV4yPp+rvWm5j0y0TG/S61cuijTg==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': '*' + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + use-sidecar@1.1.3: + resolution: {integrity: sha512-Fedw0aZvkhynoPYlA5WXrMCAMm+nSWdZt6lzJQ7Ok8S6Q+VsHmHpRWndVRJ8Be0ZbkfPc5LRYH+5XrzXcEeLRQ==} + engines: {node: '>=10'} + peerDependencies: + '@types/react': '*' + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + use-sync-external-store@1.6.0: + resolution: {integrity: sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==} + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + + util-deprecate@1.0.2: + resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} + + vaul@1.1.2: + resolution: {integrity: sha512-ZFkClGpWyI2WUQjdLJ/BaGuV6AVQiJ3uELGk3OYtP+B6yCO7Cmn9vPFXVJkRaGkOJu3m8bQMgtyzNHixULceQA==} + peerDependencies: + react: ^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc + + vite@7.2.4: + resolution: {integrity: sha512-NL8jTlbo0Tn4dUEXEsUg8KeyG/Lkmc4Fnzb8JXN/Ykm9G4HNImjtABMJgkQoVjOBN/j2WAwDTRytdqJbZsah7w==} + engines: {node: ^20.19.0 || >=22.12.0} + hasBin: true + peerDependencies: + '@types/node': ^20.19.0 || >=22.12.0 + jiti: '>=1.21.0' + less: ^4.0.0 + lightningcss: ^1.21.0 + sass: ^1.70.0 + sass-embedded: ^1.70.0 + stylus: '>=0.54.8' + sugarss: ^5.0.0 + terser: ^5.16.0 + tsx: ^4.8.1 + yaml: ^2.4.2 + peerDependenciesMeta: + '@types/node': + optional: true + jiti: + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + sass-embedded: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true + tsx: + optional: true + yaml: + optional: true + + vitest@4.0.14: + resolution: {integrity: sha512-d9B2J9Cm9dN9+6nxMnnNJKJCtcyKfnHj15N6YNJfaFHRLua/d3sRKU9RuKmO9mB0XdFtUizlxfz/VPbd3OxGhw==} + engines: {node: ^20.0.0 || ^22.0.0 || >=24.0.0} + hasBin: true + peerDependencies: + '@edge-runtime/vm': '*' + '@opentelemetry/api': ^1.9.0 + '@types/node': ^20.0.0 || ^22.0.0 || >=24.0.0 + '@vitest/browser-playwright': 4.0.14 + '@vitest/browser-preview': 4.0.14 + '@vitest/browser-webdriverio': 4.0.14 + '@vitest/ui': 4.0.14 + happy-dom: '*' + jsdom: '*' + peerDependenciesMeta: + '@edge-runtime/vm': + optional: true + '@opentelemetry/api': + optional: true + '@types/node': + optional: true + '@vitest/browser-playwright': + optional: true + '@vitest/browser-preview': + optional: true + '@vitest/browser-webdriverio': + optional: true + '@vitest/ui': + optional: true + happy-dom: + optional: true + jsdom: + optional: true + + which-boxed-primitive@1.1.1: + resolution: {integrity: sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==} + engines: {node: '>= 0.4'} + + which-builtin-type@1.2.1: + resolution: {integrity: sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==} + engines: {node: '>= 0.4'} + + which-collection@1.0.2: + resolution: {integrity: sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==} + engines: {node: '>= 0.4'} + + which-typed-array@1.1.19: + resolution: {integrity: sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw==} + engines: {node: '>= 0.4'} + + which@2.0.2: + resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} + engines: {node: '>= 8'} + hasBin: true + + why-is-node-running@2.3.0: + resolution: {integrity: sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==} + engines: {node: '>=8'} + hasBin: true + + word-wrap@1.2.5: + resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} + engines: {node: '>=0.10.0'} + + yocto-queue@0.1.0: + resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} + engines: {node: '>=10'} + + zustand@5.0.8: + resolution: {integrity: sha512-gyPKpIaxY9XcO2vSMrLbiER7QMAMGOQZVRdJ6Zi782jkbzZygq5GI9nG8g+sMgitRtndwaBSl7uiqC49o1SSiw==} + engines: {node: '>=12.20.0'} + peerDependencies: + '@types/react': '>=18.0.0' + immer: '>=9.0.6' + react: '>=18.0.0' + use-sync-external-store: '>=1.2.0' + peerDependenciesMeta: + '@types/react': + optional: true + immer: + optional: true + react: + optional: true + use-sync-external-store: + optional: true + +snapshots: + + '@alloc/quick-lru@5.2.0': {} + + '@date-fns/tz@1.4.1': {} + + '@emnapi/core@1.6.0': + dependencies: + '@emnapi/wasi-threads': 1.1.0 + tslib: 2.8.1 + optional: true + + '@emnapi/runtime@1.6.0': + dependencies: + tslib: 2.8.1 + optional: true + + '@emnapi/wasi-threads@1.1.0': + dependencies: + tslib: 2.8.1 + optional: true + + '@esbuild/aix-ppc64@0.25.12': + optional: true + + '@esbuild/android-arm64@0.25.12': + optional: true + + '@esbuild/android-arm@0.25.12': + optional: true + + '@esbuild/android-x64@0.25.12': + optional: true + + '@esbuild/darwin-arm64@0.25.12': + optional: true + + '@esbuild/darwin-x64@0.25.12': + optional: true + + '@esbuild/freebsd-arm64@0.25.12': + optional: true + + '@esbuild/freebsd-x64@0.25.12': + optional: true + + '@esbuild/linux-arm64@0.25.12': + optional: true + + '@esbuild/linux-arm@0.25.12': + optional: true + + '@esbuild/linux-ia32@0.25.12': + optional: true + + '@esbuild/linux-loong64@0.25.12': + optional: true + + '@esbuild/linux-mips64el@0.25.12': + optional: true + + '@esbuild/linux-ppc64@0.25.12': + optional: true + + '@esbuild/linux-riscv64@0.25.12': + optional: true + + '@esbuild/linux-s390x@0.25.12': + optional: true + + '@esbuild/linux-x64@0.25.12': + optional: true + + '@esbuild/netbsd-arm64@0.25.12': + optional: true + + '@esbuild/netbsd-x64@0.25.12': + optional: true + + '@esbuild/openbsd-arm64@0.25.12': + optional: true + + '@esbuild/openbsd-x64@0.25.12': + optional: true + + '@esbuild/openharmony-arm64@0.25.12': + optional: true + + '@esbuild/sunos-x64@0.25.12': + optional: true + + '@esbuild/win32-arm64@0.25.12': + optional: true + + '@esbuild/win32-ia32@0.25.12': + optional: true + + '@esbuild/win32-x64@0.25.12': + optional: true + + '@eslint-community/eslint-utils@4.9.0(eslint@9.38.0(jiti@2.6.1))': + dependencies: + eslint: 9.38.0(jiti@2.6.1) + eslint-visitor-keys: 3.4.3 + + '@eslint-community/regexpp@4.12.2': {} + + '@eslint/config-array@0.21.1': + dependencies: + '@eslint/object-schema': 2.1.7 + debug: 4.4.3 + minimatch: 3.1.2 + transitivePeerDependencies: + - supports-color + + '@eslint/config-helpers@0.4.1': + dependencies: + '@eslint/core': 0.16.0 + + '@eslint/core@0.16.0': + dependencies: + '@types/json-schema': 7.0.15 + + '@eslint/eslintrc@3.3.1': + dependencies: + ajv: 6.12.6 + debug: 4.4.3 + espree: 10.4.0 + globals: 14.0.0 + ignore: 5.3.2 + import-fresh: 3.3.1 + js-yaml: 4.1.0 + minimatch: 3.1.2 + strip-json-comments: 3.1.1 + transitivePeerDependencies: + - supports-color + + '@eslint/js@9.38.0': {} + + '@eslint/object-schema@2.1.7': {} + + '@eslint/plugin-kit@0.4.0': + dependencies: + '@eslint/core': 0.16.0 + levn: 0.4.1 + + '@floating-ui/core@1.7.3': + dependencies: + '@floating-ui/utils': 0.2.10 + + '@floating-ui/dom@1.7.4': + dependencies: + '@floating-ui/core': 1.7.3 + '@floating-ui/utils': 0.2.10 + + '@floating-ui/react-dom@2.1.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@floating-ui/dom': 1.7.4 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + '@floating-ui/react@0.26.28(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@floating-ui/react-dom': 2.1.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@floating-ui/utils': 0.2.10 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + tabbable: 6.3.0 + + '@floating-ui/utils@0.2.10': {} + + '@headlessui/react@2.2.9(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@floating-ui/react': 0.26.28(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@react-aria/focus': 3.21.2(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@react-aria/interactions': 3.25.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@tanstack/react-virtual': 3.13.12(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + use-sync-external-store: 1.6.0(react@19.1.2) + + '@humanfs/core@0.19.1': {} + + '@humanfs/node@0.16.7': + dependencies: + '@humanfs/core': 0.19.1 + '@humanwhocodes/retry': 0.4.3 + + '@humanwhocodes/module-importer@1.0.1': {} + + '@humanwhocodes/retry@0.4.3': {} + + '@img/colour@1.0.0': + optional: true + + '@img/sharp-darwin-arm64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-darwin-arm64': 1.2.3 + optional: true + + '@img/sharp-darwin-x64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-darwin-x64': 1.2.3 + optional: true + + '@img/sharp-libvips-darwin-arm64@1.2.3': + optional: true + + '@img/sharp-libvips-darwin-x64@1.2.3': + optional: true + + '@img/sharp-libvips-linux-arm64@1.2.3': + optional: true + + '@img/sharp-libvips-linux-arm@1.2.3': + optional: true + + '@img/sharp-libvips-linux-ppc64@1.2.3': + optional: true + + '@img/sharp-libvips-linux-s390x@1.2.3': + optional: true + + '@img/sharp-libvips-linux-x64@1.2.3': + optional: true + + '@img/sharp-libvips-linuxmusl-arm64@1.2.3': + optional: true + + '@img/sharp-libvips-linuxmusl-x64@1.2.3': + optional: true + + '@img/sharp-linux-arm64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linux-arm64': 1.2.3 + optional: true + + '@img/sharp-linux-arm@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linux-arm': 1.2.3 + optional: true + + '@img/sharp-linux-ppc64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linux-ppc64': 1.2.3 + optional: true + + '@img/sharp-linux-s390x@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linux-s390x': 1.2.3 + optional: true + + '@img/sharp-linux-x64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linux-x64': 1.2.3 + optional: true + + '@img/sharp-linuxmusl-arm64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linuxmusl-arm64': 1.2.3 + optional: true + + '@img/sharp-linuxmusl-x64@0.34.4': + optionalDependencies: + '@img/sharp-libvips-linuxmusl-x64': 1.2.3 + optional: true + + '@img/sharp-wasm32@0.34.4': + dependencies: + '@emnapi/runtime': 1.6.0 + optional: true + + '@img/sharp-win32-arm64@0.34.4': + optional: true + + '@img/sharp-win32-ia32@0.34.4': + optional: true + + '@img/sharp-win32-x64@0.34.4': + optional: true + + '@jridgewell/gen-mapping@0.3.13': + dependencies: + '@jridgewell/sourcemap-codec': 1.5.5 + '@jridgewell/trace-mapping': 0.3.31 + + '@jridgewell/remapping@2.3.5': + dependencies: + '@jridgewell/gen-mapping': 0.3.13 + '@jridgewell/trace-mapping': 0.3.31 + + '@jridgewell/resolve-uri@3.1.2': {} + + '@jridgewell/sourcemap-codec@1.5.5': {} + + '@jridgewell/trace-mapping@0.3.31': + dependencies: + '@jridgewell/resolve-uri': 3.1.2 + '@jridgewell/sourcemap-codec': 1.5.5 + + '@napi-rs/wasm-runtime@0.2.12': + dependencies: + '@emnapi/core': 1.6.0 + '@emnapi/runtime': 1.6.0 + '@tybys/wasm-util': 0.10.1 + optional: true + + '@next/env@15.5.7': {} + + '@next/eslint-plugin-next@15.5.4': + dependencies: + fast-glob: 3.3.1 + + '@next/swc-darwin-arm64@15.5.7': + optional: true + + '@next/swc-darwin-x64@15.5.7': + optional: true + + '@next/swc-linux-arm64-gnu@15.5.7': + optional: true + + '@next/swc-linux-arm64-musl@15.5.7': + optional: true + + '@next/swc-linux-x64-gnu@15.5.7': + optional: true + + '@next/swc-linux-x64-musl@15.5.7': + optional: true + + '@next/swc-win32-arm64-msvc@15.5.7': + optional: true + + '@next/swc-win32-x64-msvc@15.5.7': + optional: true + + '@nodelib/fs.scandir@2.1.5': + dependencies: + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 + + '@nodelib/fs.stat@2.0.5': {} + + '@nodelib/fs.walk@1.2.8': + dependencies: + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.19.1 + + '@nolyfill/is-core-module@1.0.39': {} + + '@pkgr/core@0.2.9': {} + + '@radix-ui/primitive@1.1.3': {} + + '@radix-ui/react-alert-dialog@1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/primitive': 1.1.3 + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-context': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-dialog': 1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-slot': 1.2.3(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-arrow@1.1.7(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-compose-refs@1.1.2(@types/react@19.2.2)(react@19.1.2)': + dependencies: + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-context@1.1.2(@types/react@19.2.2)(react@19.1.2)': + dependencies: + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-dialog@1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/primitive': 1.1.3 + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-context': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-dismissable-layer': 1.1.11(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-focus-guards': 1.1.3(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-focus-scope': 1.1.7(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-id': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-portal': 1.1.9(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-presence': 1.1.5(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-slot': 1.2.3(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-controllable-state': 1.2.2(@types/react@19.2.2)(react@19.1.2) + aria-hidden: 1.2.6 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + react-remove-scroll: 2.7.1(@types/react@19.2.2)(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-dismissable-layer@1.1.11(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/primitive': 1.1.3 + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-use-callback-ref': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-escape-keydown': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-focus-guards@1.1.3(@types/react@19.2.2)(react@19.1.2)': + dependencies: + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-focus-scope@1.1.7(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-use-callback-ref': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-id@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-popover@1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/primitive': 1.1.3 + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-context': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-dismissable-layer': 1.1.11(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-focus-guards': 1.1.3(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-focus-scope': 1.1.7(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-id': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-popper': 1.2.8(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-portal': 1.1.9(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-presence': 1.1.5(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-slot': 1.2.3(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-controllable-state': 1.2.2(@types/react@19.2.2)(react@19.1.2) + aria-hidden: 1.2.6 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + react-remove-scroll: 2.7.1(@types/react@19.2.2)(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-popper@1.2.8(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@floating-ui/react-dom': 2.1.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-arrow': 1.1.7(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-context': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-use-callback-ref': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-rect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-size': 1.1.1(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/rect': 1.1.1 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-portal@1.1.9(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/react-primitive': 2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-presence@1.1.5(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-primitive@2.1.3(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@radix-ui/react-slot': 1.2.3(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + '@types/react-dom': 19.2.2(@types/react@19.2.2) + + '@radix-ui/react-slot@1.2.3(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-compose-refs': 1.1.2(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-callback-ref@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-controllable-state@1.2.2(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-use-effect-event': 0.0.2(@types/react@19.2.2)(react@19.1.2) + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-effect-event@0.0.2(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-escape-keydown@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-use-callback-ref': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-layout-effect@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-rect@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/rect': 1.1.1 + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/react-use-size@1.1.1(@types/react@19.2.2)(react@19.1.2)': + dependencies: + '@radix-ui/react-use-layout-effect': 1.1.1(@types/react@19.2.2)(react@19.1.2) + react: 19.1.2 + optionalDependencies: + '@types/react': 19.2.2 + + '@radix-ui/rect@1.1.1': {} + + '@react-aria/focus@3.21.2(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@react-aria/interactions': 3.25.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@react-aria/utils': 3.31.0(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@react-types/shared': 3.32.1(react@19.1.2) + '@swc/helpers': 0.5.15 + clsx: 2.1.1 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + '@react-aria/interactions@3.25.6(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@react-aria/ssr': 3.9.10(react@19.1.2) + '@react-aria/utils': 3.31.0(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + '@react-stately/flags': 3.1.2 + '@react-types/shared': 3.32.1(react@19.1.2) + '@swc/helpers': 0.5.15 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + '@react-aria/ssr@3.9.10(react@19.1.2)': + dependencies: + '@swc/helpers': 0.5.15 + react: 19.1.2 + + '@react-aria/utils@3.31.0(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@react-aria/ssr': 3.9.10(react@19.1.2) + '@react-stately/flags': 3.1.2 + '@react-stately/utils': 3.10.8(react@19.1.2) + '@react-types/shared': 3.32.1(react@19.1.2) + '@swc/helpers': 0.5.15 + clsx: 2.1.1 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + '@react-stately/flags@3.1.2': + dependencies: + '@swc/helpers': 0.5.15 + + '@react-stately/utils@3.10.8(react@19.1.2)': + dependencies: + '@swc/helpers': 0.5.15 + react: 19.1.2 + + '@react-types/shared@3.32.1(react@19.1.2)': + dependencies: + react: 19.1.2 + + '@rollup/rollup-android-arm-eabi@4.53.3': + optional: true + + '@rollup/rollup-android-arm64@4.53.3': + optional: true + + '@rollup/rollup-darwin-arm64@4.53.3': + optional: true + + '@rollup/rollup-darwin-x64@4.53.3': + optional: true + + '@rollup/rollup-freebsd-arm64@4.53.3': + optional: true + + '@rollup/rollup-freebsd-x64@4.53.3': + optional: true + + '@rollup/rollup-linux-arm-gnueabihf@4.53.3': + optional: true + + '@rollup/rollup-linux-arm-musleabihf@4.53.3': + optional: true + + '@rollup/rollup-linux-arm64-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-arm64-musl@4.53.3': + optional: true + + '@rollup/rollup-linux-loong64-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-ppc64-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-riscv64-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-riscv64-musl@4.53.3': + optional: true + + '@rollup/rollup-linux-s390x-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-x64-gnu@4.53.3': + optional: true + + '@rollup/rollup-linux-x64-musl@4.53.3': + optional: true + + '@rollup/rollup-openharmony-arm64@4.53.3': + optional: true + + '@rollup/rollup-win32-arm64-msvc@4.53.3': + optional: true + + '@rollup/rollup-win32-ia32-msvc@4.53.3': + optional: true + + '@rollup/rollup-win32-x64-gnu@4.53.3': + optional: true + + '@rollup/rollup-win32-x64-msvc@4.53.3': + optional: true + + '@rtsao/scc@1.1.0': {} + + '@rushstack/eslint-patch@1.14.1': {} + + '@standard-schema/spec@1.0.0': {} + + '@swc/helpers@0.5.15': + dependencies: + tslib: 2.8.1 + + '@tailwindcss/node@4.1.16': + dependencies: + '@jridgewell/remapping': 2.3.5 + enhanced-resolve: 5.18.3 + jiti: 2.6.1 + lightningcss: 1.30.2 + magic-string: 0.30.21 + source-map-js: 1.2.1 + tailwindcss: 4.1.16 + + '@tailwindcss/oxide-android-arm64@4.1.16': + optional: true + + '@tailwindcss/oxide-darwin-arm64@4.1.16': + optional: true + + '@tailwindcss/oxide-darwin-x64@4.1.16': + optional: true + + '@tailwindcss/oxide-freebsd-x64@4.1.16': + optional: true + + '@tailwindcss/oxide-linux-arm-gnueabihf@4.1.16': + optional: true + + '@tailwindcss/oxide-linux-arm64-gnu@4.1.16': + optional: true + + '@tailwindcss/oxide-linux-arm64-musl@4.1.16': + optional: true + + '@tailwindcss/oxide-linux-x64-gnu@4.1.16': + optional: true + + '@tailwindcss/oxide-linux-x64-musl@4.1.16': + optional: true + + '@tailwindcss/oxide-wasm32-wasi@4.1.16': + optional: true + + '@tailwindcss/oxide-win32-arm64-msvc@4.1.16': + optional: true + + '@tailwindcss/oxide-win32-x64-msvc@4.1.16': + optional: true + + '@tailwindcss/oxide@4.1.16': + optionalDependencies: + '@tailwindcss/oxide-android-arm64': 4.1.16 + '@tailwindcss/oxide-darwin-arm64': 4.1.16 + '@tailwindcss/oxide-darwin-x64': 4.1.16 + '@tailwindcss/oxide-freebsd-x64': 4.1.16 + '@tailwindcss/oxide-linux-arm-gnueabihf': 4.1.16 + '@tailwindcss/oxide-linux-arm64-gnu': 4.1.16 + '@tailwindcss/oxide-linux-arm64-musl': 4.1.16 + '@tailwindcss/oxide-linux-x64-gnu': 4.1.16 + '@tailwindcss/oxide-linux-x64-musl': 4.1.16 + '@tailwindcss/oxide-wasm32-wasi': 4.1.16 + '@tailwindcss/oxide-win32-arm64-msvc': 4.1.16 + '@tailwindcss/oxide-win32-x64-msvc': 4.1.16 + + '@tailwindcss/postcss@4.1.16': + dependencies: + '@alloc/quick-lru': 5.2.0 + '@tailwindcss/node': 4.1.16 + '@tailwindcss/oxide': 4.1.16 + postcss: 8.5.6 + tailwindcss: 4.1.16 + + '@tanstack/query-core@5.90.5': {} + + '@tanstack/react-query@5.90.5(react@19.1.2)': + dependencies: + '@tanstack/query-core': 5.90.5 + react: 19.1.2 + + '@tanstack/react-virtual@3.13.12(react-dom@19.1.2(react@19.1.2))(react@19.1.2)': + dependencies: + '@tanstack/virtual-core': 3.13.12 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + '@tanstack/virtual-core@3.13.12': {} + + '@tybys/wasm-util@0.10.1': + dependencies: + tslib: 2.8.1 + optional: true + + '@types/chai@5.2.3': + dependencies: + '@types/deep-eql': 4.0.2 + assertion-error: 2.0.1 + + '@types/deep-eql@4.0.2': {} + + '@types/estree@1.0.8': {} + + '@types/json-schema@7.0.15': {} + + '@types/json5@0.0.29': {} + + '@types/node@20.19.23': + dependencies: + undici-types: 6.21.0 + + '@types/react-dom@19.2.2(@types/react@19.2.2)': + dependencies: + '@types/react': 19.2.2 + + '@types/react@19.2.2': + dependencies: + csstype: 3.1.3 + + '@typescript-eslint/eslint-plugin@8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3)': + dependencies: + '@eslint-community/regexpp': 4.12.2 + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/type-utils': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + '@typescript-eslint/visitor-keys': 8.46.2 + eslint: 9.38.0(jiti@2.6.1) + graphemer: 1.4.0 + ignore: 7.0.5 + natural-compare: 1.4.0 + ts-api-utils: 2.1.0(typescript@5.9.3) + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3)': + dependencies: + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + '@typescript-eslint/visitor-keys': 8.46.2 + debug: 4.4.3 + eslint: 9.38.0(jiti@2.6.1) + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/project-service@8.46.2(typescript@5.9.3)': + dependencies: + '@typescript-eslint/tsconfig-utils': 8.46.2(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + debug: 4.4.3 + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/scope-manager@8.46.2': + dependencies: + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/visitor-keys': 8.46.2 + + '@typescript-eslint/tsconfig-utils@8.46.2(typescript@5.9.3)': + dependencies: + typescript: 5.9.3 + + '@typescript-eslint/type-utils@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3)': + dependencies: + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + '@typescript-eslint/utils': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + debug: 4.4.3 + eslint: 9.38.0(jiti@2.6.1) + ts-api-utils: 2.1.0(typescript@5.9.3) + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/types@8.46.2': {} + + '@typescript-eslint/typescript-estree@8.46.2(typescript@5.9.3)': + dependencies: + '@typescript-eslint/project-service': 8.46.2(typescript@5.9.3) + '@typescript-eslint/tsconfig-utils': 8.46.2(typescript@5.9.3) + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/visitor-keys': 8.46.2 + debug: 4.4.3 + fast-glob: 3.3.3 + is-glob: 4.0.3 + minimatch: 9.0.5 + semver: 7.7.3 + ts-api-utils: 2.1.0(typescript@5.9.3) + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/utils@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3)': + dependencies: + '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@2.6.1)) + '@typescript-eslint/scope-manager': 8.46.2 + '@typescript-eslint/types': 8.46.2 + '@typescript-eslint/typescript-estree': 8.46.2(typescript@5.9.3) + eslint: 9.38.0(jiti@2.6.1) + typescript: 5.9.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/visitor-keys@8.46.2': + dependencies: + '@typescript-eslint/types': 8.46.2 + eslint-visitor-keys: 4.2.1 + + '@unrs/resolver-binding-android-arm-eabi@1.11.1': + optional: true + + '@unrs/resolver-binding-android-arm64@1.11.1': + optional: true + + '@unrs/resolver-binding-darwin-arm64@1.11.1': + optional: true + + '@unrs/resolver-binding-darwin-x64@1.11.1': + optional: true + + '@unrs/resolver-binding-freebsd-x64@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-arm-gnueabihf@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-arm-musleabihf@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-arm64-gnu@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-arm64-musl@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-ppc64-gnu@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-riscv64-gnu@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-riscv64-musl@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-s390x-gnu@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-x64-gnu@1.11.1': + optional: true + + '@unrs/resolver-binding-linux-x64-musl@1.11.1': + optional: true + + '@unrs/resolver-binding-wasm32-wasi@1.11.1': + dependencies: + '@napi-rs/wasm-runtime': 0.2.12 + optional: true + + '@unrs/resolver-binding-win32-arm64-msvc@1.11.1': + optional: true + + '@unrs/resolver-binding-win32-ia32-msvc@1.11.1': + optional: true + + '@unrs/resolver-binding-win32-x64-msvc@1.11.1': + optional: true + + '@vitest/expect@4.0.14': + dependencies: + '@standard-schema/spec': 1.0.0 + '@types/chai': 5.2.3 + '@vitest/spy': 4.0.14 + '@vitest/utils': 4.0.14 + chai: 6.2.1 + tinyrainbow: 3.0.3 + + '@vitest/mocker@4.0.14(vite@7.2.4(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2))': + dependencies: + '@vitest/spy': 4.0.14 + estree-walker: 3.0.3 + magic-string: 0.30.21 + optionalDependencies: + vite: 7.2.4(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2) + + '@vitest/pretty-format@4.0.14': + dependencies: + tinyrainbow: 3.0.3 + + '@vitest/runner@4.0.14': + dependencies: + '@vitest/utils': 4.0.14 + pathe: 2.0.3 + + '@vitest/snapshot@4.0.14': + dependencies: + '@vitest/pretty-format': 4.0.14 + magic-string: 0.30.21 + pathe: 2.0.3 + + '@vitest/spy@4.0.14': {} + + '@vitest/utils@4.0.14': + dependencies: + '@vitest/pretty-format': 4.0.14 + tinyrainbow: 3.0.3 + + acorn-jsx@5.3.2(acorn@8.15.0): + dependencies: + acorn: 8.15.0 + + acorn@8.15.0: {} + + ajv@6.12.6: + dependencies: + fast-deep-equal: 3.1.3 + fast-json-stable-stringify: 2.1.0 + json-schema-traverse: 0.4.1 + uri-js: 4.4.1 + + ansi-styles@4.3.0: + dependencies: + color-convert: 2.0.1 + + argparse@2.0.1: {} + + aria-hidden@1.2.6: + dependencies: + tslib: 2.8.1 + + aria-query@5.3.2: {} + + array-buffer-byte-length@1.0.2: + dependencies: + call-bound: 1.0.4 + is-array-buffer: 3.0.5 + + array-includes@3.1.9: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-object-atoms: 1.1.1 + get-intrinsic: 1.3.0 + is-string: 1.1.1 + math-intrinsics: 1.1.0 + + array.prototype.findlast@1.2.5: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + es-shim-unscopables: 1.1.0 + + array.prototype.findlastindex@1.2.6: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + es-shim-unscopables: 1.1.0 + + array.prototype.flat@1.3.3: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-shim-unscopables: 1.1.0 + + array.prototype.flatmap@1.3.3: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-shim-unscopables: 1.1.0 + + array.prototype.tosorted@1.1.4: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-shim-unscopables: 1.1.0 + + arraybuffer.prototype.slice@1.0.4: + dependencies: + array-buffer-byte-length: 1.0.2 + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + is-array-buffer: 3.0.5 + + assertion-error@2.0.1: {} + + ast-types-flow@0.0.8: {} + + async-function@1.0.0: {} + + asynckit@0.4.0: {} + + available-typed-arrays@1.0.7: + dependencies: + possible-typed-array-names: 1.1.0 + + axe-core@4.11.0: {} + + axios@1.12.2: + dependencies: + follow-redirects: 1.15.11 + form-data: 4.0.4 + proxy-from-env: 1.1.0 + transitivePeerDependencies: + - debug + + axobject-query@4.1.0: {} + + balanced-match@1.0.2: {} + + brace-expansion@1.1.12: + dependencies: + balanced-match: 1.0.2 + concat-map: 0.0.1 + + brace-expansion@2.0.2: + dependencies: + balanced-match: 1.0.2 + + braces@3.0.3: + dependencies: + fill-range: 7.1.1 + + call-bind-apply-helpers@1.0.2: + dependencies: + es-errors: 1.3.0 + function-bind: 1.1.2 + + call-bind@1.0.8: + dependencies: + call-bind-apply-helpers: 1.0.2 + es-define-property: 1.0.1 + get-intrinsic: 1.3.0 + set-function-length: 1.2.2 + + call-bound@1.0.4: + dependencies: + call-bind-apply-helpers: 1.0.2 + get-intrinsic: 1.3.0 + + callsites@3.1.0: {} + + caniuse-lite@1.0.30001751: {} + + chai@6.2.1: {} + + chalk@4.1.2: + dependencies: + ansi-styles: 4.3.0 + supports-color: 7.2.0 + + class-variance-authority@0.7.1: + dependencies: + clsx: 2.1.1 + + client-only@0.0.1: {} + + clsx@2.1.1: {} + + color-convert@2.0.1: + dependencies: + color-name: 1.1.4 + + color-name@1.1.4: {} + + combined-stream@1.0.8: + dependencies: + delayed-stream: 1.0.0 + + concat-map@0.0.1: {} + + core-util-is@1.0.3: {} + + cross-spawn@7.0.6: + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + + csstype@3.1.3: {} + + damerau-levenshtein@1.0.8: {} + + data-view-buffer@1.0.2: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + is-data-view: 1.0.2 + + data-view-byte-length@1.0.2: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + is-data-view: 1.0.2 + + data-view-byte-offset@1.0.1: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + is-data-view: 1.0.2 + + date-fns-jalali@4.1.0-0: {} + + date-fns@4.1.0: {} + + debug@3.2.7: + dependencies: + ms: 2.1.3 + + debug@4.4.3: + dependencies: + ms: 2.1.3 + + deep-is@0.1.4: {} + + define-data-property@1.1.4: + dependencies: + es-define-property: 1.0.1 + es-errors: 1.3.0 + gopd: 1.2.0 + + define-properties@1.2.1: + dependencies: + define-data-property: 1.1.4 + has-property-descriptors: 1.0.2 + object-keys: 1.1.1 + + delayed-stream@1.0.0: {} + + detect-libc@2.1.2: {} + + detect-node-es@1.1.0: {} + + doctrine@2.1.0: + dependencies: + esutils: 2.0.3 + + dunder-proto@1.0.1: + dependencies: + call-bind-apply-helpers: 1.0.2 + es-errors: 1.3.0 + gopd: 1.2.0 + + embla-carousel-react@8.6.0(react@19.1.2): + dependencies: + embla-carousel: 8.6.0 + embla-carousel-reactive-utils: 8.6.0(embla-carousel@8.6.0) + react: 19.1.2 + + embla-carousel-reactive-utils@8.6.0(embla-carousel@8.6.0): + dependencies: + embla-carousel: 8.6.0 + + embla-carousel@8.6.0: {} + + emoji-picker-react@4.14.2(react@19.1.2): + dependencies: + flairup: 1.0.0 + react: 19.1.2 + + emoji-regex@9.2.2: {} + + enhanced-resolve@5.18.3: + dependencies: + graceful-fs: 4.2.11 + tapable: 2.3.0 + + es-abstract@1.24.0: + dependencies: + array-buffer-byte-length: 1.0.2 + arraybuffer.prototype.slice: 1.0.4 + available-typed-arrays: 1.0.7 + call-bind: 1.0.8 + call-bound: 1.0.4 + data-view-buffer: 1.0.2 + data-view-byte-length: 1.0.2 + data-view-byte-offset: 1.0.1 + es-define-property: 1.0.1 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + es-set-tostringtag: 2.1.0 + es-to-primitive: 1.3.0 + function.prototype.name: 1.1.8 + get-intrinsic: 1.3.0 + get-proto: 1.0.1 + get-symbol-description: 1.1.0 + globalthis: 1.0.4 + gopd: 1.2.0 + has-property-descriptors: 1.0.2 + has-proto: 1.2.0 + has-symbols: 1.1.0 + hasown: 2.0.2 + internal-slot: 1.1.0 + is-array-buffer: 3.0.5 + is-callable: 1.2.7 + is-data-view: 1.0.2 + is-negative-zero: 2.0.3 + is-regex: 1.2.1 + is-set: 2.0.3 + is-shared-array-buffer: 1.0.4 + is-string: 1.1.1 + is-typed-array: 1.1.15 + is-weakref: 1.1.1 + math-intrinsics: 1.1.0 + object-inspect: 1.13.4 + object-keys: 1.1.1 + object.assign: 4.1.7 + own-keys: 1.0.1 + regexp.prototype.flags: 1.5.4 + safe-array-concat: 1.1.3 + safe-push-apply: 1.0.0 + safe-regex-test: 1.1.0 + set-proto: 1.0.0 + stop-iteration-iterator: 1.1.0 + string.prototype.trim: 1.2.10 + string.prototype.trimend: 1.0.9 + string.prototype.trimstart: 1.0.8 + typed-array-buffer: 1.0.3 + typed-array-byte-length: 1.0.3 + typed-array-byte-offset: 1.0.4 + typed-array-length: 1.0.7 + unbox-primitive: 1.1.0 + which-typed-array: 1.1.19 + + es-define-property@1.0.1: {} + + es-errors@1.3.0: {} + + es-iterator-helpers@1.2.1: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-set-tostringtag: 2.1.0 + function-bind: 1.1.2 + get-intrinsic: 1.3.0 + globalthis: 1.0.4 + gopd: 1.2.0 + has-property-descriptors: 1.0.2 + has-proto: 1.2.0 + has-symbols: 1.1.0 + internal-slot: 1.1.0 + iterator.prototype: 1.1.5 + safe-array-concat: 1.1.3 + + es-module-lexer@1.7.0: {} + + es-object-atoms@1.1.1: + dependencies: + es-errors: 1.3.0 + + es-set-tostringtag@2.1.0: + dependencies: + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + has-tostringtag: 1.0.2 + hasown: 2.0.2 + + es-shim-unscopables@1.1.0: + dependencies: + hasown: 2.0.2 + + es-to-primitive@1.3.0: + dependencies: + is-callable: 1.2.7 + is-date-object: 1.1.0 + is-symbol: 1.1.1 + + esbuild@0.25.12: + optionalDependencies: + '@esbuild/aix-ppc64': 0.25.12 + '@esbuild/android-arm': 0.25.12 + '@esbuild/android-arm64': 0.25.12 + '@esbuild/android-x64': 0.25.12 + '@esbuild/darwin-arm64': 0.25.12 + '@esbuild/darwin-x64': 0.25.12 + '@esbuild/freebsd-arm64': 0.25.12 + '@esbuild/freebsd-x64': 0.25.12 + '@esbuild/linux-arm': 0.25.12 + '@esbuild/linux-arm64': 0.25.12 + '@esbuild/linux-ia32': 0.25.12 + '@esbuild/linux-loong64': 0.25.12 + '@esbuild/linux-mips64el': 0.25.12 + '@esbuild/linux-ppc64': 0.25.12 + '@esbuild/linux-riscv64': 0.25.12 + '@esbuild/linux-s390x': 0.25.12 + '@esbuild/linux-x64': 0.25.12 + '@esbuild/netbsd-arm64': 0.25.12 + '@esbuild/netbsd-x64': 0.25.12 + '@esbuild/openbsd-arm64': 0.25.12 + '@esbuild/openbsd-x64': 0.25.12 + '@esbuild/openharmony-arm64': 0.25.12 + '@esbuild/sunos-x64': 0.25.12 + '@esbuild/win32-arm64': 0.25.12 + '@esbuild/win32-ia32': 0.25.12 + '@esbuild/win32-x64': 0.25.12 + + escape-string-regexp@4.0.0: {} + + eslint-config-next@15.5.4(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3): + dependencies: + '@next/eslint-plugin-next': 15.5.4 + '@rushstack/eslint-patch': 1.14.1 + '@typescript-eslint/eslint-plugin': 8.46.2(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + eslint: 9.38.0(jiti@2.6.1) + eslint-import-resolver-node: 0.3.9 + eslint-import-resolver-typescript: 3.10.1(eslint-plugin-import@2.32.0)(eslint@9.38.0(jiti@2.6.1)) + eslint-plugin-import: 2.32.0(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.38.0(jiti@2.6.1)) + eslint-plugin-jsx-a11y: 6.10.2(eslint@9.38.0(jiti@2.6.1)) + eslint-plugin-react: 7.37.5(eslint@9.38.0(jiti@2.6.1)) + eslint-plugin-react-hooks: 5.2.0(eslint@9.38.0(jiti@2.6.1)) + optionalDependencies: + typescript: 5.9.3 + transitivePeerDependencies: + - eslint-import-resolver-webpack + - eslint-plugin-import-x + - supports-color + + eslint-config-prettier@10.1.8(eslint@9.38.0(jiti@2.6.1)): + dependencies: + eslint: 9.38.0(jiti@2.6.1) + + eslint-import-resolver-node@0.3.9: + dependencies: + debug: 3.2.7 + is-core-module: 2.16.1 + resolve: 1.22.11 + transitivePeerDependencies: + - supports-color + + eslint-import-resolver-typescript@3.10.1(eslint-plugin-import@2.32.0)(eslint@9.38.0(jiti@2.6.1)): + dependencies: + '@nolyfill/is-core-module': 1.0.39 + debug: 4.4.3 + eslint: 9.38.0(jiti@2.6.1) + get-tsconfig: 4.13.0 + is-bun-module: 2.0.0 + stable-hash: 0.0.5 + tinyglobby: 0.2.15 + unrs-resolver: 1.11.1 + optionalDependencies: + eslint-plugin-import: 2.32.0(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.38.0(jiti@2.6.1)) + transitivePeerDependencies: + - supports-color + + eslint-module-utils@2.12.1(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.10.1)(eslint@9.38.0(jiti@2.6.1)): + dependencies: + debug: 3.2.7 + optionalDependencies: + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + eslint: 9.38.0(jiti@2.6.1) + eslint-import-resolver-node: 0.3.9 + eslint-import-resolver-typescript: 3.10.1(eslint-plugin-import@2.32.0)(eslint@9.38.0(jiti@2.6.1)) + transitivePeerDependencies: + - supports-color + + eslint-plugin-import@2.32.0(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.38.0(jiti@2.6.1)): + dependencies: + '@rtsao/scc': 1.1.0 + array-includes: 3.1.9 + array.prototype.findlastindex: 1.2.6 + array.prototype.flat: 1.3.3 + array.prototype.flatmap: 1.3.3 + debug: 3.2.7 + doctrine: 2.1.0 + eslint: 9.38.0(jiti@2.6.1) + eslint-import-resolver-node: 0.3.9 + eslint-module-utils: 2.12.1(@typescript-eslint/parser@8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.10.1)(eslint@9.38.0(jiti@2.6.1)) + hasown: 2.0.2 + is-core-module: 2.16.1 + is-glob: 4.0.3 + minimatch: 3.1.2 + object.fromentries: 2.0.8 + object.groupby: 1.0.3 + object.values: 1.2.1 + semver: 6.3.1 + string.prototype.trimend: 1.0.9 + tsconfig-paths: 3.15.0 + optionalDependencies: + '@typescript-eslint/parser': 8.46.2(eslint@9.38.0(jiti@2.6.1))(typescript@5.9.3) + transitivePeerDependencies: + - eslint-import-resolver-typescript + - eslint-import-resolver-webpack + - supports-color + + eslint-plugin-jsx-a11y@6.10.2(eslint@9.38.0(jiti@2.6.1)): + dependencies: + aria-query: 5.3.2 + array-includes: 3.1.9 + array.prototype.flatmap: 1.3.3 + ast-types-flow: 0.0.8 + axe-core: 4.11.0 + axobject-query: 4.1.0 + damerau-levenshtein: 1.0.8 + emoji-regex: 9.2.2 + eslint: 9.38.0(jiti@2.6.1) + hasown: 2.0.2 + jsx-ast-utils: 3.3.5 + language-tags: 1.0.9 + minimatch: 3.1.2 + object.fromentries: 2.0.8 + safe-regex-test: 1.1.0 + string.prototype.includes: 2.0.1 + + eslint-plugin-prettier@5.5.4(eslint-config-prettier@10.1.8(eslint@9.38.0(jiti@2.6.1)))(eslint@9.38.0(jiti@2.6.1))(prettier@3.6.2): + dependencies: + eslint: 9.38.0(jiti@2.6.1) + prettier: 3.6.2 + prettier-linter-helpers: 1.0.0 + synckit: 0.11.11 + optionalDependencies: + eslint-config-prettier: 10.1.8(eslint@9.38.0(jiti@2.6.1)) + + eslint-plugin-react-hooks@5.2.0(eslint@9.38.0(jiti@2.6.1)): + dependencies: + eslint: 9.38.0(jiti@2.6.1) + + eslint-plugin-react@7.37.5(eslint@9.38.0(jiti@2.6.1)): + dependencies: + array-includes: 3.1.9 + array.prototype.findlast: 1.2.5 + array.prototype.flatmap: 1.3.3 + array.prototype.tosorted: 1.1.4 + doctrine: 2.1.0 + es-iterator-helpers: 1.2.1 + eslint: 9.38.0(jiti@2.6.1) + estraverse: 5.3.0 + hasown: 2.0.2 + jsx-ast-utils: 3.3.5 + minimatch: 3.1.2 + object.entries: 1.1.9 + object.fromentries: 2.0.8 + object.values: 1.2.1 + prop-types: 15.8.1 + resolve: 2.0.0-next.5 + semver: 6.3.1 + string.prototype.matchall: 4.0.12 + string.prototype.repeat: 1.0.0 + + eslint-scope@8.4.0: + dependencies: + esrecurse: 4.3.0 + estraverse: 5.3.0 + + eslint-visitor-keys@3.4.3: {} + + eslint-visitor-keys@4.2.1: {} + + eslint@9.38.0(jiti@2.6.1): + dependencies: + '@eslint-community/eslint-utils': 4.9.0(eslint@9.38.0(jiti@2.6.1)) + '@eslint-community/regexpp': 4.12.2 + '@eslint/config-array': 0.21.1 + '@eslint/config-helpers': 0.4.1 + '@eslint/core': 0.16.0 + '@eslint/eslintrc': 3.3.1 + '@eslint/js': 9.38.0 + '@eslint/plugin-kit': 0.4.0 + '@humanfs/node': 0.16.7 + '@humanwhocodes/module-importer': 1.0.1 + '@humanwhocodes/retry': 0.4.3 + '@types/estree': 1.0.8 + ajv: 6.12.6 + chalk: 4.1.2 + cross-spawn: 7.0.6 + debug: 4.4.3 + escape-string-regexp: 4.0.0 + eslint-scope: 8.4.0 + eslint-visitor-keys: 4.2.1 + espree: 10.4.0 + esquery: 1.6.0 + esutils: 2.0.3 + fast-deep-equal: 3.1.3 + file-entry-cache: 8.0.0 + find-up: 5.0.0 + glob-parent: 6.0.2 + ignore: 5.3.2 + imurmurhash: 0.1.4 + is-glob: 4.0.3 + json-stable-stringify-without-jsonify: 1.0.1 + lodash.merge: 4.6.2 + minimatch: 3.1.2 + natural-compare: 1.4.0 + optionator: 0.9.4 + optionalDependencies: + jiti: 2.6.1 + transitivePeerDependencies: + - supports-color + + espree@10.4.0: + dependencies: + acorn: 8.15.0 + acorn-jsx: 5.3.2(acorn@8.15.0) + eslint-visitor-keys: 4.2.1 + + esquery@1.6.0: + dependencies: + estraverse: 5.3.0 + + esrecurse@4.3.0: + dependencies: + estraverse: 5.3.0 + + estraverse@5.3.0: {} + + estree-walker@3.0.3: + dependencies: + '@types/estree': 1.0.8 + + esutils@2.0.3: {} + + exifr@7.1.3: {} + + expect-type@1.2.2: {} + + fast-deep-equal@3.1.3: {} + + fast-diff@1.3.0: {} + + fast-glob@3.3.1: + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.8 + + fast-glob@3.3.3: + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.8 + + fast-json-stable-stringify@2.1.0: {} + + fast-levenshtein@2.0.6: {} + + fastq@1.19.1: + dependencies: + reusify: 1.1.0 + + fdir@6.5.0(picomatch@4.0.3): + optionalDependencies: + picomatch: 4.0.3 + + file-entry-cache@8.0.0: + dependencies: + flat-cache: 4.0.1 + + fill-range@7.1.1: + dependencies: + to-regex-range: 5.0.1 + + find-up@5.0.0: + dependencies: + locate-path: 6.0.0 + path-exists: 4.0.0 + + flairup@1.0.0: {} + + flat-cache@4.0.1: + dependencies: + flatted: 3.3.3 + keyv: 4.5.4 + + flatted@3.3.3: {} + + follow-redirects@1.15.11: {} + + for-each@0.3.5: + dependencies: + is-callable: 1.2.7 + + form-data@4.0.4: + dependencies: + asynckit: 0.4.0 + combined-stream: 1.0.8 + es-set-tostringtag: 2.1.0 + hasown: 2.0.2 + mime-types: 2.1.35 + + framer-motion@12.23.24(react-dom@19.1.2(react@19.1.2))(react@19.1.2): + dependencies: + motion-dom: 12.23.23 + motion-utils: 12.23.6 + tslib: 2.8.1 + optionalDependencies: + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + fsevents@2.3.3: + optional: true + + function-bind@1.1.2: {} + + function.prototype.name@1.1.8: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + functions-have-names: 1.2.3 + hasown: 2.0.2 + is-callable: 1.2.7 + + functions-have-names@1.2.3: {} + + generator-function@2.0.1: {} + + get-intrinsic@1.3.0: + 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-nonce@1.0.1: {} + + get-proto@1.0.1: + dependencies: + dunder-proto: 1.0.1 + es-object-atoms: 1.1.1 + + get-symbol-description@1.1.0: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + + get-tsconfig@4.13.0: + dependencies: + resolve-pkg-maps: 1.0.0 + + glob-parent@5.1.2: + dependencies: + is-glob: 4.0.3 + + glob-parent@6.0.2: + dependencies: + is-glob: 4.0.3 + + globals@14.0.0: {} + + globalthis@1.0.4: + dependencies: + define-properties: 1.2.1 + gopd: 1.2.0 + + gopd@1.2.0: {} + + graceful-fs@4.2.11: {} + + graphemer@1.4.0: {} + + has-bigints@1.1.0: {} + + has-flag@4.0.0: {} + + has-property-descriptors@1.0.2: + dependencies: + es-define-property: 1.0.1 + + has-proto@1.2.0: + dependencies: + dunder-proto: 1.0.1 + + has-symbols@1.1.0: {} + + has-tostringtag@1.0.2: + dependencies: + has-symbols: 1.1.0 + + hasown@2.0.2: + dependencies: + function-bind: 1.1.2 + + heic-to@1.3.0: {} + + html-to-image@1.11.13: {} + + ignore@5.3.2: {} + + ignore@7.0.5: {} + + immediate@3.0.6: {} + + import-fresh@3.3.1: + dependencies: + parent-module: 1.0.1 + resolve-from: 4.0.0 + + imurmurhash@0.1.4: {} + + inherits@2.0.4: {} + + internal-slot@1.1.0: + dependencies: + es-errors: 1.3.0 + hasown: 2.0.2 + side-channel: 1.1.0 + + is-array-buffer@3.0.5: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + get-intrinsic: 1.3.0 + + is-async-function@2.1.1: + dependencies: + async-function: 1.0.0 + call-bound: 1.0.4 + get-proto: 1.0.1 + has-tostringtag: 1.0.2 + safe-regex-test: 1.1.0 + + is-bigint@1.1.0: + dependencies: + has-bigints: 1.1.0 + + is-boolean-object@1.2.2: + dependencies: + call-bound: 1.0.4 + has-tostringtag: 1.0.2 + + is-bun-module@2.0.0: + dependencies: + semver: 7.7.3 + + is-callable@1.2.7: {} + + is-core-module@2.16.1: + dependencies: + hasown: 2.0.2 + + is-data-view@1.0.2: + dependencies: + call-bound: 1.0.4 + get-intrinsic: 1.3.0 + is-typed-array: 1.1.15 + + is-date-object@1.1.0: + dependencies: + call-bound: 1.0.4 + has-tostringtag: 1.0.2 + + is-extglob@2.1.1: {} + + is-finalizationregistry@1.1.1: + dependencies: + call-bound: 1.0.4 + + is-generator-function@1.1.2: + dependencies: + call-bound: 1.0.4 + generator-function: 2.0.1 + get-proto: 1.0.1 + has-tostringtag: 1.0.2 + safe-regex-test: 1.1.0 + + is-glob@4.0.3: + dependencies: + is-extglob: 2.1.1 + + is-map@2.0.3: {} + + is-negative-zero@2.0.3: {} + + is-number-object@1.1.1: + dependencies: + call-bound: 1.0.4 + has-tostringtag: 1.0.2 + + is-number@7.0.0: {} + + is-regex@1.2.1: + dependencies: + call-bound: 1.0.4 + gopd: 1.2.0 + has-tostringtag: 1.0.2 + hasown: 2.0.2 + + is-set@2.0.3: {} + + is-shared-array-buffer@1.0.4: + dependencies: + call-bound: 1.0.4 + + is-string@1.1.1: + dependencies: + call-bound: 1.0.4 + has-tostringtag: 1.0.2 + + is-symbol@1.1.1: + dependencies: + call-bound: 1.0.4 + has-symbols: 1.1.0 + safe-regex-test: 1.1.0 + + is-typed-array@1.1.15: + dependencies: + which-typed-array: 1.1.19 + + is-weakmap@2.0.2: {} + + is-weakref@1.1.1: + dependencies: + call-bound: 1.0.4 + + is-weakset@2.0.4: + dependencies: + call-bound: 1.0.4 + get-intrinsic: 1.3.0 + + isarray@1.0.0: {} + + isarray@2.0.5: {} + + isexe@2.0.0: {} + + iterator.prototype@1.1.5: + dependencies: + define-data-property: 1.1.4 + es-object-atoms: 1.1.1 + get-intrinsic: 1.3.0 + get-proto: 1.0.1 + has-symbols: 1.1.0 + set-function-name: 2.0.2 + + jiti@2.6.1: {} + + js-tokens@4.0.0: {} + + js-yaml@4.1.0: + dependencies: + argparse: 2.0.1 + + json-buffer@3.0.1: {} + + json-schema-traverse@0.4.1: {} + + json-stable-stringify-without-jsonify@1.0.1: {} + + json5@1.0.2: + dependencies: + minimist: 1.2.8 + + jsx-ast-utils@3.3.5: + dependencies: + array-includes: 3.1.9 + array.prototype.flat: 1.3.3 + object.assign: 4.1.7 + object.values: 1.2.1 + + jszip@3.10.1: + dependencies: + lie: 3.3.0 + pako: 1.0.11 + readable-stream: 2.3.8 + setimmediate: 1.0.5 + + keyv@4.5.4: + dependencies: + json-buffer: 3.0.1 + + language-subtag-registry@0.3.23: {} + + language-tags@1.0.9: + dependencies: + language-subtag-registry: 0.3.23 + + levn@0.4.1: + dependencies: + prelude-ls: 1.2.1 + type-check: 0.4.0 + + lie@3.3.0: + dependencies: + immediate: 3.0.6 + + lightningcss-android-arm64@1.30.2: + optional: true + + lightningcss-darwin-arm64@1.30.2: + optional: true + + lightningcss-darwin-x64@1.30.2: + optional: true + + lightningcss-freebsd-x64@1.30.2: + optional: true + + lightningcss-linux-arm-gnueabihf@1.30.2: + optional: true + + lightningcss-linux-arm64-gnu@1.30.2: + optional: true + + lightningcss-linux-arm64-musl@1.30.2: + optional: true + + lightningcss-linux-x64-gnu@1.30.2: + optional: true + + lightningcss-linux-x64-musl@1.30.2: + optional: true + + lightningcss-win32-arm64-msvc@1.30.2: + optional: true + + lightningcss-win32-x64-msvc@1.30.2: + optional: true + + lightningcss@1.30.2: + dependencies: + detect-libc: 2.1.2 + optionalDependencies: + lightningcss-android-arm64: 1.30.2 + lightningcss-darwin-arm64: 1.30.2 + lightningcss-darwin-x64: 1.30.2 + lightningcss-freebsd-x64: 1.30.2 + lightningcss-linux-arm-gnueabihf: 1.30.2 + lightningcss-linux-arm64-gnu: 1.30.2 + lightningcss-linux-arm64-musl: 1.30.2 + lightningcss-linux-x64-gnu: 1.30.2 + lightningcss-linux-x64-musl: 1.30.2 + lightningcss-win32-arm64-msvc: 1.30.2 + lightningcss-win32-x64-msvc: 1.30.2 + + locate-path@6.0.0: + dependencies: + p-locate: 5.0.0 + + lodash.merge@4.6.2: {} + + loose-envify@1.4.0: + dependencies: + js-tokens: 4.0.0 + + lottie-react@2.4.1(react-dom@19.1.2(react@19.1.2))(react@19.1.2): + dependencies: + lottie-web: 5.13.0 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + + lottie-web@5.13.0: {} + + lucide-react@0.546.0(react@19.1.2): + dependencies: + react: 19.1.2 + + magic-string@0.30.21: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.5 + + math-intrinsics@1.1.0: {} + + merge2@1.4.1: {} + + micromatch@4.0.8: + dependencies: + braces: 3.0.3 + picomatch: 2.3.1 + + mime-db@1.52.0: {} + + mime-types@2.1.35: + dependencies: + mime-db: 1.52.0 + + minimatch@3.1.2: + dependencies: + brace-expansion: 1.1.12 + + minimatch@9.0.5: + dependencies: + brace-expansion: 2.0.2 + + minimist@1.2.8: {} + + motion-dom@12.23.23: + dependencies: + motion-utils: 12.23.6 + + motion-utils@12.23.6: {} + + ms@2.1.3: {} + + nanoid@3.3.11: {} + + napi-postinstall@0.3.4: {} + + natural-compare@1.4.0: {} + + next@15.5.7(react-dom@19.1.2(react@19.1.2))(react@19.1.2): + dependencies: + '@next/env': 15.5.7 + '@swc/helpers': 0.5.15 + caniuse-lite: 1.0.30001751 + postcss: 8.4.31 + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + styled-jsx: 5.1.6(react@19.1.2) + optionalDependencies: + '@next/swc-darwin-arm64': 15.5.7 + '@next/swc-darwin-x64': 15.5.7 + '@next/swc-linux-arm64-gnu': 15.5.7 + '@next/swc-linux-arm64-musl': 15.5.7 + '@next/swc-linux-x64-gnu': 15.5.7 + '@next/swc-linux-x64-musl': 15.5.7 + '@next/swc-win32-arm64-msvc': 15.5.7 + '@next/swc-win32-x64-msvc': 15.5.7 + sharp: 0.34.4 + transitivePeerDependencies: + - '@babel/core' + - babel-plugin-macros + + object-assign@4.1.1: {} + + object-inspect@1.13.4: {} + + object-keys@1.1.1: {} + + object.assign@4.1.7: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-object-atoms: 1.1.1 + has-symbols: 1.1.0 + object-keys: 1.1.1 + + object.entries@1.1.9: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-object-atoms: 1.1.1 + + object.fromentries@2.0.8: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-object-atoms: 1.1.1 + + object.groupby@1.0.3: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + + object.values@1.2.1: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-object-atoms: 1.1.1 + + obug@2.1.1: {} + + optionator@0.9.4: + dependencies: + deep-is: 0.1.4 + fast-levenshtein: 2.0.6 + levn: 0.4.1 + prelude-ls: 1.2.1 + type-check: 0.4.0 + word-wrap: 1.2.5 + + own-keys@1.0.1: + dependencies: + get-intrinsic: 1.3.0 + object-keys: 1.1.1 + safe-push-apply: 1.0.0 + + p-limit@3.1.0: + dependencies: + yocto-queue: 0.1.0 + + p-locate@5.0.0: + dependencies: + p-limit: 3.1.0 + + pako@1.0.11: {} + + parent-module@1.0.1: + dependencies: + callsites: 3.1.0 + + path-exists@4.0.0: {} + + path-key@3.1.1: {} + + path-parse@1.0.7: {} + + pathe@2.0.3: {} + + picocolors@1.1.1: {} + + picomatch@2.3.1: {} + + picomatch@4.0.3: {} + + possible-typed-array-names@1.1.0: {} + + postcss@8.4.31: + dependencies: + nanoid: 3.3.11 + picocolors: 1.1.1 + source-map-js: 1.2.1 + + postcss@8.5.6: + dependencies: + nanoid: 3.3.11 + picocolors: 1.1.1 + source-map-js: 1.2.1 + + prelude-ls@1.2.1: {} + + prettier-linter-helpers@1.0.0: + dependencies: + fast-diff: 1.3.0 + + prettier-plugin-tailwindcss@0.7.1(prettier@3.6.2): + dependencies: + prettier: 3.6.2 + + prettier@3.6.2: {} + + process-nextick-args@2.0.1: {} + + prop-types@15.8.1: + dependencies: + loose-envify: 1.4.0 + object-assign: 4.1.1 + react-is: 16.13.1 + + proxy-from-env@1.1.0: {} + + punycode@2.3.1: {} + + qr.js@0.0.0: {} + + queue-microtask@1.2.3: {} + + react-day-picker@9.11.1(react@19.1.2): + dependencies: + '@date-fns/tz': 1.4.1 + date-fns: 4.1.0 + date-fns-jalali: 4.1.0-0 + react: 19.1.2 + + react-dom@19.1.2(react@19.1.2): + dependencies: + react: 19.1.2 + scheduler: 0.26.0 + + react-is@16.13.1: {} + + react-qr-code@2.0.18(react@19.1.2): + dependencies: + prop-types: 15.8.1 + qr.js: 0.0.0 + react: 19.1.2 + + react-remove-scroll-bar@2.3.8(@types/react@19.2.2)(react@19.1.2): + dependencies: + react: 19.1.2 + react-style-singleton: 2.2.3(@types/react@19.2.2)(react@19.1.2) + tslib: 2.8.1 + optionalDependencies: + '@types/react': 19.2.2 + + react-remove-scroll@2.7.1(@types/react@19.2.2)(react@19.1.2): + dependencies: + react: 19.1.2 + react-remove-scroll-bar: 2.3.8(@types/react@19.2.2)(react@19.1.2) + react-style-singleton: 2.2.3(@types/react@19.2.2)(react@19.1.2) + tslib: 2.8.1 + use-callback-ref: 1.3.3(@types/react@19.2.2)(react@19.1.2) + use-sidecar: 1.1.3(@types/react@19.2.2)(react@19.1.2) + optionalDependencies: + '@types/react': 19.2.2 + + react-style-singleton@2.2.3(@types/react@19.2.2)(react@19.1.2): + dependencies: + get-nonce: 1.0.1 + react: 19.1.2 + tslib: 2.8.1 + optionalDependencies: + '@types/react': 19.2.2 + + react@19.1.2: {} + + readable-stream@2.3.8: + dependencies: + core-util-is: 1.0.3 + inherits: 2.0.4 + isarray: 1.0.0 + process-nextick-args: 2.0.1 + safe-buffer: 5.1.2 + string_decoder: 1.1.1 + util-deprecate: 1.0.2 + + reflect.getprototypeof@1.0.10: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + get-intrinsic: 1.3.0 + get-proto: 1.0.1 + which-builtin-type: 1.2.1 + + regexp.prototype.flags@1.5.4: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-errors: 1.3.0 + get-proto: 1.0.1 + gopd: 1.2.0 + set-function-name: 2.0.2 + + resolve-from@4.0.0: {} + + resolve-pkg-maps@1.0.0: {} + + resolve@1.22.11: + dependencies: + is-core-module: 2.16.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + + resolve@2.0.0-next.5: + dependencies: + is-core-module: 2.16.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + + reusify@1.1.0: {} + + rollup@4.53.3: + dependencies: + '@types/estree': 1.0.8 + optionalDependencies: + '@rollup/rollup-android-arm-eabi': 4.53.3 + '@rollup/rollup-android-arm64': 4.53.3 + '@rollup/rollup-darwin-arm64': 4.53.3 + '@rollup/rollup-darwin-x64': 4.53.3 + '@rollup/rollup-freebsd-arm64': 4.53.3 + '@rollup/rollup-freebsd-x64': 4.53.3 + '@rollup/rollup-linux-arm-gnueabihf': 4.53.3 + '@rollup/rollup-linux-arm-musleabihf': 4.53.3 + '@rollup/rollup-linux-arm64-gnu': 4.53.3 + '@rollup/rollup-linux-arm64-musl': 4.53.3 + '@rollup/rollup-linux-loong64-gnu': 4.53.3 + '@rollup/rollup-linux-ppc64-gnu': 4.53.3 + '@rollup/rollup-linux-riscv64-gnu': 4.53.3 + '@rollup/rollup-linux-riscv64-musl': 4.53.3 + '@rollup/rollup-linux-s390x-gnu': 4.53.3 + '@rollup/rollup-linux-x64-gnu': 4.53.3 + '@rollup/rollup-linux-x64-musl': 4.53.3 + '@rollup/rollup-openharmony-arm64': 4.53.3 + '@rollup/rollup-win32-arm64-msvc': 4.53.3 + '@rollup/rollup-win32-ia32-msvc': 4.53.3 + '@rollup/rollup-win32-x64-gnu': 4.53.3 + '@rollup/rollup-win32-x64-msvc': 4.53.3 + fsevents: 2.3.3 + + run-parallel@1.2.0: + dependencies: + queue-microtask: 1.2.3 + + safe-array-concat@1.1.3: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + get-intrinsic: 1.3.0 + has-symbols: 1.1.0 + isarray: 2.0.5 + + safe-buffer@5.1.2: {} + + safe-push-apply@1.0.0: + dependencies: + es-errors: 1.3.0 + isarray: 2.0.5 + + safe-regex-test@1.1.0: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + is-regex: 1.2.1 + + scheduler@0.26.0: {} + + semver@6.3.1: {} + + semver@7.7.3: {} + + set-function-length@1.2.2: + dependencies: + define-data-property: 1.1.4 + es-errors: 1.3.0 + function-bind: 1.1.2 + get-intrinsic: 1.3.0 + gopd: 1.2.0 + has-property-descriptors: 1.0.2 + + set-function-name@2.0.2: + dependencies: + define-data-property: 1.1.4 + es-errors: 1.3.0 + functions-have-names: 1.2.3 + has-property-descriptors: 1.0.2 + + set-proto@1.0.0: + dependencies: + dunder-proto: 1.0.1 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + + setimmediate@1.0.5: {} + + sharp@0.34.4: + dependencies: + '@img/colour': 1.0.0 + detect-libc: 2.1.2 + semver: 7.7.3 + optionalDependencies: + '@img/sharp-darwin-arm64': 0.34.4 + '@img/sharp-darwin-x64': 0.34.4 + '@img/sharp-libvips-darwin-arm64': 1.2.3 + '@img/sharp-libvips-darwin-x64': 1.2.3 + '@img/sharp-libvips-linux-arm': 1.2.3 + '@img/sharp-libvips-linux-arm64': 1.2.3 + '@img/sharp-libvips-linux-ppc64': 1.2.3 + '@img/sharp-libvips-linux-s390x': 1.2.3 + '@img/sharp-libvips-linux-x64': 1.2.3 + '@img/sharp-libvips-linuxmusl-arm64': 1.2.3 + '@img/sharp-libvips-linuxmusl-x64': 1.2.3 + '@img/sharp-linux-arm': 0.34.4 + '@img/sharp-linux-arm64': 0.34.4 + '@img/sharp-linux-ppc64': 0.34.4 + '@img/sharp-linux-s390x': 0.34.4 + '@img/sharp-linux-x64': 0.34.4 + '@img/sharp-linuxmusl-arm64': 0.34.4 + '@img/sharp-linuxmusl-x64': 0.34.4 + '@img/sharp-wasm32': 0.34.4 + '@img/sharp-win32-arm64': 0.34.4 + '@img/sharp-win32-ia32': 0.34.4 + '@img/sharp-win32-x64': 0.34.4 + optional: true + + shebang-command@2.0.0: + dependencies: + shebang-regex: 3.0.0 + + shebang-regex@3.0.0: {} + + side-channel-list@1.0.0: + dependencies: + es-errors: 1.3.0 + object-inspect: 1.13.4 + + side-channel-map@1.0.1: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + object-inspect: 1.13.4 + + side-channel-weakmap@1.0.2: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + get-intrinsic: 1.3.0 + object-inspect: 1.13.4 + side-channel-map: 1.0.1 + + side-channel@1.1.0: + dependencies: + es-errors: 1.3.0 + object-inspect: 1.13.4 + side-channel-list: 1.0.0 + side-channel-map: 1.0.1 + side-channel-weakmap: 1.0.2 + + siginfo@2.0.0: {} + + source-map-js@1.2.1: {} + + stable-hash@0.0.5: {} + + stackback@0.0.2: {} + + std-env@3.10.0: {} + + stop-iteration-iterator@1.1.0: + dependencies: + es-errors: 1.3.0 + internal-slot: 1.1.0 + + string.prototype.includes@2.0.1: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-abstract: 1.24.0 + + string.prototype.matchall@4.0.12: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-errors: 1.3.0 + es-object-atoms: 1.1.1 + get-intrinsic: 1.3.0 + gopd: 1.2.0 + has-symbols: 1.1.0 + internal-slot: 1.1.0 + regexp.prototype.flags: 1.5.4 + set-function-name: 2.0.2 + side-channel: 1.1.0 + + string.prototype.repeat@1.0.0: + dependencies: + define-properties: 1.2.1 + es-abstract: 1.24.0 + + string.prototype.trim@1.2.10: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-data-property: 1.1.4 + define-properties: 1.2.1 + es-abstract: 1.24.0 + es-object-atoms: 1.1.1 + has-property-descriptors: 1.0.2 + + string.prototype.trimend@1.0.9: + dependencies: + call-bind: 1.0.8 + call-bound: 1.0.4 + define-properties: 1.2.1 + es-object-atoms: 1.1.1 + + string.prototype.trimstart@1.0.8: + dependencies: + call-bind: 1.0.8 + define-properties: 1.2.1 + es-object-atoms: 1.1.1 + + string_decoder@1.1.1: + dependencies: + safe-buffer: 5.1.2 + + strip-bom@3.0.0: {} + + strip-json-comments@3.1.1: {} + + styled-jsx@5.1.6(react@19.1.2): + dependencies: + client-only: 0.0.1 + react: 19.1.2 + + supports-color@7.2.0: + dependencies: + has-flag: 4.0.0 + + supports-preserve-symlinks-flag@1.0.0: {} + + swiper@12.0.3: {} + + synckit@0.11.11: + dependencies: + '@pkgr/core': 0.2.9 + + tabbable@6.3.0: {} + + tailwind-merge@3.3.1: {} + + tailwindcss@4.1.16: {} + + tapable@2.3.0: {} + + tinybench@2.9.0: {} + + tinyexec@0.3.2: {} + + tinyglobby@0.2.15: + dependencies: + fdir: 6.5.0(picomatch@4.0.3) + picomatch: 4.0.3 + + tinyrainbow@3.0.3: {} + + to-regex-range@5.0.1: + dependencies: + is-number: 7.0.0 + + ts-api-utils@2.1.0(typescript@5.9.3): + dependencies: + typescript: 5.9.3 + + tsconfig-paths@3.15.0: + dependencies: + '@types/json5': 0.0.29 + json5: 1.0.2 + minimist: 1.2.8 + strip-bom: 3.0.0 + + tslib@2.8.1: {} + + tw-animate-css@1.4.0: {} + + type-check@0.4.0: + dependencies: + prelude-ls: 1.2.1 + + typed-array-buffer@1.0.3: + dependencies: + call-bound: 1.0.4 + es-errors: 1.3.0 + is-typed-array: 1.1.15 + + typed-array-byte-length@1.0.3: + dependencies: + call-bind: 1.0.8 + for-each: 0.3.5 + gopd: 1.2.0 + has-proto: 1.2.0 + is-typed-array: 1.1.15 + + typed-array-byte-offset@1.0.4: + dependencies: + available-typed-arrays: 1.0.7 + call-bind: 1.0.8 + for-each: 0.3.5 + gopd: 1.2.0 + has-proto: 1.2.0 + is-typed-array: 1.1.15 + reflect.getprototypeof: 1.0.10 + + typed-array-length@1.0.7: + dependencies: + call-bind: 1.0.8 + for-each: 0.3.5 + gopd: 1.2.0 + is-typed-array: 1.1.15 + possible-typed-array-names: 1.1.0 + reflect.getprototypeof: 1.0.10 + + typescript@5.9.3: {} + + unbox-primitive@1.1.0: + dependencies: + call-bound: 1.0.4 + has-bigints: 1.1.0 + has-symbols: 1.1.0 + which-boxed-primitive: 1.1.1 + + undici-types@6.21.0: {} + + unrs-resolver@1.11.1: + dependencies: + napi-postinstall: 0.3.4 + optionalDependencies: + '@unrs/resolver-binding-android-arm-eabi': 1.11.1 + '@unrs/resolver-binding-android-arm64': 1.11.1 + '@unrs/resolver-binding-darwin-arm64': 1.11.1 + '@unrs/resolver-binding-darwin-x64': 1.11.1 + '@unrs/resolver-binding-freebsd-x64': 1.11.1 + '@unrs/resolver-binding-linux-arm-gnueabihf': 1.11.1 + '@unrs/resolver-binding-linux-arm-musleabihf': 1.11.1 + '@unrs/resolver-binding-linux-arm64-gnu': 1.11.1 + '@unrs/resolver-binding-linux-arm64-musl': 1.11.1 + '@unrs/resolver-binding-linux-ppc64-gnu': 1.11.1 + '@unrs/resolver-binding-linux-riscv64-gnu': 1.11.1 + '@unrs/resolver-binding-linux-riscv64-musl': 1.11.1 + '@unrs/resolver-binding-linux-s390x-gnu': 1.11.1 + '@unrs/resolver-binding-linux-x64-gnu': 1.11.1 + '@unrs/resolver-binding-linux-x64-musl': 1.11.1 + '@unrs/resolver-binding-wasm32-wasi': 1.11.1 + '@unrs/resolver-binding-win32-arm64-msvc': 1.11.1 + '@unrs/resolver-binding-win32-ia32-msvc': 1.11.1 + '@unrs/resolver-binding-win32-x64-msvc': 1.11.1 + + uri-js@4.4.1: + dependencies: + punycode: 2.3.1 + + use-callback-ref@1.3.3(@types/react@19.2.2)(react@19.1.2): + dependencies: + react: 19.1.2 + tslib: 2.8.1 + optionalDependencies: + '@types/react': 19.2.2 + + use-sidecar@1.1.3(@types/react@19.2.2)(react@19.1.2): + dependencies: + detect-node-es: 1.1.0 + react: 19.1.2 + tslib: 2.8.1 + optionalDependencies: + '@types/react': 19.2.2 + + use-sync-external-store@1.6.0(react@19.1.2): + dependencies: + react: 19.1.2 + + util-deprecate@1.0.2: {} + + vaul@1.1.2(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2): + dependencies: + '@radix-ui/react-dialog': 1.1.15(@types/react-dom@19.2.2(@types/react@19.2.2))(@types/react@19.2.2)(react-dom@19.1.2(react@19.1.2))(react@19.1.2) + react: 19.1.2 + react-dom: 19.1.2(react@19.1.2) + transitivePeerDependencies: + - '@types/react' + - '@types/react-dom' + + vite@7.2.4(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2): + dependencies: + esbuild: 0.25.12 + fdir: 6.5.0(picomatch@4.0.3) + picomatch: 4.0.3 + postcss: 8.5.6 + rollup: 4.53.3 + tinyglobby: 0.2.15 + optionalDependencies: + '@types/node': 20.19.23 + fsevents: 2.3.3 + jiti: 2.6.1 + lightningcss: 1.30.2 + + vitest@4.0.14(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2): + dependencies: + '@vitest/expect': 4.0.14 + '@vitest/mocker': 4.0.14(vite@7.2.4(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2)) + '@vitest/pretty-format': 4.0.14 + '@vitest/runner': 4.0.14 + '@vitest/snapshot': 4.0.14 + '@vitest/spy': 4.0.14 + '@vitest/utils': 4.0.14 + es-module-lexer: 1.7.0 + expect-type: 1.2.2 + magic-string: 0.30.21 + obug: 2.1.1 + pathe: 2.0.3 + picomatch: 4.0.3 + std-env: 3.10.0 + tinybench: 2.9.0 + tinyexec: 0.3.2 + tinyglobby: 0.2.15 + tinyrainbow: 3.0.3 + vite: 7.2.4(@types/node@20.19.23)(jiti@2.6.1)(lightningcss@1.30.2) + why-is-node-running: 2.3.0 + optionalDependencies: + '@types/node': 20.19.23 + transitivePeerDependencies: + - jiti + - less + - lightningcss + - msw + - sass + - sass-embedded + - stylus + - sugarss + - terser + - tsx + - yaml + + which-boxed-primitive@1.1.1: + dependencies: + is-bigint: 1.1.0 + is-boolean-object: 1.2.2 + is-number-object: 1.1.1 + is-string: 1.1.1 + is-symbol: 1.1.1 + + which-builtin-type@1.2.1: + dependencies: + call-bound: 1.0.4 + function.prototype.name: 1.1.8 + has-tostringtag: 1.0.2 + is-async-function: 2.1.1 + is-date-object: 1.1.0 + is-finalizationregistry: 1.1.1 + is-generator-function: 1.1.2 + is-regex: 1.2.1 + is-weakref: 1.1.1 + isarray: 2.0.5 + which-boxed-primitive: 1.1.1 + which-collection: 1.0.2 + which-typed-array: 1.1.19 + + which-collection@1.0.2: + dependencies: + is-map: 2.0.3 + is-set: 2.0.3 + is-weakmap: 2.0.2 + is-weakset: 2.0.4 + + which-typed-array@1.1.19: + dependencies: + available-typed-arrays: 1.0.7 + call-bind: 1.0.8 + call-bound: 1.0.4 + for-each: 0.3.5 + get-proto: 1.0.1 + gopd: 1.2.0 + has-tostringtag: 1.0.2 + + which@2.0.2: + dependencies: + isexe: 2.0.0 + + why-is-node-running@2.3.0: + dependencies: + siginfo: 2.0.0 + stackback: 0.0.2 + + word-wrap@1.2.5: {} + + yocto-queue@0.1.0: {} + + zustand@5.0.8(@types/react@19.2.2)(react@19.1.2)(use-sync-external-store@1.6.0(react@19.1.2)): + optionalDependencies: + '@types/react': 19.2.2 + react: 19.1.2 + use-sync-external-store: 1.6.0(react@19.1.2) diff --git a/postcss.config.mjs b/postcss.config.mjs new file mode 100644 index 00000000..ba720fe5 --- /dev/null +++ b/postcss.config.mjs @@ -0,0 +1,5 @@ +const config = { + plugins: ['@tailwindcss/postcss'], +}; + +export default config; diff --git a/public/assets/album/3Tags_Fill_Album.json b/public/assets/album/3Tags_Fill_Album.json new file mode 100644 index 00000000..2c1985cc --- /dev/null +++ b/public/assets/album/3Tags_Fill_Album.json @@ -0,0 +1 @@ +{"nm":"Comp 1","ddd":0,"h":500,"w":1200,"meta":{"g":"@lottiefiles/toolkit-js 0.66.4","tc":"#ffffff"},"layers":[{"ty":0,"nm":"3개 태그","sr":1.3,"st":-117,"op":248.3,"ip":0,"ln":"253","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"1","ind":1}],"v":"5.7.0","fr":24,"op":246,"ip":0,"assets":[{"nm":"3개 태그","id":"1","fr":24,"layers":[{"ty":0,"nm":"군침돌게_end_2","sr":1,"st":209,"op":428,"ip":293,"ln":"241","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":260.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":271.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":286.999},{"s":[600,110,0],"t":298.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":1},{"ty":0,"nm":"우리만_end_2","sr":1,"st":182,"op":401,"ip":266,"ln":"240","bm":17,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":182},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":233.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":244.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":259.999},{"s":[600,110,0],"t":271.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":2},{"ty":0,"nm":"못생기게","sr":1,"st":157,"op":376,"ip":241,"ln":"239","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":208.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":219.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":234.999},{"s":[600,110,0],"t":246.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"9","ind":3},{"ty":0,"nm":"아름다운 풍경_2","sr":1,"st":131,"op":350,"ip":215,"ln":"238","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":182.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":193.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":208.999},{"s":[600,110,0],"t":220.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"12","ind":4},{"ty":0,"nm":"흔들렸지만_2","sr":1,"st":105,"op":324,"ip":189,"ln":"237","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":156.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":167.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":182.999},{"s":[600,110,0],"t":194.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"15","ind":5},{"ty":0,"nm":"예상치 못한_2","sr":1,"st":79,"op":298,"ip":163,"ln":"236","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":130.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":141.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":156.999},{"s":[600,110,0],"t":168.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"18","ind":6},{"ty":0,"nm":"우연히_2","sr":1,"st":53,"op":272,"ip":131,"ln":"235","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":104.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":115.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":130.999},{"s":[600,110,0],"t":142.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":7},{"ty":0,"nm":"군침돌게_2","sr":1,"st":26,"op":245,"ip":110,"ln":"234","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":77.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":88.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":103.999},{"s":[600,110,0],"t":115.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":8},{"ty":0,"nm":"우리만","sr":1,"st":0,"op":219,"ip":0,"ln":"233","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":51.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":77.999},{"s":[600,110,0],"t":89.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":9},{"ty":0,"nm":"군침돌게","sr":1,"st":26,"op":245,"ip":26,"ln":"232","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":77.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":88.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":103.999},{"s":[600,110,0],"t":115.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":10},{"ty":0,"nm":"우연히","sr":1,"st":53,"op":272,"ip":53,"ln":"231","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":104.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":115.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":130.999},{"s":[600,110,0],"t":142.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":11},{"ty":0,"nm":"예상치 못한","sr":1,"st":79,"op":298,"ip":79,"ln":"230","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":130.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":141.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":156.999},{"s":[600,110,0],"t":168.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"18","ind":12},{"ty":0,"nm":"흔들렸지만","sr":1,"st":105,"op":324,"ip":105,"ln":"229","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":156.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":167.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":182.999},{"s":[600,110,0],"t":194.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"15","ind":13},{"ty":0,"nm":"아름다운 풍경","sr":1,"st":131,"op":350,"ip":131,"ln":"228","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":182.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":193.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":208.999},{"s":[600,110,0],"t":220.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"12","ind":14},{"ty":0,"nm":"못생기게","sr":1,"st":157,"op":376,"ip":157,"ln":"227","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":208.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":219.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":234.999},{"s":[600,110,0],"t":246.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"9","ind":15},{"ty":0,"nm":"우리만_end","sr":1,"st":182,"op":401,"ip":206,"ln":"226","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":182},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":233.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":244.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":259.999},{"s":[600,110,0],"t":271.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":16},{"ty":0,"nm":"군침돌게_end","sr":1,"st":209,"op":428,"ip":209,"ln":"225","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":260.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":271.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":286.999},{"s":[600,110,0],"t":298.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":17},{"ty":0,"nm":"우연히_end","sr":1,"st":236,"op":455,"ip":236,"ln":"224","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":287.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":298.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":313.999},{"s":[600,110,0],"t":325.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":18}]},{"nm":"우연히","id":"2","fr":24,"layers":[{"ty":2,"nm":"Element-2.png","sr":1,"st":0,"op":219,"ip":0,"ln":"93","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"3","ind":1},{"ty":2,"nm":"Element_S-2.png","sr":1,"st":0,"op":219,"ip":0,"ln":"92","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"4","ind":2}]},{"id":"3","e":1,"w":600,"h":88,"p":"","u":""},{"id":"4","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"군침돌게","id":"5","fr":24,"layers":[{"ty":2,"nm":"군침돌게","sr":1,"st":0,"op":219,"ip":0,"ln":"68","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"6","ind":1},{"ty":2,"nm":"군침돌게_s","sr":1,"st":0,"op":219,"ip":0,"ln":"67","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"7","ind":2}]},{"id":"6","e":1,"w":600,"h":88,"p":"","u":""},{"id":"7","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"우리만","id":"8","fr":24,"layers":[{"ty":2,"nm":"우리만","sr":1,"st":0,"op":219,"ip":0,"ln":"50","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100.862],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":0},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,150],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"21","ind":1},{"ty":2,"nm":"우리만_s","sr":1,"st":0,"op":219,"ip":0,"ln":"49","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100.385],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":0},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,98.182],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"22","ind":2}]},{"nm":"못생기게","id":"9","fr":24,"layers":[{"ty":2,"nm":"Element-6.png","sr":1,"st":0,"op":219,"ip":0,"ln":"195","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"10","ind":1},{"ty":2,"nm":"Element_S-6.png","sr":1,"st":0,"op":219,"ip":0,"ln":"194","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"11","ind":2}]},{"id":"10","e":1,"w":600,"h":88,"p":"","u":""},{"id":"11","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"아름다운 풍경","id":"12","fr":24,"layers":[{"ty":2,"nm":"Element-5.png","sr":1,"st":0,"op":219,"ip":0,"ln":"168","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"13","ind":1},{"ty":2,"nm":"Element_S-5.png","sr":1,"st":0,"op":219,"ip":0,"ln":"167","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"14","ind":2}]},{"id":"13","e":1,"w":600,"h":88,"p":"","u":""},{"id":"14","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"흔들렸지만","id":"15","fr":24,"layers":[{"ty":2,"nm":"Element-4.png","sr":1,"st":0,"op":219,"ip":0,"ln":"142","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"16","ind":1},{"ty":2,"nm":"Element_S-4.png","sr":1,"st":0,"op":219,"ip":0,"ln":"141","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"17","ind":2}]},{"id":"16","e":1,"w":600,"h":88,"p":"","u":""},{"id":"17","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"예상치 못한","id":"18","fr":24,"layers":[{"ty":2,"nm":"Element-3.png","sr":1,"st":0,"op":219,"ip":0,"ln":"117","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"19","ind":1},{"ty":2,"nm":"Element_S-3.png","sr":1,"st":0,"op":219,"ip":0,"ln":"116","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"20","ind":2}]},{"id":"19","e":1,"w":600,"h":88,"p":"","u":""},{"id":"20","e":1,"w":480,"h":72,"p":"","u":""},{"id":"21","e":1,"w":600,"h":88,"p":"","u":""},{"id":"22","e":1,"w":480,"h":72,"p":"","u":""}]} \ No newline at end of file diff --git a/public/assets/album/CheeseCart_Loading.json b/public/assets/album/CheeseCart_Loading.json new file mode 100644 index 00000000..3143d18d --- /dev/null +++ b/public/assets/album/CheeseCart_Loading.json @@ -0,0 +1 @@ +{"nm":"cheeseCart_Loading","ddd":0,"h":360,"w":1200,"meta":{"g":"@lottiefiles/toolkit-js 0.66.4","tc":"#ffffff"},"layers":[{"ty":0,"nm":"치즈카트","sr":1,"st":0,"op":106,"ip":0,"ln":"40","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,180]},"s":{"a":0,"k":[100,100]},"p":{"s":true,"x":{"a":1,"k":[{"o":{"x":0.794,"y":-0.372},"i":{"x":0.339,"y":1.281},"s":[600],"t":39},{"s":[-330.75],"t":69.999}]},"y":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[180],"t":39},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[180],"t":69.999},{"s":[180],"t":76.999}]},"z":{"a":0,"k":0}},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":360,"refId":"1","ind":1}],"v":"5.7.0","fr":24,"op":75,"ip":0,"assets":[{"nm":"치즈카트","id":"1","fr":24,"layers":[{"ty":2,"nm":"치즈_3.png","sr":1,"st":19,"op":125,"ip":19,"ln":"29","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[86,18]},"s":{"a":0,"k":[154.199,154.199,89.535]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,-209.5,0],"t":32},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,53.5,0],"t":36},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,49.7,0],"t":37},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,50.402,0],"t":38},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,72.5,0],"t":40},{"s":[579.5,84.5,0],"t":42}]},"r":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[23],"t":32},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[7.667],"t":35},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[-7.167],"t":36},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[7.267],"t":37},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[12.45],"t":38},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[-3],"t":40},{"s":[0],"t":42}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"2","ind":1},{"ty":2,"nm":"치즈_2.png","sr":1,"st":18,"op":124,"ip":18,"ln":"28","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[85,18]},"s":{"a":0,"k":[165.425,165.425,93.051]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[545.5,-146.5,0],"t":29},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[545.5,129.5,0],"t":32},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[545.5,127.5,0],"t":33},{"s":[545.5,147.5,0],"t":37}]},"r":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[-14],"t":32},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[6],"t":35},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[-7],"t":36},{"s":[0],"t":37}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"3","ind":2},{"ty":2,"nm":"치즈_1.png","sr":1,"st":15,"op":121,"ip":15,"ln":"27","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[86,18]},"s":{"a":0,"k":[154.199,154.199,89.535]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,-84.5,0],"t":22},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,191.5,0],"t":28},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,211.5,0],"t":29},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[579.5,191.5,0],"t":30},{"s":[579.5,211.5,0],"t":34}]},"r":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[25],"t":25},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":29},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[-9],"t":30},{"s":[0],"t":34}]},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"2","ind":3},{"ty":2,"nm":"카트.png","sr":1,"st":0,"op":106,"ip":0,"ln":"26","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[111,69]},"s":{"a":0,"k":[156.796,158.911,100]},"p":{"s":true,"x":{"a":1,"k":[{"o":{"x":0.933,"y":0},"i":{"x":0.157,"y":1},"s":[1532.75],"t":4},{"s":[602.833],"t":22}]},"y":{"a":1,"k":[{"o":{"x":0.167,"y":0.167},"i":{"x":0.833,"y":0.833},"s":[192],"t":4},{"s":[192],"t":22}]},"z":{"a":0,"k":0}},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"refId":"4","ind":4}]},{"id":"2","e":1,"w":172,"h":36,"p":"","u":""},{"id":"3","e":1,"w":170,"h":36,"p":"","u":""},{"id":"4","e":1,"w":222,"h":138,"p":"","u":""}]} \ No newline at end of file diff --git a/public/assets/album/bg-album-default.png b/public/assets/album/bg-album-default.png new file mode 100644 index 00000000..d482c111 Binary files /dev/null and b/public/assets/album/bg-album-default.png differ diff --git a/public/assets/album/bg-album-entry.png b/public/assets/album/bg-album-entry.png new file mode 100644 index 00000000..02666a1a Binary files /dev/null and b/public/assets/album/bg-album-entry.png differ diff --git a/public/assets/album/letter-full-size.svg b/public/assets/album/letter-full-size.svg new file mode 100644 index 00000000..3cfea587 --- /dev/null +++ b/public/assets/album/letter-full-size.svg @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/album/no-album-icon.svg b/public/assets/album/no-album-icon.svg new file mode 100644 index 00000000..f5d0af59 --- /dev/null +++ b/public/assets/album/no-album-icon.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/public/assets/album/test-lottie.svg b/public/assets/album/test-lottie.svg new file mode 100644 index 00000000..5c5ec880 --- /dev/null +++ b/public/assets/album/test-lottie.svg @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/public/assets/login/cheese-icon.svg b/public/assets/login/cheese-icon.svg new file mode 100644 index 00000000..8382e81f --- /dev/null +++ b/public/assets/login/cheese-icon.svg @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/login/cheese-logo.svg b/public/assets/login/cheese-logo.svg new file mode 100644 index 00000000..7f62ec5b --- /dev/null +++ b/public/assets/login/cheese-logo.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/public/assets/login/kakao-logo.svg b/public/assets/login/kakao-logo.svg new file mode 100644 index 00000000..52c7019c --- /dev/null +++ b/public/assets/login/kakao-logo.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/assets/login/reverse-triangle-black.svg b/public/assets/login/reverse-triangle-black.svg new file mode 100644 index 00000000..a201d2a9 --- /dev/null +++ b/public/assets/login/reverse-triangle-black.svg @@ -0,0 +1,3 @@ + + + diff --git a/public/assets/login/triangle-polygon.svg b/public/assets/login/triangle-polygon.svg new file mode 100644 index 00000000..5e89dff8 --- /dev/null +++ b/public/assets/login/triangle-polygon.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/public/assets/og/og_kakao.png b/public/assets/og/og_kakao.png new file mode 100644 index 00000000..754121d6 Binary files /dev/null and b/public/assets/og/og_kakao.png differ diff --git a/public/assets/onboarding/congratulation.svg b/public/assets/onboarding/congratulation.svg new file mode 100644 index 00000000..7a3aa5d7 --- /dev/null +++ b/public/assets/onboarding/congratulation.svg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/rending/blur.svg b/public/assets/rending/blur.svg new file mode 100644 index 00000000..f426253c --- /dev/null +++ b/public/assets/rending/blur.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/public/assets/rending/box.png b/public/assets/rending/box.png new file mode 100644 index 00000000..f8b73ab4 Binary files /dev/null and b/public/assets/rending/box.png differ diff --git a/public/assets/rending/box.svg b/public/assets/rending/box.svg new file mode 100644 index 00000000..32e0201c --- /dev/null +++ b/public/assets/rending/box.svg @@ -0,0 +1,87 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/rending/first.svg b/public/assets/rending/first.svg new file mode 100644 index 00000000..8f319d6a --- /dev/null +++ b/public/assets/rending/first.svg @@ -0,0 +1,166 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/rending/phone.png b/public/assets/rending/phone.png new file mode 100644 index 00000000..dffb3d8b Binary files /dev/null and b/public/assets/rending/phone.png differ diff --git a/public/assets/rending/second.svg b/public/assets/rending/second.svg new file mode 100644 index 00000000..14cef82c --- /dev/null +++ b/public/assets/rending/second.svg @@ -0,0 +1,134 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/rending/swipe/1.png b/public/assets/rending/swipe/1.png new file mode 100644 index 00000000..abb58752 Binary files /dev/null and b/public/assets/rending/swipe/1.png differ diff --git a/public/assets/rending/swipe/2.png b/public/assets/rending/swipe/2.png new file mode 100644 index 00000000..e6cf72e0 Binary files /dev/null and b/public/assets/rending/swipe/2.png differ diff --git a/public/assets/rending/swipe/3.png b/public/assets/rending/swipe/3.png new file mode 100644 index 00000000..72ba561c Binary files /dev/null and b/public/assets/rending/swipe/3.png differ diff --git a/public/assets/rending/swipe/4.png b/public/assets/rending/swipe/4.png new file mode 100644 index 00000000..8a3849d6 Binary files /dev/null and b/public/assets/rending/swipe/4.png differ diff --git a/public/assets/rending/text-group.svg b/public/assets/rending/text-group.svg new file mode 100644 index 00000000..df172aaa --- /dev/null +++ b/public/assets/rending/text-group.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/public/assets/rending/third.svg b/public/assets/rending/third.svg new file mode 100644 index 00000000..629412d0 --- /dev/null +++ b/public/assets/rending/third.svg @@ -0,0 +1,211 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/upload/3Tags_Fill Album.json b/public/assets/upload/3Tags_Fill Album.json new file mode 100644 index 00000000..2c1985cc --- /dev/null +++ b/public/assets/upload/3Tags_Fill Album.json @@ -0,0 +1 @@ +{"nm":"Comp 1","ddd":0,"h":500,"w":1200,"meta":{"g":"@lottiefiles/toolkit-js 0.66.4","tc":"#ffffff"},"layers":[{"ty":0,"nm":"3개 태그","sr":1.3,"st":-117,"op":248.3,"ip":0,"ln":"253","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"1","ind":1}],"v":"5.7.0","fr":24,"op":246,"ip":0,"assets":[{"nm":"3개 태그","id":"1","fr":24,"layers":[{"ty":0,"nm":"군침돌게_end_2","sr":1,"st":209,"op":428,"ip":293,"ln":"241","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":260.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":271.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":286.999},{"s":[600,110,0],"t":298.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":1},{"ty":0,"nm":"우리만_end_2","sr":1,"st":182,"op":401,"ip":266,"ln":"240","bm":17,"hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":182},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":233.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":244.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":259.999},{"s":[600,110,0],"t":271.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":2},{"ty":0,"nm":"못생기게","sr":1,"st":157,"op":376,"ip":241,"ln":"239","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":208.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":219.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":234.999},{"s":[600,110,0],"t":246.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"9","ind":3},{"ty":0,"nm":"아름다운 풍경_2","sr":1,"st":131,"op":350,"ip":215,"ln":"238","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":182.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":193.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":208.999},{"s":[600,110,0],"t":220.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"12","ind":4},{"ty":0,"nm":"흔들렸지만_2","sr":1,"st":105,"op":324,"ip":189,"ln":"237","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":156.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":167.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":182.999},{"s":[600,110,0],"t":194.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"15","ind":5},{"ty":0,"nm":"예상치 못한_2","sr":1,"st":79,"op":298,"ip":163,"ln":"236","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":130.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":141.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":156.999},{"s":[600,110,0],"t":168.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"18","ind":6},{"ty":0,"nm":"우연히_2","sr":1,"st":53,"op":272,"ip":131,"ln":"235","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":104.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":115.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":130.999},{"s":[600,110,0],"t":142.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":7},{"ty":0,"nm":"군침돌게_2","sr":1,"st":26,"op":245,"ip":110,"ln":"234","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":77.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":88.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":103.999},{"s":[600,110,0],"t":115.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":8},{"ty":0,"nm":"우리만","sr":1,"st":0,"op":219,"ip":0,"ln":"233","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":51.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":77.999},{"s":[600,110,0],"t":89.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":9},{"ty":0,"nm":"군침돌게","sr":1,"st":26,"op":245,"ip":26,"ln":"232","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":77.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":88.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":103.999},{"s":[600,110,0],"t":115.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":10},{"ty":0,"nm":"우연히","sr":1,"st":53,"op":272,"ip":53,"ln":"231","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":104.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":115.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":130.999},{"s":[600,110,0],"t":142.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":11},{"ty":0,"nm":"예상치 못한","sr":1,"st":79,"op":298,"ip":79,"ln":"230","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":130.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":141.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":156.999},{"s":[600,110,0],"t":168.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"18","ind":12},{"ty":0,"nm":"흔들렸지만","sr":1,"st":105,"op":324,"ip":105,"ln":"229","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":156.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":167.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":182.999},{"s":[600,110,0],"t":194.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"15","ind":13},{"ty":0,"nm":"아름다운 풍경","sr":1,"st":131,"op":350,"ip":131,"ln":"228","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":182.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":193.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":208.999},{"s":[600,110,0],"t":220.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"12","ind":14},{"ty":0,"nm":"못생기게","sr":1,"st":157,"op":376,"ip":157,"ln":"227","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":208.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":219.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":234.999},{"s":[600,110,0],"t":246.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"9","ind":15},{"ty":0,"nm":"우리만_end","sr":1,"st":182,"op":401,"ip":206,"ln":"226","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0,"y":0},"i":{"x":0.667,"y":0.667},"s":[600,422,0],"t":182},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":233.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":244.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":259.999},{"s":[600,110,0],"t":271.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"8","ind":16},{"ty":0,"nm":"군침돌게_end","sr":1,"st":209,"op":428,"ip":209,"ln":"225","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":260.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":271.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":286.999},{"s":[600,110,0],"t":298.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"5","ind":17},{"ty":0,"nm":"우연히_end","sr":1,"st":236,"op":455,"ip":236,"ln":"224","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[600,250]},"s":{"a":0,"k":[100,100]},"p":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,422,0],"t":287.999},{"o":{"x":0.333,"y":0.333},"i":{"x":0.667,"y":0.667},"s":[600,266,0],"t":298.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[600,266,0],"t":313.999},{"s":[600,110,0],"t":325.999}]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":0,"k":100}},"w":1200,"h":500,"refId":"2","ind":18}]},{"nm":"우연히","id":"2","fr":24,"layers":[{"ty":2,"nm":"Element-2.png","sr":1,"st":0,"op":219,"ip":0,"ln":"93","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"3","ind":1},{"ty":2,"nm":"Element_S-2.png","sr":1,"st":0,"op":219,"ip":0,"ln":"92","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"4","ind":2}]},{"id":"3","e":1,"w":600,"h":88,"p":"","u":""},{"id":"4","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"군침돌게","id":"5","fr":24,"layers":[{"ty":2,"nm":"군침돌게","sr":1,"st":0,"op":219,"ip":0,"ln":"68","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"6","ind":1},{"ty":2,"nm":"군침돌게_s","sr":1,"st":0,"op":219,"ip":0,"ln":"67","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"7","ind":2}]},{"id":"6","e":1,"w":600,"h":88,"p":"","u":""},{"id":"7","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"우리만","id":"8","fr":24,"layers":[{"ty":2,"nm":"우리만","sr":1,"st":0,"op":219,"ip":0,"ln":"50","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100.862],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":0},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,150],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"21","ind":1},{"ty":2,"nm":"우리만_s","sr":1,"st":0,"op":219,"ip":0,"ln":"49","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100.385],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":0},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,98.182],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":0},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"22","ind":2}]},{"nm":"못생기게","id":"9","fr":24,"layers":[{"ty":2,"nm":"Element-6.png","sr":1,"st":0,"op":219,"ip":0,"ln":"195","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"10","ind":1},{"ty":2,"nm":"Element_S-6.png","sr":1,"st":0,"op":219,"ip":0,"ln":"194","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"11","ind":2}]},{"id":"10","e":1,"w":600,"h":88,"p":"","u":""},{"id":"11","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"아름다운 풍경","id":"12","fr":24,"layers":[{"ty":2,"nm":"Element-5.png","sr":1,"st":0,"op":219,"ip":0,"ln":"168","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"13","ind":1},{"ty":2,"nm":"Element_S-5.png","sr":1,"st":0,"op":219,"ip":0,"ln":"167","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"14","ind":2}]},{"id":"13","e":1,"w":600,"h":88,"p":"","u":""},{"id":"14","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"흔들렸지만","id":"15","fr":24,"layers":[{"ty":2,"nm":"Element-4.png","sr":1,"st":0,"op":219,"ip":0,"ln":"142","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"16","ind":1},{"ty":2,"nm":"Element_S-4.png","sr":1,"st":0,"op":219,"ip":0,"ln":"141","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"17","ind":2}]},{"id":"16","e":1,"w":600,"h":88,"p":"","u":""},{"id":"17","e":1,"w":480,"h":72,"p":"","u":""},{"nm":"예상치 못한","id":"18","fr":24,"layers":[{"ty":2,"nm":"Element-3.png","sr":1,"st":0,"op":219,"ip":0,"ln":"117","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[300,44]},"s":{"a":1,"k":[{"s":[117,117,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[150,150,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[117,117,100],"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[0],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":53.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":82.999},{"s":[0],"t":89.999}]}},"refId":"19","ind":1},{"ty":2,"nm":"Element_S-3.png","sr":1,"st":0,"op":219,"ip":0,"ln":"116","hasMask":false,"ao":0,"ks":{"a":{"a":0,"k":[240,36]},"s":{"a":1,"k":[{"s":[145,145,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":48},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":62.999},{"s":[186,186,100],"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":77.999},{"s":[145,145,100],"i":{"x":[0.833,0.833,0.833],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":89.999}]},"p":{"a":0,"k":[600,250]},"r":{"a":0,"k":0},"sa":{"a":0,"k":0},"o":{"a":1,"k":[{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":48},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":62.999},{"o":{"x":0.333,"y":0},"i":{"x":0.667,"y":1},"s":[100],"t":77.999},{"s":[100],"t":89.999}]}},"refId":"20","ind":2}]},{"id":"19","e":1,"w":600,"h":88,"p":"","u":""},{"id":"20","e":1,"w":480,"h":72,"p":"","u":""},{"id":"21","e":1,"w":600,"h":88,"p":"","u":""},{"id":"22","e":1,"w":480,"h":72,"p":"","u":""}]} \ No newline at end of file diff --git a/public/background/bg_qrcode.svg b/public/background/bg_qrcode.svg new file mode 100644 index 00000000..d5ebb9bb --- /dev/null +++ b/public/background/bg_qrcode.svg @@ -0,0 +1,88 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 00000000..39069967 Binary files /dev/null and b/public/favicon.ico differ diff --git a/public/file.svg b/public/file.svg new file mode 100644 index 00000000..004145cd --- /dev/null +++ b/public/file.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/fonts/PretendardVariable.woff2 b/public/fonts/PretendardVariable.woff2 new file mode 100644 index 00000000..49c54b51 Binary files /dev/null and b/public/fonts/PretendardVariable.woff2 differ diff --git a/public/globe.svg b/public/globe.svg new file mode 100644 index 00000000..567f17b0 --- /dev/null +++ b/public/globe.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/icon/error-image.svg b/public/icon/error-image.svg new file mode 100644 index 00000000..276b6ed8 --- /dev/null +++ b/public/icon/error-image.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/public/icon/icon_kakao.png b/public/icon/icon_kakao.png new file mode 100644 index 00000000..ac83a157 Binary files /dev/null and b/public/icon/icon_kakao.png differ diff --git a/public/next.svg b/public/next.svg new file mode 100644 index 00000000..5174b28c --- /dev/null +++ b/public/next.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/og/default_og.png b/public/og/default_og.png new file mode 100644 index 00000000..9e36b57e Binary files /dev/null and b/public/og/default_og.png differ diff --git a/public/og/invite_og.png b/public/og/invite_og.png new file mode 100644 index 00000000..ffe2452b Binary files /dev/null and b/public/og/invite_og.png differ diff --git a/public/og/toMaker_og.png b/public/og/toMaker_og.png new file mode 100644 index 00000000..51868fc0 Binary files /dev/null and b/public/og/toMaker_og.png differ diff --git a/public/robots.txt b/public/robots.txt new file mode 100644 index 00000000..28a33c28 --- /dev/null +++ b/public/robots.txt @@ -0,0 +1,10 @@ +# * +User-agent: * +Allow: / + +# Disallow private pages +Disallow: /mypage/ +Disallow: /onboarding/ +Disallow: /album/detail/ +# Sitemap +Sitemap: https://say-cheese.me/sitemap.xml diff --git a/public/test/4cut-example.png b/public/test/4cut-example.png new file mode 100644 index 00000000..ed2f9e88 Binary files /dev/null and b/public/test/4cut-example.png differ diff --git a/public/ut/1.jpg b/public/ut/1.jpg new file mode 100644 index 00000000..a4b87ece Binary files /dev/null and b/public/ut/1.jpg differ diff --git a/public/ut/10.jpg b/public/ut/10.jpg new file mode 100644 index 00000000..43e7e8a7 Binary files /dev/null and b/public/ut/10.jpg differ diff --git a/public/ut/11.jpg b/public/ut/11.jpg new file mode 100644 index 00000000..35af5229 Binary files /dev/null and b/public/ut/11.jpg differ diff --git "a/public/ut/1\354\243\274\354\260\250_1.jpg" "b/public/ut/1\354\243\274\354\260\250_1.jpg" new file mode 100644 index 00000000..f67e6770 Binary files /dev/null and "b/public/ut/1\354\243\274\354\260\250_1.jpg" differ diff --git "a/public/ut/1\354\243\274\354\260\250_2.jpg" "b/public/ut/1\354\243\274\354\260\250_2.jpg" new file mode 100644 index 00000000..8fe2c889 Binary files /dev/null and "b/public/ut/1\354\243\274\354\260\250_2.jpg" differ diff --git "a/public/ut/1\354\243\274\354\260\250_3.jpg" "b/public/ut/1\354\243\274\354\260\250_3.jpg" new file mode 100644 index 00000000..f10fa012 Binary files /dev/null and "b/public/ut/1\354\243\274\354\260\250_3.jpg" differ diff --git "a/public/ut/1\354\243\274\354\260\250_4.jpg" "b/public/ut/1\354\243\274\354\260\250_4.jpg" new file mode 100644 index 00000000..4a29deb3 Binary files /dev/null and "b/public/ut/1\354\243\274\354\260\250_4.jpg" differ diff --git a/public/ut/2.jpg b/public/ut/2.jpg new file mode 100644 index 00000000..6cafbd9e Binary files /dev/null and b/public/ut/2.jpg differ diff --git "a/public/ut/2\354\243\274\354\260\250_1.jpg" "b/public/ut/2\354\243\274\354\260\250_1.jpg" new file mode 100644 index 00000000..ab8bfe54 Binary files /dev/null and "b/public/ut/2\354\243\274\354\260\250_1.jpg" differ diff --git "a/public/ut/2\354\243\274\354\260\250_2.jpg" "b/public/ut/2\354\243\274\354\260\250_2.jpg" new file mode 100644 index 00000000..01fe3985 Binary files /dev/null and "b/public/ut/2\354\243\274\354\260\250_2.jpg" differ diff --git "a/public/ut/2\354\243\274\354\260\250_3.jpg" "b/public/ut/2\354\243\274\354\260\250_3.jpg" new file mode 100644 index 00000000..01fe3985 Binary files /dev/null and "b/public/ut/2\354\243\274\354\260\250_3.jpg" differ diff --git "a/public/ut/2\354\243\274\354\260\250_4.jpg" "b/public/ut/2\354\243\274\354\260\250_4.jpg" new file mode 100644 index 00000000..a5d681a8 Binary files /dev/null and "b/public/ut/2\354\243\274\354\260\250_4.jpg" differ diff --git a/public/ut/3.jpg b/public/ut/3.jpg new file mode 100644 index 00000000..97a0b32f Binary files /dev/null and b/public/ut/3.jpg differ diff --git "a/public/ut/3\354\243\274\354\260\250_1.jpg" "b/public/ut/3\354\243\274\354\260\250_1.jpg" new file mode 100644 index 00000000..545a289d Binary files /dev/null and "b/public/ut/3\354\243\274\354\260\250_1.jpg" differ diff --git "a/public/ut/3\354\243\274\354\260\250_2.jpg" "b/public/ut/3\354\243\274\354\260\250_2.jpg" new file mode 100644 index 00000000..f1144c0d Binary files /dev/null and "b/public/ut/3\354\243\274\354\260\250_2.jpg" differ diff --git "a/public/ut/3\354\243\274\354\260\250_3.jpg" "b/public/ut/3\354\243\274\354\260\250_3.jpg" new file mode 100644 index 00000000..9427ae76 Binary files /dev/null and "b/public/ut/3\354\243\274\354\260\250_3.jpg" differ diff --git a/public/ut/4.jpg b/public/ut/4.jpg new file mode 100644 index 00000000..a2ee9440 Binary files /dev/null and b/public/ut/4.jpg differ diff --git a/public/ut/5.jpg b/public/ut/5.jpg new file mode 100644 index 00000000..140df0d9 Binary files /dev/null and b/public/ut/5.jpg differ diff --git a/public/ut/6.jpg b/public/ut/6.jpg new file mode 100644 index 00000000..575a2813 Binary files /dev/null and b/public/ut/6.jpg differ diff --git a/public/ut/7.jpg b/public/ut/7.jpg new file mode 100644 index 00000000..ecce28c2 Binary files /dev/null and b/public/ut/7.jpg differ diff --git a/public/ut/8.jpg b/public/ut/8.jpg new file mode 100644 index 00000000..ba8d133c Binary files /dev/null and b/public/ut/8.jpg differ diff --git a/public/ut/9.jpg b/public/ut/9.jpg new file mode 100644 index 00000000..491e5a65 Binary files /dev/null and b/public/ut/9.jpg differ diff --git a/public/vercel.svg b/public/vercel.svg new file mode 100644 index 00000000..77053960 --- /dev/null +++ b/public/vercel.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/window.svg b/public/window.svg new file mode 100644 index 00000000..b2b2a44f --- /dev/null +++ b/public/window.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/scripts/generate-color-token.mjs b/scripts/generate-color-token.mjs new file mode 100644 index 00000000..04d57b3e --- /dev/null +++ b/scripts/generate-color-token.mjs @@ -0,0 +1,626 @@ +import { readFile, writeFile } from 'node:fs/promises'; +import { resolve } from 'node:path'; + +function toKebabCase(input) { + return String(input) + .replaceAll(/[^a-zA-Z0-9]+/g, '-') + .replaceAll(/([a-z0-9])([A-Z])/g, '$1-$2') + .toLowerCase() + .replaceAll(/-{2,}/g, '-') + .replaceAll(/^-|-$/g, ''); +} + +function isColorToken(node) { + return ( + node && + typeof node === 'object' && + 'value' in node && + 'type' in node && + node.type === 'color' + ); +} + +function isTypedToken(node) { + return node && typeof node === 'object' && 'value' in node && 'type' in node; +} + +function isPlainObject(value) { + return value && typeof value === 'object' && !Array.isArray(value); +} + +function flattenColorTokens( + root, + pathParts = [], + out = [], + pathToVarMap = new Map(), + refPrefixSegments = [], + refParts = [], +) { + if (!root || typeof root !== 'object') return { tokens: out, pathToVarMap }; + + if (isColorToken(root)) { + const variableName = `--${toKebabCase(pathParts.join('-'))}`; + out.push({ name: variableName, value: root.value }); + // Save mapping from dotted path (e.g. Color.Blue.500) to variable name (e.g. --color-blue-500) + const dottedKey = pathParts.join('.'); + pathToVarMap.set(dottedKey, variableName); + return { tokens: out, pathToVarMap }; + } + + for (const [key, value] of Object.entries(root)) { + const nextPath = [...pathParts, toKebabCase(key)]; + // Track JSON reference path using original keys only (no kebab-case) + const nextRefParts = [...refParts, key]; + if (isColorToken(value)) { + const varName = `--${toKebabCase(nextPath.join('-'))}`; + out.push({ name: varName, value: value.value }); + const refKey = [...refPrefixSegments, ...nextRefParts].join('.'); + const canonicalRefKey = refKey + .split('.') + .map((part) => part.replaceAll(' ', '')) + .join('.'); + pathToVarMap.set(refKey, varName); + pathToVarMap.set(canonicalRefKey, varName); + } else if (value && typeof value === 'object') { + flattenColorTokens( + value, + nextPath, + out, + pathToVarMap, + refPrefixSegments, + nextRefParts, + ); + } + } + return { tokens: out, pathToVarMap }; +} + +function flattenSemanticTokens(root, pathParts = [], out = []) { + if (!root || typeof root !== 'object') return out; + if (isTypedToken(root)) { + out.push({ path: pathParts, value: root.value, type: root.type }); + return out; + } + for (const [key, value] of Object.entries(root)) { + flattenSemanticTokens(value, [...pathParts, toKebabCase(key)], out); + } + return out; +} + +function resolveReferenceValue(value, pathToVarMap) { + if (typeof value !== 'string') return null; + const match = value.match(/^\{([^}]+)\}$/); + if (!match) return null; + let ref = match[1]; // e.g., Color.Blue.500 + // Try direct key first + if (pathToVarMap.has(ref)) return `var(${pathToVarMap.get(ref)})`; + // Try normalized key (remove spaces and normalize case of first segment) + const parts = ref.split('.'); + if (parts.length > 0) parts[0] = parts[0].trim(); + const canonical = parts.map((p) => p.replaceAll(' ', '')).join('.'); + if (pathToVarMap.has(canonical)) return `var(${pathToVarMap.get(canonical)})`; + // Last attempt: lower-cased, kebab combined map + const lower = parts.map((p) => p.toLowerCase()).join('.'); + if (pathToVarMap.has(lower)) return `var(${pathToVarMap.get(lower)})`; + return null; +} + +// Map common weight names to numeric values + +function toPx(value) { + if (typeof value === 'number') return `${value}px`; + if (typeof value === 'string') + return /px$/.test(value) ? value : `${value}px`; + return String(value); +} + +function collectTextPrimitives(primitiveRoot) { + const textRoot = {}; + if (!isPlainObject(primitiveRoot)) return textRoot; + + const fontRoot = primitiveRoot.font; + if (!isPlainObject(fontRoot)) return textRoot; + + // Extract font family + const fontFamily = fontRoot.family?.pretendard?.value; + if (fontFamily) { + textRoot.font = { fontFamily }; + } + + // Extract font weights + const fontWeights = {}; + if (isPlainObject(fontRoot.weight)) { + for (const [wKey, wVal] of Object.entries(fontRoot.weight)) { + if (isTypedToken(wVal)) { + fontWeights[wKey] = wVal.value; + } + } + } + + // Extract sizes and line heights + const sizes = fontRoot.size || {}; + const lineHeights = fontRoot['line height'] || {}; + const letterSpacings = fontRoot['letter spacing'] || {}; + + // Create font tokens for each size + for (const [sizeKey, sizeObj] of Object.entries(sizes)) { + if (!isTypedToken(sizeObj)) continue; + + const sizeValue = sizeObj.value; + const lineHeightObj = lineHeights[sizeKey]; + const letterSpacingObj = letterSpacings[sizeKey]; + + const sizeName = `size-${sizeKey}`; + textRoot[sizeName] = { + fontSize: toPx(sizeValue), + }; + + if (isTypedToken(lineHeightObj)) { + textRoot[sizeName].lineHeight = toPx(lineHeightObj.value); + } + + if (isTypedToken(letterSpacingObj)) { + textRoot[sizeName].letterSpacing = toPx(letterSpacingObj.value); + } + } + + // Add font weights to textRoot + if (Object.keys(fontWeights).length > 0) { + textRoot.weight = { fontWeight: fontWeights }; + } + + return textRoot; +} + +function buildTextRootCss(textRoot) { + const lines = []; + for (const [groupName, groupVal] of Object.entries(textRoot)) { + if (groupName === 'font' && groupVal.fontFamily) { + lines.push(` --font-family-primary: ${groupVal.fontFamily};`); + } + if (groupName === 'weight' && groupVal.fontWeight) { + for (const [wKey, wVal] of Object.entries(groupVal.fontWeight)) { + lines.push(` --font-weight-${wKey}: ${wVal};`); + } + } + if (groupName.startsWith('size-')) { + const sizeKey = groupName.replace('size-', ''); + if (groupVal.fontSize) { + lines.push(` --font-size-${sizeKey}: ${groupVal.fontSize};`); + } + if (groupVal.lineHeight) { + lines.push(` --line-height-${sizeKey}: ${groupVal.lineHeight};`); + } + if (groupVal.letterSpacing) { + lines.push(` --letter-spacing-${sizeKey}: ${groupVal.letterSpacing};`); + } + } + } + return lines; +} + +function buildTextThemeCss(textRoot) { + const lines = []; + for (const [groupName, groupVal] of Object.entries(textRoot)) { + if (groupName === 'font' && groupVal.fontFamily) { + lines.push(` --font-primary: var(--font-family-primary);`); + } + if (groupName.startsWith('size-')) { + const sizeKey = groupName.replace('size-', ''); + if (groupVal.fontSize) { + lines.push(` --text-${sizeKey}: var(--font-size-${sizeKey});`); + } + if (groupVal.lineHeight) { + lines.push(` --leading-${sizeKey}: var(--line-height-${sizeKey});`); + } + if (groupVal.letterSpacing) { + lines.push( + ` --tracking-${sizeKey}: var(--letter-spacing-${sizeKey});`, + ); + } + } + } + return lines; +} + +// 새로운 typography utility 생성: token.json의 title/body/heading/caption 등 스타일 네이밍 기반 +function buildTypographyUtilitiesCssFromToken(token) { + const lines = []; + const start = '/* generated-typography-utilities:start */'; + const end = '/* generated-typography-utilities:end */'; + lines.push(start); + + // 지원하는 스타일 그룹 (global에서) + const styleGroups = ['title', 'heading', 'body', 'caption']; + + // fontSize/weight 매핑용 + const sizeMap = { + 0: '12', + 1: '13', + 2: '14', + 3: '15', + 4: '16', + 5: '17', + 6: '18', + 7: '20', + 8: '24', + 9: '28', + 10: '32', + 11: '36', + 12: '40', + 13: '44', + }; + + const weightMap = { + 'pretendard-0': '700', + 'pretendard-1': '600', + 'pretendard-2': '500', + 'pretendard-3': '400', + }; + + // 그룹별 유틸리티 생성 + for (const group of styleGroups) { + const groupObj = token[group]; + + if (!groupObj) continue; + + for (const sizeKey of Object.keys(groupObj)) { + const sizeObj = groupObj[sizeKey]; + for (const weightKey of Object.keys(sizeObj)) { + const styleObj = sizeObj[weightKey]; + if (!styleObj || styleObj.type !== 'typography') continue; + + const value = styleObj.value; + + // 클래스명: typo-{group}-{sizeKey}-{weightKey} + // ex) @utility typo-body-lg-semibold { … } + const utilityName = `@utility typo-${group}-${sizeKey}-${weightKey} {`; + lines.push(utilityName); + + // font-family + if (value.fontFamily) { + lines.push(` font-family: var(--font-primary);`); + } + + // font-weight + let weightValue = '400'; + if (value.fontWeight) { + const weightRef = value.fontWeight; + if (weightRef.includes('pretendard-0')) weightValue = '700'; + else if (weightRef.includes('pretendard-1')) weightValue = '600'; + else if (weightRef.includes('pretendard-2')) weightValue = '500'; + else if (weightRef.includes('pretendard-3')) weightValue = '400'; + lines.push(` font-weight: ${weightValue};`); + } + + // font-size + let actualSize = '16'; + if (value.fontSize) { + const sizeRef = value.fontSize; + const sizeMatch = sizeRef.match(/fontSize\.(\d+)/); + if (sizeMatch) { + actualSize = sizeMap[sizeMatch[1]] || '16'; + lines.push(` font-size: var(--font-size-${actualSize});`); + } + } + + // letter-spacing + if (value.letterSpacing && !value.letterSpacing.includes('0')) { + lines.push(` letter-spacing: var(--letter-spacing-0);`); + } + + // line-height + if (value.lineHeight) { + const lineRef = value.lineHeight.match(/(\d+)/); + if (lineRef) { + lines.push(` line-height: var(--line-height-${lineRef[1]});`); + } + } + + lines.push('}'); + lines.push(''); // 줄바꿈 + } + } + } + + // dropshadow 스타일 처리 + for (const key of Object.keys(token)) { + const shadowObj = token[key]; + + if ( + shadowObj.type === 'boxShadow' && + shadowObj.value.type === 'dropShadow' + ) { + const value = shadowObj.value; + const shadowKey = key; // dropshadow-{blur}-{spread} + + // 클래스명: drop-shadow-{blur}-{spread} + const shadowUtility = `@utility drop-shadow-${value.blur}-${value.spread} {`; + lines.push(shadowUtility); + + // box-shadow 속성 추가 + const boxShadowValue = `${value.x}px ${value.y}px ${value.blur}px ${value.spread}px ${value.color}`; + lines.push(` box-shadow: ${boxShadowValue};`); + lines.push('}'); + lines.push(''); + } + } + + lines.push(end); + return lines; +} + +async function main() { + const projectRoot = process.cwd(); + const inputPath = resolve(projectRoot, 'src/app/token.json'); + const globalsPath = resolve(projectRoot, 'src/app/globals.css'); + + const raw = await readFile(inputPath, 'utf8'); + const tokens = JSON.parse(raw); + + const primitiveMode = tokens['primitive/Mode 1'] || {}; + const primitive = primitiveMode?.color; + if (!primitive) { + console.error( + 'No primitive color tokens found at "primitive/Mode 1" → "color"', + ); + } + + const { tokens: colorVars, pathToVarMap } = flattenColorTokens( + primitive, + ['color'], + [], + new Map(), + ['color'], + [], + ); + + // Use semantic/Mode 1 as the main semantic tokens + const semanticRoot = tokens['semantic/Mode 1']; + const semanticTokens = semanticRoot + ? flattenSemanticTokens(semanticRoot) + : []; + + // Collect text primitives from primitive/Mode 1 (excluding Color) + const textPrimitives = collectTextPrimitives(primitiveMode); + + // typography utility를 token.json의 스타일 네이밍 기반으로 생성 + const typographyUtilityLines = buildTypographyUtilitiesCssFromToken( + tokens.global || {}, + ); + + // Read globals.css to inject tokens + let globalsCss = await readFile(globalsPath, 'utf8'); + + // Remove any tokens.css import if exists + globalsCss = globalsCss.replace( + /\n?@import\s+["']\.\.\/styles\/tokens\.css["'];?\n?/g, + '\n', + ); + + // Helper to find block range by selector start index + function findBlockRange(source, startIndexOfBrace) { + let depth = 0; + for (let i = startIndexOfBrace; i < source.length; i++) { + const ch = source[i]; + if (ch === '{') depth++; + else if (ch === '}') { + depth--; + if (depth === 0) { + return { start: startIndexOfBrace, end: i }; + } + } + } + return null; + } + + // Locate top-level :root block (first occurrence) + const rootSelRegex = /:root\s*\{/g; + const rootSelMatch = rootSelRegex.exec(globalsCss); + if (!rootSelMatch) { + throw new Error('Could not find :root block in src/app/globals.css'); + } + const rootBlock = findBlockRange( + globalsCss, + rootSelMatch.index + rootSelMatch[0].length - 1, + ); + if (!rootBlock) throw new Error('Failed to parse :root block'); + + // Prepare primitive color tokens content + const primitiveStart = '/* generated-color-tokens:start */'; + const primitiveEnd = '/* generated-color-tokens:end */'; + const primitiveLines = [primitiveStart]; + for (const { name, value } of colorVars) { + primitiveLines.push(` ${name}: ${value};`); + } + primitiveLines.push(primitiveEnd); + const primitiveBlock = '\n' + primitiveLines.join('\n') + '\n'; + + // Prepare primitive text tokens content + const textStart = '/* generated-text-tokens:start */'; + const textEnd = '/* generated-text-tokens:end */'; + const textLines = [textStart, ...buildTextRootCss(textPrimitives), textEnd]; + const textBlock = '\n' + textLines.join('\n') + '\n'; + + // Replace or insert into :root block + const rootContent = globalsCss.slice(rootBlock.start + 1, rootBlock.end); + const existingPrimitiveRe = new RegExp( + primitiveStart.replace(/[/*]/g, (m) => `\\${m}`) + + '[\\s\\S]*?' + + primitiveEnd.replace(/[/*]/g, (m) => `\\${m}`), + 'm', + ); + let newRootContent; + if (existingPrimitiveRe.test(rootContent)) { + newRootContent = rootContent.replace( + existingPrimitiveRe, + primitiveBlock.trim(), + ); + } else { + newRootContent = rootContent.replace(/\n\s*\}$/, '') + primitiveBlock + '}'; + // The '}' will be re-added when we reconstruct full content below; ensure no duplicate + newRootContent = newRootContent.replace(/\}\s*$/, ''); + } + + // Reconstruct globals with updated :root + globalsCss = + globalsCss.slice(0, rootBlock.start + 1) + + newRootContent + + globalsCss.slice(rootBlock.end); + + // Inject or replace text tokens in :root + const rootSelMatch2 = /:root\s*\{/g.exec(globalsCss); + if (!rootSelMatch2) throw new Error(':root disappeared unexpectedly'); + const rootBlock2 = findBlockRange( + globalsCss, + rootSelMatch2.index + rootSelMatch2[0].length - 1, + ); + const rootContent2 = globalsCss.slice(rootBlock2.start + 1, rootBlock2.end); + const existingTextRe = new RegExp( + textStart.replace(/[/*]/g, (m) => `\\${m}`) + + '[\\s\\S]*?' + + textEnd.replace(/[/*]/g, (m) => `\\${m}`), + 'm', + ); + let newRootContent2; + if (existingTextRe.test(rootContent2)) { + newRootContent2 = rootContent2.replace(existingTextRe, textBlock.trim()); + } else { + newRootContent2 = rootContent2.replace(/\n\s*\}$/, '') + textBlock + '}'; + newRootContent2 = newRootContent2.replace(/\}\s*$/, ''); + } + globalsCss = + globalsCss.slice(0, rootBlock2.start + 1) + + newRootContent2 + + globalsCss.slice(rootBlock2.end); + + // Locate @theme inline block + const themeRegex = /@theme\s+inline\s*\{/g; + const themeMatch = themeRegex.exec(globalsCss); + if (!themeMatch) { + throw new Error( + 'Could not find "@theme inline {" block in src/app/globals.css', + ); + } + const themeBlock = findBlockRange( + globalsCss, + themeMatch.index + themeMatch[0].length - 1, + ); + if (!themeBlock) throw new Error('Failed to parse @theme inline block'); + + // Prepare semantic tokens (without 'semantic' prefix) to go into @theme inline + const semanticStart = '/* generated-semantic-tokens:start */'; + const semanticEnd = '/* generated-semantic-tokens:end */'; + const semanticLines = [semanticStart]; + for (const { path, value, type } of semanticTokens) { + if (type !== 'color') continue; + const varName = `--${toKebabCase(path.join('-'))}`; // e.g., --color-bg-interactive-primary + const refValue = resolveReferenceValue(value, pathToVarMap); + const finalValue = refValue ?? value; + semanticLines.push(` ${varName}: ${finalValue};`); + } + semanticLines.push(semanticEnd); + const semanticBlock = '\n' + semanticLines.join('\n') + '\n'; + + // Prepare typography theme tokens + const typoStart = '/* generated-typography-tokens:start */'; + const typoEnd = '/* generated-typography-tokens:end */'; + const typoLines = [typoStart, ...buildTextThemeCss(textPrimitives), typoEnd]; + const typoBlock = '\n' + typoLines.join('\n') + '\n'; + + const themeContent = globalsCss.slice(themeBlock.start + 1, themeBlock.end); + const existingSemanticRe = new RegExp( + semanticStart.replace(/[/*]/g, (m) => `\\${m}`) + + '[\\s\\S]*?' + + semanticEnd.replace(/[/*]/g, (m) => `\\${m}`), + 'm', + ); + let newThemeContent; + if (existingSemanticRe.test(themeContent)) { + newThemeContent = themeContent.replace( + existingSemanticRe, + semanticBlock.trim(), + ); + } else { + newThemeContent = + themeContent.replace(/\n\s*\}$/, '') + semanticBlock + '}'; + newThemeContent = newThemeContent.replace(/\}\s*$/, ''); + } + + globalsCss = + globalsCss.slice(0, themeBlock.start + 1) + + newThemeContent + + globalsCss.slice(themeBlock.end); + + // Inject or replace typography theme tokens + const themeMatch2 = /@theme\s+inline\s*\{/g.exec(globalsCss); + if (!themeMatch2) throw new Error('@theme inline disappeared unexpectedly'); + const themeBlock2 = findBlockRange( + globalsCss, + themeMatch2.index + themeMatch2[0].length - 1, + ); + const themeContent2 = globalsCss.slice( + themeBlock2.start + 1, + themeBlock2.end, + ); + const existingTypoRe = new RegExp( + typoStart.replace(/[/*]/g, (m) => `\\${m}`) + + '[\\s\\S]*?' + + typoEnd.replace(/[/*]/g, (m) => `\\${m}`), + 'm', + ); + let newThemeContent2; + if (existingTypoRe.test(themeContent2)) { + newThemeContent2 = themeContent2.replace(existingTypoRe, typoBlock.trim()); + } else { + newThemeContent2 = themeContent2.replace(/\n\s*\}$/, '') + typoBlock + '}'; + newThemeContent2 = newThemeContent2.replace(/\}\s*$/, ''); + } + + globalsCss = + globalsCss.slice(0, themeBlock2.start + 1) + + newThemeContent2 + + globalsCss.slice(themeBlock2.end); + + // Inject or replace typography utilities at file root + const utilStart = '/* generated-typography-utilities:start */'; + const utilEnd = '/* generated-typography-utilities:end */'; + const utilBlock = typographyUtilityLines.join('\n') + '\n'; + const utilRe = new RegExp( + utilStart.replace(/[/*]/g, (m) => `\\${m}`) + + '[\\s\\S]*?' + + utilEnd.replace(/[/*]/g, (m) => `\\${m}`), + 'm', + ); + if (utilRe.test(globalsCss)) { + globalsCss = globalsCss.replace(utilRe, utilBlock.trim()); + } else { + // Append at end of file with a preceding newline + globalsCss = globalsCss.replace(/\s*$/, '\n' + utilBlock); + } + + // body에 font-family 자동 생성/치환 + const bodyFontFamily = 'font-family: var(--font-pretendard);'; + const bodyRe = /body\s*\{[^}]*\}/m; + if (bodyRe.test(globalsCss)) { + globalsCss = globalsCss.replace(bodyRe, (match) => { + // 기존 font-family 라인 치환 또는 추가 + if (/font-family:[^;]+;/.test(match)) { + return match.replace(/font-family:[^;]+;/, bodyFontFamily); + } + // font-family가 없으면 추가 + return match.replace(/\}/, ` ${bodyFontFamily}\n}`); + }); + } else { + // body 블록이 없으면 새로 추가 + globalsCss += `\nbody {\n ${bodyFontFamily}\n}`; + } + + await writeFile(globalsPath, globalsCss, 'utf8'); + console.log(`Updated ${globalsPath}`); +} + +main().catch((err) => { + console.error(err); + process.exit(1); +}); diff --git a/scripts/generate-endpoint.ts b/scripts/generate-endpoint.ts new file mode 100644 index 00000000..e40297b1 --- /dev/null +++ b/scripts/generate-endpoint.ts @@ -0,0 +1,482 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ +// scripts/generate-ep.ts +// Usage: +// npx ts-node scripts/generate-ep.ts +// npx ts-node scripts/generate-ep.ts https://dev.say-cheese.me/v3/api-docs +// +// Output: +// src/api/ep.ts (EP only + PhotoSorting type) + +import fs from 'node:fs'; +import path from 'node:path'; +import process from 'node:process'; + +const SPEC_URL = process.argv[2] || 'https://dev.say-cheese.me/v3/api-docs'; +const OUT_FILE = path.join(process.cwd(), 'src/global/api/ep.ts'); + +type Route = { path: string; method: string }; + +// --- 그룹 매핑 (경로 prefix 기반 강제 매핑) --- +function groupKeyByPath(p: string): keyof typeof EP_SKELETON { + if (p.startsWith('/v1/global/')) return 'global'; + if (p.startsWith('/v1/auth/')) return 'auth'; + if (p.startsWith('/v1/user/')) return 'user'; + if (p.startsWith('/v1/cheese4cut/')) return 'cheese4cut'; + if (p.startsWith('/v1/album/')) return 'album'; + if (p.startsWith('/v1/photo/')) return 'photo'; + if (p.startsWith('/internal/')) return 'internal'; + // 기본값: album로 (실제 스펙 상 여기에 안걸림) + return 'album'; +} + +function desiredNameFor(pathStr: string, method: string): string { + // 1) 기존 정확 매핑 우선 + // ---------------------------------- + // Global + if (pathStr === '/v1/global/health-check') return 'health'; + + // Auth + if (pathStr === '/v1/auth/logout') return 'logout'; + if (pathStr === '/v1/auth/reissue') return 'reissue'; + if (pathStr === '/v1/auth/exchange') return 'exchange'; + + // User + if (pathStr === '/v1/user/agreement') return 'agreement'; + if (pathStr === '/v1/user/me/profile') return 'updateProfile'; + + // Album + if (pathStr === '/v1/album' && method === 'post') return 'create'; + if (/^\/v1\/album\/\{code\}\/enter$/.test(pathStr)) return 'enter'; + if (/^\/v1\/album\/\{code\}\/photos$/.test(pathStr)) return 'photos'; + if (/^\/v1\/album\/\{code\}\/photos\/\{photoId\}$/.test(pathStr)) + return 'photoDetail'; + if (/^\/v1\/album\/\{code\}\/photos\/liked$/.test(pathStr)) + return 'likedPhotos'; + if (/^\/v1\/album\/\{code\}\/participants$/.test(pathStr)) + return 'participants'; + if (/^\/v1\/album\/\{code\}\/invitation$/.test(pathStr)) return 'invitation'; + if (/^\/v1\/album\/\{code\}\/available-count$/.test(pathStr)) + return 'availableCount'; + + // Photo + if (/^\/v1\/photo\/\{photoId\}\/liked$/.test(pathStr)) return 'like'; + if (/^\/v1\/photo\/\{photoId\}\/unliked$/.test(pathStr)) return 'unlike'; + if (pathStr === '/v1/photo/report') return 'reportUploadResult'; + if (pathStr === '/v1/photo/presigned-url') return 'presignedUpload'; + if (pathStr === '/v1/photo/download-url') return 'presignedDownload'; + + // Cheese4cut + if (/^\/v1\/cheese4cut\/\{code\}\/presigned-url$/.test(pathStr)) + return 'presignedUpload'; + if (/^\/v1\/cheese4cut\/\{code\}\/fixed$/.test(pathStr)) return 'finalize'; + if (/^\/v1\/cheese4cut\/\{code\}\/preview$/.test(pathStr)) return 'preview'; + + // Internal + if (pathStr === '/internal/thumbnail/complete') return 'thumbnailComplete'; + + // 2) 여기부터 자동 fallback 생성 + // ---------------------------------- + return autoName(pathStr); +} + +function autoName(pathStr: string): string { + const parts = pathStr + .split('/') + .filter(Boolean) + .filter((p) => !p.startsWith('{')) // 경로 파라미터 제거 + .filter((p) => !/^v\d+$/.test(p)); // v1, v2 같은 prefix 제거 + + // 하이픈(-)을 카멜케이스로 변환 + const toCamel = (str: string) => + str.replace(/-([a-zA-Z])/g, (_, c) => c.toUpperCase()); + + const camel = parts + .map((p, i) => { + const camelPart = toCamel(p); + return i === 0 + ? camelPart.toLowerCase() + : camelPart.charAt(0).toUpperCase() + camelPart.slice(1); + }) + .join(''); + + return camel || 'unknown'; +} + +function paramsSignatureFor(pathStr: string): string { + const hasCode = pathStr.includes('{code}'); + const hasPhotoId = pathStr.includes('{photoId}'); + + // 1) code + photoId 둘 다 있을 때 + if (hasCode && hasPhotoId) { + return '(code: string, photoId: number)'; + } + + // 2) code만 있을 때 + if (hasCode) { + return '(code: string)'; + } + + // 3) photoId만 있을 때 + if (hasPhotoId) { + return '(photoId: number)'; + } + + return '()'; +} + +// --- 바디: 템플릿 문자열 경로 --- +function pathTemplateToTs(pathStr: string): string { + // {param} -> ${param} + return '`' + pathStr.replace(/{([^}]+)}/g, '${$1}') + '`'; +} + +// EP 골격 (키 순서 고정) +const EP_SKELETON = { + global: [] as Route[], + auth: [] as Route[], + user: [] as Route[], + album: [] as Route[], + photo: [] as Route[], + cheese4cut: [] as Route[], + internal: [] as Route[], +}; + +function sortByDesiredOrder(routes: Route[]) { + // 그냥 path 기준 정렬(안정적) + return routes.slice().sort((a, b) => a.path.localeCompare(b.path)); +} + +function renderGroup(name: keyof typeof EP_SKELETON, routes: Route[]): string { + const lines: string[] = []; + + sortByDesiredOrder(routes).forEach(({ path: p, method }) => { + const fn = desiredNameFor(p, method); + if (!fn) return; + + const sig = paramsSignatureFor(p); + const pathExpr = pathTemplateToTs(p); + + // 🔥 key 항상 문자열 처리 + lines.push(` "${fn}": ${sig} => ${pathExpr},`); + }); + + return ` ${name}: {\n${lines.join('\n')}\n }`; +} + +// ...위쪽 동일 + +type Schema = + | { + type?: string; + enum?: any[]; + items?: Schema; + properties?: Record; + required?: string[]; + allOf?: Schema[]; + oneOf?: Schema[]; + anyOf?: Schema[]; + $ref?: string; + additionalProperties?: boolean | Schema; + description?: string; + } + | any; + +// [ADD] OpenAPI 확장 +type OpenAPI = { + paths: Record< + string, + Record< + string, + { + operationId?: string; + tags?: string[]; + summary?: string; + description?: string; + parameters?: Array<{ + name: string; + in: 'path' | 'query' | 'header' | 'cookie'; + required?: boolean; + schema?: Schema; + }>; + requestBody?: { + content?: Record; + }; + responses?: Record< + string, + { + description?: string; + content?: Record; + } + >; + } + > + >; + components?: { + schemas?: Record; + }; +}; + +// [ADD] 파스칼 케이스 유틸(안전하게 이름 정리) +function toPascalCase(s: string) { + return s + .replace(/[^a-zA-Z0-9]+/g, ' ') + .trim() + .split(' ') + .map((w) => w.charAt(0).toUpperCase() + w.slice(1)) + .join(''); +} + +// [ADD] Components 타입 이름 규칙 (Schema 접미사) +function componentTypeName(raw: string) { + return `${toPascalCase(raw)}Schema`; +} + +// [CHANGE] $ref 해석 시 components 타입명 치환 +function resolveRef(ref: string) { + // '#/components/schemas/Foo' 형태만 지원 + const parts = ref.split('/'); + const raw = parts[parts.length - 1]; + return componentTypeName(raw); // <-- 여기! +} + +// [CHANGE] components.schemas -> 타입 생성 시 이름 변경 +function generateComponentsTypes(components: Record): string { + const lines: string[] = []; + Object.entries(components).forEach(([name, schema]) => { + const ts = schemaToTs(schema, components, name); + const outName = componentTypeName(name); // <-- 여기! + + const isObjectLike = /^\{\s/.test(ts); + if (isObjectLike) { + lines.push(`export interface ${outName} ${ts}`); + } else { + lines.push(`export type ${outName} = ${ts};`); + } + }); + return lines.join('\n'); +} +// [ADD] 스키마 -> TS 변환기 (기본형/enum/array/object/$ref) +function schemaToTs( + schema: Schema | undefined, + components: Record, + inlineNameHint?: string, +): string { + if (!schema) return 'unknown'; + + // $ref + if (schema.$ref) { + return resolveRef(schema.$ref); + } + + // allOf (간단히 & 교차) + if (schema.allOf && schema.allOf.length > 0) { + return schema.allOf.map((s: any) => schemaToTs(s, components)).join(' & '); + } + + // oneOf/anyOf (간단히 union) + if ( + (schema.oneOf && schema.oneOf.length) || + (schema.anyOf && schema.anyOf.length) + ) { + const arr = (schema.oneOf ?? schema.anyOf)!; + return arr.map((s: any) => schemaToTs(s, components)).join(' | '); + } + + // enum + if (schema.enum && schema.enum.length) { + return schema.enum + .map((v: any) => (typeof v === 'string' ? JSON.stringify(v) : `${v}`)) + .join(' | '); + } + + const t = schema.type; + + if ( + t === 'string' || + t === 'number' || + t === 'integer' || + t === 'boolean' || + t === 'null' + ) { + if (t === 'integer') return 'number'; + return t === 'null' ? 'null' : t; + } + + if (t === 'array') { + const it = schema.items ? schemaToTs(schema.items, components) : 'unknown'; + return `${it}[]`; + } + + if (t === 'object' || schema.properties || schema.additionalProperties) { + const props = schema.properties ?? {}; + const req = new Set(schema.required ?? []); + const body: string[] = []; + + for (const [k, v] of Object.entries(props)) { + const optional = req.has(k) ? '' : '?'; + body.push( + `${JSON.stringify(k)}${optional}: ${schemaToTs(v, components, k)};`, + ); + } + + // index signature + if (schema.additionalProperties) { + const ap = + schema.additionalProperties === true + ? 'unknown' + : schemaToTs(schema.additionalProperties, components); + body.push(`[key: string]: ${ap};`); + } + + return `{ ${body.join(' ')} }`; + } + + // fallback + return 'unknown'; +} + +// [ADD] 응답 스키마 추출(200/201/2xx 우선) +function pickSuccessResponseSchema(op: any): Schema | undefined { + const res = op.responses || {}; + const codes = Object.keys(res).sort(); // 안정 + const prefer = [ + '200', + '201', + ...codes.filter((c) => /^2\d\d$/.test(c) && c !== '200' && c !== '201'), + ]; + for (const code of prefer) { + const content = res[code]?.content; + if (!content) continue; + // JSON 우선 + const jsonKey = + Object.keys(content).find((k) => k.includes('json')) || + Object.keys(content)[0]; + const sch = content[jsonKey]?.schema; + if (sch) return sch; + } + return undefined; +} + +// [ADD] 이름 규칙: 그룹+함수명 기반 Response 타입명 +function responseTypeName(group: string, fn: string) { + const cap = (s: string) => toPascalCase(s); + return `${cap(group)}${cap(fn)}Response`; +} + +// [ADD] 2패스: 타입 별칭/인터페이스 모아 쓰고, ApiReturns는 마지막에 +function buildAllOperationTypesAndApiReturns( + groups: typeof EP_SKELETON, + spec: OpenAPI, +) { + const components = spec.components?.schemas ?? {}; + const typeDecls: string[] = []; + const apiReturns: string[] = ['export interface ApiReturns {']; + + (Object.keys(EP_SKELETON) as Array).forEach( + (group) => { + const routes = sortByDesiredOrder(groups[group]); + routes.forEach(({ path: p, method }) => { + const fn = desiredNameFor(p, method); + if (!fn) return; + const op = spec.paths[p]?.[method]; + if (!op) return; + + const respSchema = pickSuccessResponseSchema(op); + if (!respSchema) return; + + const typeName = responseTypeName(group, fn); + const ts = schemaToTs(respSchema, components, typeName); + const isObjectLike = /^\{\s/.test(ts); + + if (isObjectLike) { + typeDecls.push(`export interface ${typeName} ${ts}["result"];`); + } else { + typeDecls.push(`export type ${typeName} = ${ts}["result"];`); + } + + apiReturns.push( + ` ${JSON.stringify(`${group}.${fn}`)}: ${typeName}; // ${method.toUpperCase()} ${p}`, + ); + }); + }, + ); + + apiReturns.push('}'); + return { typeDecls: typeDecls.join('\n'), apiReturns: apiReturns.join('\n') }; +} + +// --- main +async function main() { + console.log(`[generate-ep] Fetching spec: ${SPEC_URL}`); + const res = await fetch(SPEC_URL); + if (!res.ok) { + console.error( + `[generate-ep] Failed to fetch spec: ${res.status} ${res.statusText}`, + ); + process.exit(1); + } + + const spec = (await res.json()) as OpenAPI; + const groups = JSON.parse(JSON.stringify(EP_SKELETON)) as typeof EP_SKELETON; + + Object.entries(spec.paths).forEach(([p, methods]) => { + Object.keys(methods).forEach((m) => { + const method = m.toLowerCase(); + if ( + !['get', 'post', 'put', 'patch', 'delete', 'options', 'head'].includes( + method, + ) + ) + return; + const g = groupKeyByPath(p); + groups[g].push({ path: p, method }); + }); + }); + + // [ADD] components.shema 타입 생성 + const componentsTypes = generateComponentsTypes( + spec.components?.schemas ?? {}, + ); + + // [ADD] 각 오퍼레이션 응답 타입 + ApiReturns 인터페이스 + const { typeDecls, apiReturns } = buildAllOperationTypesAndApiReturns( + groups, + spec, + ); + + const content = `/* AUTO-GENERATED FILE. DO NOT EDIT. + * Generated by scripts/generate-ep.ts + */ + +export const EP = { +${(Object.keys(EP_SKELETON) as Array) + .map((g) => renderGroup(g, groups[g])) + .join(',\n')} +} as const; + +// 선택지 Enum(정렬) +export type PhotoSorting = 'POPULAR' | 'CAPTURED_AT' | 'CREATED_AT'; + +/* ======================= + * Generated Types + * ======================= */ + +// --- Components +${componentsTypes} + +// --- Operation Response Types +${typeDecls} + +// --- Mapping: 'group.fn' -> Response Type +${apiReturns} +`; + + fs.mkdirSync(path.dirname(OUT_FILE), { recursive: true }); + fs.writeFileSync(OUT_FILE, content, 'utf8'); + console.log(`[generate-ep] Wrote ${OUT_FILE}`); +} + +main().catch((e) => { + console.error(e); + process.exit(1); +}); diff --git a/src/app/album/4cut/[albumId]/page.tsx b/src/app/album/4cut/[albumId]/page.tsx new file mode 100644 index 00000000..5cfd97eb --- /dev/null +++ b/src/app/album/4cut/[albumId]/page.tsx @@ -0,0 +1,53 @@ +import ScreenAlbum4Cut from '@/feature/album/4cut/components/ScreenAlbum4Cut'; +import { ApiReturns, EP } from '@/global/api/ep'; +import { serverApi } from '@/global/utils/serverApi'; +import { Metadata, ResolvingMetadata } from 'next'; + +export async function generateMetadata( + { params }: PageProps, + parent: ResolvingMetadata, +): Promise { + let title = ''; + const { albumId } = await params; + try { + const { result } = await serverApi.get({ + path: EP.album.invitation(albumId), + }); + + if (result?.title) title = result.title; + } catch (e) { + console.error(e); + } + + return { + title: `치즈네컷이 궁금해요 | ${title}`, + description: '메이커님 네컷 확정해주세요 (*´ー`*人)', + openGraph: { + title: `치즈네컷이 궁금해요 | ${title}`, + description: '메이커님 네컷 확정해주세요 (*´ー`*人)', + url: `https://say-cheese.me/album/4cut/${albumId}`, + siteName: '치이이즈', + images: [ + { + url: '/og/toMaker_og.png', + width: 1200, + height: 630, + alt: '치이이즈. 우리 공유앨범에 초대합니다. 일주일 뒤에는 앨범이 사라져요!', + }, + ], + locale: 'ko_KR', + type: 'website', + }, + }; +} + +interface PageProps { + params: Promise<{ + albumId: string; + }>; +} + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + return ; +} diff --git a/src/app/album/[albumId]/select/page.tsx b/src/app/album/[albumId]/select/page.tsx new file mode 100644 index 00000000..83a6f8bc --- /dev/null +++ b/src/app/album/[albumId]/select/page.tsx @@ -0,0 +1,122 @@ +'use client'; + +import { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogTitle, +} from '@/components/ui/alert-dialog'; +import SelectAlbumBody from '@/feature/album-select/components/SelectAlbumBody'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import { useParams, useRouter } from 'next/navigation'; +import { useEffect, useRef, useState } from 'react'; + +export default function Page() { + const [modalOpen, setModalOpen] = useState(false); + const confirmedRef = useRef(false); + const router = useRouter(); + const params = useParams(); + const albumId = params?.albumId; + + // data-scroll-locked 속성 제거 + useEffect(() => { + const removeScrollLock = () => { + const body = document.body; + if (body.hasAttribute('data-scroll-locked')) { + body.removeAttribute('data-scroll-locked'); + } + }; + + removeScrollLock(); + const interval = setInterval(removeScrollLock, 100); + + return () => { + clearInterval(interval); + removeScrollLock(); + }; + }, []); + + // 뒤로가기(브라우저/하드웨어) 감지 + useEffect(() => { + const handler = () => { + if (modalOpen) { + setModalOpen(false); // 모달만 닫기 + } else if (!confirmedRef.current) { + setModalOpen(true); + history.pushState(null, '', location.href); // 뒤로가기 무효화 + } + }; + + // 초기 히스토리 상태 추가 + history.pushState(null, '', location.href); + window.addEventListener('popstate', handler); + + return () => { + window.removeEventListener('popstate', handler); + }; + }, [modalOpen]); + + const handleConfirm = () => { + confirmedRef.current = true; + setModalOpen(false); + if (albumId) { + window.location.replace('/main'); + } + }; + + const handleCancel = () => { + // 히스토리 상태 다시 추가만 수행 + history.pushState(null, '', location.href); + }; + + const handleOpenChange = (open: boolean) => { + setModalOpen(open); + if (!open) { + // 모달이 닫힐 때 (바깥 클릭, ESC 등) + handleCancel(); + } + }; + + return ( +
+ setModalOpen(true)} + /> + + + + + + + 앨범 채우기를 그만둘까요? + + + 사진을 앨범에 채우기 전이에요.{'\n'}지금 나가면 다시 사진을 + 불러와야 해요. + + + + + 취소 + + + 나가기 + + + + +
+ ); +} diff --git a/src/app/album/[albumId]/waiting/page.tsx b/src/app/album/[albumId]/waiting/page.tsx new file mode 100644 index 00000000..1db9cf46 --- /dev/null +++ b/src/app/album/[albumId]/waiting/page.tsx @@ -0,0 +1,12 @@ +import WaitingAlbum from '@/feature/create-album/components/WaitingAlbum'; + +type PageProps = { + params: Promise<{ + albumId: string; + }>; +}; + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + return ; +} diff --git a/src/app/album/detail/[albumId]/page.tsx b/src/app/album/detail/[albumId]/page.tsx new file mode 100644 index 00000000..23c984fd --- /dev/null +++ b/src/app/album/detail/[albumId]/page.tsx @@ -0,0 +1,13 @@ +import ScreenAlbumDetail from '@/feature/album/detail/components/ScreenAlbumDetail'; + +interface PageProps { + params: Promise<{ + albumId: string; + }>; +} + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + + return ; +} diff --git a/src/app/album/detail/[albumId]/sidebar/page.tsx b/src/app/album/detail/[albumId]/sidebar/page.tsx new file mode 100644 index 00000000..6faab89a --- /dev/null +++ b/src/app/album/detail/[albumId]/sidebar/page.tsx @@ -0,0 +1,12 @@ +import ScreenAlbumSidebar from '@/feature/album/detail/sidebar/components/ScreenAlbumSidebar'; + +interface PageProps { + params: Promise<{ + albumId: string; + }>; +} + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + return ; +} diff --git a/src/app/album/entry/[albumId]/page.tsx b/src/app/album/entry/[albumId]/page.tsx new file mode 100644 index 00000000..9ce46732 --- /dev/null +++ b/src/app/album/entry/[albumId]/page.tsx @@ -0,0 +1,52 @@ +import ScreenAlbumEntry from '@/feature/album-entry/components/ScreenAlbumEntry'; +import { ApiReturns, EP } from '@/global/api/ep'; +import { serverApi } from '@/global/utils/serverApi'; +import { Metadata, ResolvingMetadata } from 'next'; + +export async function generateMetadata( + { params }: PageProps, + parent: ResolvingMetadata, +): Promise { + let title = ''; + const { albumId } = await params; + try { + const { result } = await serverApi.get({ + path: EP.album.invitation(albumId), + }); + + if (result?.title) title = result.title; + } catch (e) { + console.log(e); + } + + return { + title: `${title} | 앨범에 초대해요`, + description: '치이이즈: 추억은 따끈할 때 제맛', + openGraph: { + title: `${title} | 앨범에 초대해요`, + description: '치이이즈: 추억은 따끈할 때 제맛', + url: `https://say-cheese.me/album/entry/${albumId}`, + siteName: '치이이즈', + images: [ + { + url: '/og/invite_og.png', + width: 1200, + height: 630, + alt: '치이이즈. 우리 공유앨범에 초대합니다. 일주일 뒤에는 앨범이 사라져요!', + }, + ], + locale: 'ko_KR', + type: 'website', + }, + }; +} + +interface PageProps { + params: Promise<{ albumId: string }>; +} + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + + return ; +} diff --git a/src/app/album/qrcode/[albumId]/page.tsx b/src/app/album/qrcode/[albumId]/page.tsx new file mode 100644 index 00000000..9d1f321a --- /dev/null +++ b/src/app/album/qrcode/[albumId]/page.tsx @@ -0,0 +1,12 @@ +import ScreenAlbumQrcode from '@/feature/album/qrcode/components/ScreenAlbumQrcode'; + +interface PageProps { + params: Promise<{ + albumId: string; + }>; +} + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + return ; +} diff --git a/src/app/album/upload/[albumId]/page.tsx b/src/app/album/upload/[albumId]/page.tsx new file mode 100644 index 00000000..bbdf11c7 --- /dev/null +++ b/src/app/album/upload/[albumId]/page.tsx @@ -0,0 +1,29 @@ +import UploadAlbumPage from '@/feature/upload/components/UploadAlbumPage'; +import { getAlbumDataWithRoleServer } from '@/feature/upload/hooks/useGetAlbumInform.server'; +import { EP } from '@/global/api/ep'; +import { + dehydrate, + HydrationBoundary, + QueryClient, +} from '@tanstack/react-query'; + +type PageProps = { + params: Promise<{ albumId: string }>; +}; + +export default async function Page({ params }: PageProps) { + const { albumId } = await params; + const queryClient = new QueryClient(); + await queryClient.prefetchQuery({ + queryKey: [EP.album.participants(albumId)], + queryFn: () => getAlbumDataWithRoleServer(albumId), + staleTime: 60 * 1000 * 10, + }); + + const dehydratedState = dehydrate(queryClient); + return ( + + + + ); +} diff --git a/src/app/create-album/[albumId]/complete/page.tsx b/src/app/create-album/[albumId]/complete/page.tsx new file mode 100644 index 00000000..795fe18f --- /dev/null +++ b/src/app/create-album/[albumId]/complete/page.tsx @@ -0,0 +1,10 @@ +import CreateComplete from '@/feature/create-album/components/CreateComplete'; + +export default async function Page({ + params, +}: { + params: Promise<{ albumId: string }>; +}) { + const { albumId } = await params; + return ; +} diff --git a/src/app/create-album/page.tsx b/src/app/create-album/page.tsx new file mode 100644 index 00000000..8022c959 --- /dev/null +++ b/src/app/create-album/page.tsx @@ -0,0 +1,19 @@ +'use client'; +import CreateAlbumList from '@/feature/create-album/components/CreateAlbumList'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import { useRouter } from 'next/navigation'; + +export default function Page() { + const router = useRouter(); + + return ( +
+ router.push('/main')} + /> + +
+ ); +} diff --git a/src/app/globals.css b/src/app/globals.css new file mode 100644 index 00000000..d59ae045 --- /dev/null +++ b/src/app/globals.css @@ -0,0 +1,715 @@ +@import 'tailwindcss'; +@import "tw-animate-css"; + +@custom-variant dark (&:is(.dark *)); + +:root { + + /* generated-color-tokens:start */ + --color-primary-25: #fffbeb; + --color-primary-50: #fff7d6; + --color-primary-100: #fff2c2; + --color-primary-200: #ffe894; + --color-primary-300: #ffdc5c; + --color-primary-400: #ffcd14; + --color-primary-500: #ffb700; + --color-primary-600: #e09900; + --color-primary-700: #996300; + --color-primary-800: #664200; + --color-primary-900: #332100; + --color-accent-25: #fff5f6; + --color-accent-50: #ffe5ea; + --color-accent-100: #ffccd4; + --color-accent-200: #ff99aa; + --color-accent-300: #ff6680; + --color-accent-400: #ff3355; + --color-accent-500: #ff002b; + --color-accent-600: #cc0022; + --color-accent-700: #99001a; + --color-accent-800: #660011; + --color-accent-900: #330009; + --color-success-25: #ebffeb; + --color-success-50: #c7ffc7; + --color-success-100: #a8ffa8; + --color-success-200: #7aff7a; + --color-success-300: #1aff1a; + --color-success-400: #00e500; + --color-success-500: #00c400; + --color-success-600: #009900; + --color-success-700: #006600; + --color-success-800: #004d00; + --color-success-900: #002900; + --color-neutral-0: #ffffff; + --color-neutral-25: #f7f7f8; + --color-neutral-50: #f1f2f3; + --color-neutral-100: #e5e5e7; + --color-neutral-200: #c9cacf; + --color-neutral-300: #afb0b7; + --color-neutral-400: #94969e; + --color-neutral-500: #747681; + --color-neutral-600: #56575f; + --color-neutral-700: #424349; + --color-neutral-800: #2c2c30; + --color-neutral-900: #18191b; + --color-alpha-white-white-100: #ffffff; + --color-alpha-white-white-80: #ffffffcc; + --color-alpha-white-white-50: #ffffff80; + --color-alpha-white-white-20: #ffffff33; + --color-alpha-white-white-0: #ffffff00; + --color-alpha-black-black-100: #18191b; + --color-alpha-black-black-75: #18191bcc; + --color-alpha-black-black-50: #18191b80; + --color-alpha-black-black-20: #18191b33; + --color-alpha-black-black-0: #18191b00; + --color-alpha-black-black-10: #18191b1a; + --color-alpha-primary-yellow-100: #ffcd14; + --color-alpha-primary-yellow-80: #ffcd14cc; + --color-alpha-primary-yellow-50: #ffcd1480; + --color-alpha-primary-yellow-20: #ffcd1433; + --color-alpha-primary-yellow-0: #ffcd1400; +/* generated-color-tokens:end */ + + /* generated-text-tokens:start */ + --font-family-primary: pretendard; + --font-size-12: 12px; + --font-size-13: 13px; + --font-size-14: 14px; + --font-size-15: 15px; + --font-size-16: 16px; + --font-size-17: 17px; + --font-size-18: 18px; + --line-height-18: 18px; + --font-size-20: 20px; + --line-height-20: 20px; + --font-size-24: 24px; + --line-height-24: 24px; + --font-size-28: 28px; + --line-height-28: 28px; + --font-size-32: 32px; + --font-size-36: 36px; + --line-height-36: 36px; + --font-size-40: 40px; + --font-size-44: 44px; + --font-weight-400: 400; + --font-weight-500: 500; + --font-weight-600: 600; + --font-weight-700: 700; +/* generated-text-tokens:end */ + --radius: 0.625rem; + --background: oklch(1 0 0); + --foreground: oklch(0.145 0 0); + --card: oklch(1 0 0); + --card-foreground: oklch(0.145 0 0); + --popover: oklch(1 0 0); + --popover-foreground: oklch(0.145 0 0); + --primary: oklch(0.205 0 0); + --primary-foreground: oklch(0.985 0 0); + --secondary: oklch(0.97 0 0); + --secondary-foreground: oklch(0.205 0 0); + --muted: oklch(0.97 0 0); + --muted-foreground: oklch(0.556 0 0); + --accent: oklch(0.97 0 0); + --accent-foreground: oklch(0.205 0 0); + --destructive: oklch(0.577 0.245 27.325); + --border: oklch(0.922 0 0); + --input: oklch(0.922 0 0); + --ring: oklch(0.708 0 0); + --chart-1: oklch(0.646 0.222 41.116); + --chart-2: oklch(0.6 0.118 184.704); + --chart-3: oklch(0.398 0.07 227.392); + --chart-4: oklch(0.828 0.189 84.429); + --chart-5: oklch(0.769 0.188 70.08); + --sidebar: oklch(0.985 0 0); + --sidebar-foreground: oklch(0.145 0 0); + --sidebar-primary: oklch(0.205 0 0); + --sidebar-primary-foreground: oklch(0.985 0 0); + --sidebar-accent: oklch(0.97 0 0); + --sidebar-accent-foreground: oklch(0.205 0 0); + --sidebar-border: oklch(0.922 0 0); + --sidebar-ring: oklch(0.708 0 0); +} + +@theme inline { + --color-background: var(--background); + --color-foreground: var(--foreground); + --font-sans: var(--font-geist-sans); + --font-mono: var(--font-geist-mono); + + /* generated-semantic-tokens:start */ + --color-button-primary-fill: var(--color-primary-400); + --color-button-primary-fill-pressed: var(--color-primary-500); + --color-button-disabled-fill: var(--color-neutral-100); + --color-button-secondary-fill-pressed: var(--color-primary-100); + --color-button-secondary-fill: var(--color-primary-25); + --color-button-secodnary-border: var(--color-primary-400); + --color-button-disabled-border: var(--color-neutral-200); + --color-button-tertiary-border: var(--color-neutral-600); + --color-button-tertiary-fill-pressed: var(--color-neutral-50); + --color-button-tertiary-fill: var(--color-neutral-25); + --color-button-accent-fill: var(--color-accent-500); + --color-button-accent-pressed: var(--color-accent-600); + --color-background-white: var(--color-neutral-0); + --color-background-dim: var(--color-alpha-black-black-75); + --color-background-dim-dark: var(--color-alpha-black-black-20); + --color-background-dim-darker: var(--color-alpha-black-black-50); + --color-background-dim-darkest: var(--color-alpha-black-black-75); + --color-background-brand: var(--color-primary-25); + --color-surface-surface-default: var(--color-neutral-50); + --color-surface-surface-muted: var(--color-neutral-100); + --color-surface-surface-elevated: var(--color-neutral-25); + --color-surface-white: var(--color-neutral-0); + --color-surface-info: var(--color-alpha-black-black-75); + --color-surface-inverse: var(--color-neutral-800); + --color-surface-inverse-default: var(--color-neutral-900); + --color-text-basic: var(--color-neutral-900); + --color-text-subtle: var(--color-neutral-700); + --color-text-disabled: var(--color-neutral-400); + --color-text-basic-inverse: var(--color-neutral-0); + --color-text-subtle-inverse: var(--color-neutral-300); + --color-text-primary: var(--color-primary-900); + --color-text-accent: var(--color-accent-700); + --color-text-success: var(--color-success-700); + --color-text-secondary: var(--color-primary-700); + --color-text-subtler: var(--color-neutral-500); + --color-text-brand: var(--color-primary-400); + --color-text-error: var(--color-accent-500); + --color-icon-inverse: var(--color-neutral-0); + --color-icon-primary: var(--color-primary-400); + --color-icon-subtler: var(--color-neutral-500); + --color-icon-subtle: var(--color-neutral-500); + --color-icon-disabled: var(--color-neutral-400); + --color-icon-basic: var(--color-neutral-700); + --color-icon-gray: var(--color-neutral-200); + --color-icon-secondary: var(--color-primary-700); + --color-icon-tertiary: var(--color-primary-900); + --color-brand-primary: var(--color-primary-400); + --color-element-primary: var(--color-primary-400); + --color-element-primary-light: var(--color-primary-100); + --color-element-primary-lighter: var(--color-primary-50); + --color-element-primary-alpha: var(--color-alpha-primary-yellow-20); + --color-element-gray-subtler: var(--color-neutral-25); + --color-element-gray-subtle: var(--color-neutral-50); + --color-element-gray: var(--color-neutral-100); + --color-element-gray-lighter: var(--color-neutral-25); + --color-element-gray-light: var(--color-neutral-50); + --color-element-gray-dark: var(--color-neutral-200); + --color-element-alpha-light: var(--color-alpha-white-white-80); + --color-element-white: var(--color-neutral-0); + --color-element-alpha-dark: var(--color-alpha-black-black-50); + --color-element-primary-lightest: var(--color-primary-25); + --color-element-gray-darker: var(--color-neutral-400); + --color-element-disabled: var(--color-neutral-100); + --color-element-accent-light: var(--color-accent-300); + --color-element-letter: var(--color-primary-200); + --color-border-primary: var(--color-primary-400); + --color-border-primary-light: var(--color-primary-200); + --color-border-error: var(--color-accent-500); + --color-border-gray: var(--color-neutral-300); + --color-border-gray-light: var(--color-neutral-200); + --color-border-gray-dark: var(--color-neutral-600); + --color-border-gray-darker: var(--color-neutral-800); + --color-border-primary-lighter: var(--color-primary-100); + --color-border-gray-lighter: var(--color-neutral-100); + --color-divider-gray: var(--color-neutral-100); + --color-divider-gray-dark: var(--color-neutral-600); + --color-divider-gray-light: var(--color-neutral-25); + --color-divider-inverse: var(--color-neutral-0); + --color-action-secondary-pressed: var(--color-neutral-100); + --color-action-secondary: var(--color-neutral-0); + --color-action-primary-pressed: var(--color-primary-400); + --color-action-primary: var(--color-primary-25); +/* generated-semantic-tokens:end */ + + /* generated-typography-tokens:start */ + --font-primary: var(--font-family-primary); + --text-12: var(--font-size-12); + --text-13: var(--font-size-13); + --text-14: var(--font-size-14); + --text-15: var(--font-size-15); + --text-16: var(--font-size-16); + --text-17: var(--font-size-17); + --text-18: var(--font-size-18); + --leading-18: var(--line-height-18); + --text-20: var(--font-size-20); + --leading-20: var(--line-height-20); + --text-24: var(--font-size-24); + --leading-24: var(--line-height-24); + --text-28: var(--font-size-28); + --leading-28: var(--line-height-28); + --text-32: var(--font-size-32); + --text-36: var(--font-size-36); + --leading-36: var(--line-height-36); + --text-40: var(--font-size-40); + --text-44: var(--font-size-44); + --color-sidebar-ring: var(--sidebar-ring); + --color-sidebar-border: var(--sidebar-border); + --color-sidebar-accent-foreground: var(--sidebar-accent-foreground); + --color-sidebar-accent: var(--sidebar-accent); + --color-sidebar-primary-foreground: var(--sidebar-primary-foreground); + --color-sidebar-primary: var(--sidebar-primary); + --color-sidebar-foreground: var(--sidebar-foreground); + --color-sidebar: var(--sidebar); + --color-chart-5: var(--chart-5); + --color-chart-4: var(--chart-4); + --color-chart-3: var(--chart-3); + --color-chart-2: var(--chart-2); + --color-chart-1: var(--chart-1); + --color-ring: var(--ring); + --color-input: var(--input); + --color-border: var(--border); + --color-destructive: var(--destructive); + --color-accent-foreground: var(--accent-foreground); + --color-accent: var(--accent); + --color-muted-foreground: var(--muted-foreground); + --color-muted: var(--muted); + --color-secondary-foreground: var(--secondary-foreground); + --color-secondary: var(--secondary); + --color-primary-foreground: var(--primary-foreground); + --color-primary: var(--primary); + --color-popover-foreground: var(--popover-foreground); + --color-popover: var(--popover); + --color-card-foreground: var(--card-foreground); + --color-card: var(--card); +/* generated-typography-tokens:end */ + --radius-sm: calc(var(--radius) - 4px); + --radius-md: calc(var(--radius) - 2px); + --radius-lg: var(--radius); + --radius-xl: calc(var(--radius) + 4px); +} + +body { + font-family: var(--font-pretendard); +} + +/* 모달 애니메이션 */ +@keyframes slide-up { + from { + transform: translateY(100%); + } + to { + transform: translateY(0); + } +} + +.animate-slide-up { + animation: slide-up 0.3s ease-out; +} +/* generated-typography-utilities:start */ +@utility typo-title-lg-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-44); + line-height: var(--line-height-58); +} + +@utility typo-title-lg-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-44); + line-height: var(--line-height-58); +} + +@utility typo-title-md-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-40); + line-height: var(--line-height-52); +} + +@utility typo-title-md-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-40); + line-height: var(--line-height-52); +} + +@utility typo-title-sm-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-36); + line-height: var(--line-height-50); +} + +@utility typo-title-sm-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-36); + line-height: var(--line-height-50); +} + +@utility typo-heading-2xl-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-32); + line-height: var(--line-height-48); +} + +@utility typo-heading-2xl-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-32); + line-height: var(--line-height-48); +} + +@utility typo-heading-2xl-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-32); + line-height: var(--line-height-48); +} + +@utility typo-heading-2xl-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-32); + line-height: var(--line-height-48); +} + +@utility typo-heading-1xl-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-28); + line-height: var(--line-height-42); +} + +@utility typo-heading-1xl-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-28); + line-height: var(--line-height-42); +} + +@utility typo-heading-1xl-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-28); + line-height: var(--line-height-42); +} + +@utility typo-heading-1xl-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-28); + line-height: var(--line-height-42); +} + +@utility typo-heading-lg-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-24); + line-height: var(--line-height-36); +} + +@utility typo-heading-lg-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-24); + line-height: var(--line-height-36); +} + +@utility typo-heading-lg-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-24); + line-height: var(--line-height-36); +} + +@utility typo-heading-lg-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-24); + line-height: var(--line-height-36); +} + +@utility typo-heading-md-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-20); + line-height: var(--line-height-30); +} + +@utility typo-heading-md-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-20); + line-height: var(--line-height-30); +} + +@utility typo-heading-md-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-20); + line-height: var(--line-height-30); +} + +@utility typo-heading-md-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-20); + line-height: var(--line-height-30); +} + +@utility typo-heading-sm-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-18); + line-height: var(--line-height-28); +} + +@utility typo-heading-sm-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-18); + line-height: var(--line-height-28); +} + +@utility typo-heading-sm-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-18); + line-height: var(--line-height-28); +} + +@utility typo-heading-sm-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-18); + line-height: var(--line-height-28); +} + +@utility typo-body-1xl-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-17); + line-height: var(--line-height-26); +} + +@utility typo-body-1xl-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-17); + line-height: var(--line-height-26); +} + +@utility typo-body-1xl-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-17); + line-height: var(--line-height-26); +} + +@utility typo-body-1xl-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-17); + line-height: var(--line-height-26); +} + +@utility typo-body-lg-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-16); + line-height: var(--line-height-24); +} + +@utility typo-body-lg-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-16); + line-height: var(--line-height-24); +} + +@utility typo-body-lg-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-16); + line-height: var(--line-height-24); +} + +@utility typo-body-lg-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-16); + line-height: var(--line-height-24); +} + +@utility typo-body-md-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-15); + line-height: var(--line-height-24); +} + +@utility typo-body-md-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-15); + line-height: var(--line-height-24); +} + +@utility typo-body-md-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-15); + line-height: var(--line-height-24); +} + +@utility typo-body-md-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-15); + line-height: var(--line-height-24); +} + +@utility typo-body-sm-bold { + font-family: var(--font-primary); + font-weight: 700; + font-size: var(--font-size-14); + line-height: var(--line-height-20); +} + +@utility typo-body-sm-semibold { + font-family: var(--font-primary); + font-weight: 600; + font-size: var(--font-size-14); + line-height: var(--line-height-20); +} + +@utility typo-body-sm-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-14); + line-height: var(--line-height-20); +} + +@utility typo-body-sm-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-14); + line-height: var(--line-height-20); +} + +@utility typo-caption-md-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-13); + line-height: var(--line-height-20); +} + +@utility typo-caption-md-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-13); + line-height: var(--line-height-20); +} + +@utility typo-caption-sm-medium { + font-family: var(--font-primary); + font-weight: 500; + font-size: var(--font-size-12); + line-height: var(--line-height-18); +} + +@utility typo-caption-sm-regular { + font-family: var(--font-primary); + font-weight: 400; + font-size: var(--font-size-12); + line-height: var(--line-height-18); +} + +@utility drop-shadow-25-5 { + box-shadow: 0px 0px 25px 5px #00000014; +} + +@utility drop-shadow-10-5 { + box-shadow: 0px 0px 10px 5px #0000000d; +} + +/* generated-typography-utilities:end */ + +.dark { + --background: oklch(0.145 0 0); + --foreground: oklch(0.985 0 0); + --card: oklch(0.205 0 0); + --card-foreground: oklch(0.985 0 0); + --popover: oklch(0.205 0 0); + --popover-foreground: oklch(0.985 0 0); + --primary: oklch(0.922 0 0); + --primary-foreground: oklch(0.205 0 0); + --secondary: oklch(0.269 0 0); + --secondary-foreground: oklch(0.985 0 0); + --muted: oklch(0.269 0 0); + --muted-foreground: oklch(0.708 0 0); + --accent: oklch(0.269 0 0); + --accent-foreground: oklch(0.985 0 0); + --destructive: oklch(0.704 0.191 22.216); + --border: oklch(1 0 0 / 10%); + --input: oklch(1 0 0 / 15%); + --ring: oklch(0.556 0 0); + --chart-1: oklch(0.488 0.243 264.376); + --chart-2: oklch(0.696 0.17 162.48); + --chart-3: oklch(0.769 0.188 70.08); + --chart-4: oklch(0.627 0.265 303.9); + --chart-5: oklch(0.645 0.246 16.439); + --sidebar: oklch(0.205 0 0); + --sidebar-foreground: oklch(0.985 0 0); + --sidebar-primary: oklch(0.488 0.243 264.376); + --sidebar-primary-foreground: oklch(0.985 0 0); + --sidebar-accent: oklch(0.269 0 0); + --sidebar-accent-foreground: oklch(0.985 0 0); + --sidebar-border: oklch(1 0 0 / 10%); + --sidebar-ring: oklch(0.556 0 0); +} + +@layer base { + a { + -webkit-tap-highlight-color: rgba(0, 0, 0, 0.1); /* a tag 누르면 살짝 회색 배경 보이게 */ + } + + * { + @apply border-border outline-ring/50; + } + body { + @apply bg-background text-foreground; + } +} + +@keyframes marquee { + from { + transform: translateX(0); + } + to { + transform: translateX(-50%); + } +} + +.tab-highlight { + -webkit-tap-highlight-color: rgba(0, 0, 0, 0.1); +} + +.animate-marquee { + animation-name: marquee; + animation-timing-function: linear; + animation-iteration-count: infinite; +} + +.scrollbar-hide::-webkit-scrollbar { + display: none; /* Chrome, Safari, Opera */ +} +.scrollbar-hide { + -ms-overflow-style: none; /* IE, Edge */ + scrollbar-width: none; /* Firefox */ +} + +@supports (-webkit-touch-callout: none) { + .h-screen { + height: -webkit-fill-available; + } + .min-h-screen { + min-height: -webkit-fill-available; + } + .max-h-screen { + max-height: -webkit-fill-available; + } +} + diff --git a/src/app/image-test/page.tsx b/src/app/image-test/page.tsx new file mode 100644 index 00000000..3c3b079f --- /dev/null +++ b/src/app/image-test/page.tsx @@ -0,0 +1,244 @@ +'use client'; + +import exifr from 'exifr'; // ★ 추가 +import { ChangeEvent, SyntheticEvent, useEffect, useState } from 'react'; + +function generateUUID() { + if (window.crypto?.randomUUID) { + return window.crypto.randomUUID(); + } + // fallback + return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => { + const r = (Math.random() * 16) | 0; + const v = c === 'x' ? r : (r & 0x3) | 0x8; + return v.toString(16); + }); +} + +const FILE_SIZE = 2000; + +type PreviewItem = { + id: string; + name: string; + size: number; + type: string; + url: string; // object URL + pickedAt: number; // URL 만든 시점 (performance.now) - 디코드 시간 계산용 + createdAt: number; // EXIF(또는 lastModified) 기반 "이미지 생성/촬영 시각" + createdAtSource: 'exif' | 'file'; // 출처 + loadedAt?: number; // onLoad 시점 (performance.now) + width?: number; + height?: number; +}; + +function formatBytes(n: number) { + const u = ['B', 'KB', 'MB', 'GB']; + let i = 0; + let v = n; + while (v >= 1024 && i < u.length - 1) { + v /= 1024; + i++; + } + return `${v.toFixed(1)} ${u[i]}`; +} + +function formatDate(ts: number) { + const d = new Date(ts); + // 예: 2025-10-29 11:22:33 + const pad = (x: number) => String(x).padStart(2, '0'); + return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())} ${pad(d.getHours())}:${pad(d.getMinutes())}:${pad(d.getSeconds())}`; +} + +// EXIF에서 생성시각(촬영시각)을 가져오되, 없으면 file.lastModified로 폴백 +async function getImageCreatedMs( + file: File, +): Promise<{ ts: number; source: 'exif' | 'file' }> { + try { + const tags = await exifr.parse(file, { + pick: ['DateTimeOriginal', 'CreateDate', 'ModifyDate'], + translateValues: true, + }); + + // 우선순위: DateTimeOriginal > CreateDate > ModifyDate + const dt: Date | undefined = + (tags?.DateTimeOriginal as Date | undefined) ?? + (tags?.CreateDate as Date | undefined) ?? + (tags?.ModifyDate as Date | undefined); + + if (dt && typeof dt.getTime === 'function') { + const ms = dt.getTime(); + if (!Number.isNaN(ms)) return { ts: ms, source: 'exif' }; + } + } catch {} + // 스크린샷/편집본/소셜 다운로드 등 EXIF 없음 → 파일 시스템 타임스탬프 사용 + return { ts: file.lastModified, source: 'file' }; +} + +export default function UploadPreview100() { + const [items, setItems] = useState([]); + const [limit, setLimit] = useState(FILE_SIZE); + const [loadingCount, setLoadingCount] = useState(0); + + useEffect(() => { + return () => { + items.forEach((it) => URL.revokeObjectURL(it.url)); + }; + }, []); + + async function handlePick(e: ChangeEvent) { + const fl = e.target.files; + if (!fl) return; + + const files = Array.from(fl) + .filter((f) => f.type.startsWith('image/')) + .slice(0, limit); + + // 이전 미리보기 정리 + setItems((prev) => { + prev.forEach((p) => URL.revokeObjectURL(p.url)); + return []; + }); + + // EXIF 읽기 + 프리뷰 구성 병렬 처리 + const next: PreviewItem[] = await Promise.all( + files.map(async (f) => { + const { ts, source } = await getImageCreatedMs(f); + return { + id: `${f.name}-${generateUUID()}`, // 중복 파일명 대비 + name: f.name, + size: f.size, + type: f.type, + url: URL.createObjectURL(f), + pickedAt: performance.now(), + createdAt: ts, + createdAtSource: source, + }; + }), + ); + + setItems(next); + setLoadingCount(next.length); + } + + function onImgLoad(i: number, ev: SyntheticEvent) { + const el = ev.currentTarget; + const loadedAt = performance.now(); + setItems((prev) => { + const cp = [...prev]; + const it = cp[i]; + if (!it) return prev; + cp[i] = { + ...it, + loadedAt, + width: el.naturalWidth, + height: el.naturalHeight, + }; + return cp; + }); + setLoadingCount((c) => Math.max(0, c - 1)); + } + + const totalBytes = items.reduce((a, b) => a + b.size, 0); + const loadedItems = items.filter((i) => i.loadedAt); + const avgDecodeMs = loadedItems.length + ? Math.round( + loadedItems.reduce( + (a, b) => a + Math.max(0, b.loadedAt! - b.pickedAt), + 0, + ) / loadedItems.length, + ) + : 0; + + return ( +
+

+ 이미지 {FILE_SIZE}장 동시 업로드(미리보기) 테스트 +

+ +
+
+
+ + setLimit(Number(e.target.value))} + className='w-full rounded-xl border px-3 py-2' + /> +
+
+ + +
+
+ +
+
+ 총 개수: {items.length} +
+ {/*
로딩 중 개수: {loadingCount}
*/} +
+ 총 용량: {formatBytes(totalBytes)} +
+
+ 평균 디코드: {avgDecodeMs} ms +
+
+
+ + {/* Grid */} +
+ {items.map((it, i) => ( +
+ {it.name} onImgLoad(i, e)} + className='max-h-[320px] w-full rounded-lg object-cover' + /> +
+
+ {it.name} +
+
+ + {formatBytes(it.size)} · {it.width ?? '-'}×{it.height ?? '-'} + + + 생성시각:{' '} + {formatDate(it.createdAt)} + + ({it.createdAtSource === 'exif' ? 'EXIF' : '파일시스템'}) + + + {/* 디코드 시간 보려면 주석 해제 + {it.loadedAt && ( + {Math.max(0, Math.round(it.loadedAt - it.pickedAt))} ms + )} */} +
+
+
+ ))} +
+
+ ); +} diff --git a/src/app/layout.tsx b/src/app/layout.tsx new file mode 100644 index 00000000..5a4309b9 --- /dev/null +++ b/src/app/layout.tsx @@ -0,0 +1,109 @@ +import KakaoProvider from '@/global/context/KakaoProvider'; +import QueryProvider from '@/global/context/QueryProvider'; +import type { Metadata } from 'next'; +import localFont from 'next/font/local'; +import './globals.css'; + +const pretendard = localFont({ + src: '../../public/fonts/PretendardVariable.woff2', + display: 'swap', + weight: '100 900', + variable: '--font-pretendard', + preload: false, +}); + +export const metadata: Metadata = { + title: '치이이즈: 추억은 따끈할 때 제맛', + description: '딱 7일만 열리는 특별한 공유 앨범 서비스', + metadataBase: new URL('https://say-cheese.me'), + openGraph: { + title: '치이이즈: 추억은 따끈할 때 제맛', + description: '딱 7일만 열리는 특별한 공유 앨범 서비스', + url: 'https://say-cheese.me/main', + siteName: '치이이즈', + images: [ + { + url: '/og/default_og.png', + width: 1200, + height: 630, + alt: '치이이즈', + }, + ], + locale: 'ko_KR', + type: 'website', + }, +}; + +export default function RootLayout({ + children, +}: Readonly<{ + children: React.ReactNode; +}>) { + return ( + + + + + */} + + +
+
+ + {children} + +
+
+ + + ); +} diff --git a/src/app/login/page.tsx b/src/app/login/page.tsx new file mode 100644 index 00000000..65469ca3 --- /dev/null +++ b/src/app/login/page.tsx @@ -0,0 +1,48 @@ +import KakaoSignupButton from '@/feature/login/components/KakaoSignupButton'; +import Image from 'next/image'; +import { Suspense } from 'react'; + +export default function LoginPage() { + return ( +
+ {/* */} +
+ 치즈 아이콘 + 치즈 아이콘 + + 우리가 특별한 순간을 기억하는 법 + +
+ +
+
+
+ + ⚡️3초만에 빠른 회원가입 + +
+ 삼각형 +
+ + + + +
+
+ ); +} diff --git a/src/app/main/closed-album/page.tsx b/src/app/main/closed-album/page.tsx new file mode 100644 index 00000000..91ec5566 --- /dev/null +++ b/src/app/main/closed-album/page.tsx @@ -0,0 +1,7 @@ +import ScreenMainClosedAlbum from '@/feature/main/closed-album/components/ScreenMainClosedAlbum'; + +interface PageProps {} + +export default function Page({}: PageProps) { + return ; +} diff --git a/src/app/main/page.tsx b/src/app/main/page.tsx new file mode 100644 index 00000000..26fc4dbd --- /dev/null +++ b/src/app/main/page.tsx @@ -0,0 +1,7 @@ +import ScreenMain from '@/feature/main/components/ScreenMain'; + +interface MainPageProps {} + +export default function MainPage({}: MainPageProps) { + return ; +} diff --git a/src/app/mypage/setting/page.tsx b/src/app/mypage/setting/page.tsx new file mode 100644 index 00000000..f06aec82 --- /dev/null +++ b/src/app/mypage/setting/page.tsx @@ -0,0 +1,11 @@ +import ScreenMypage from '@/feature/mypage/components/ScreenMypage'; + +interface PageProps {} + +export default function Page({}: PageProps) { + return ( + <> + + + ); +} diff --git a/src/app/oauth/callback/route.ts b/src/app/oauth/callback/route.ts new file mode 100644 index 00000000..133d7403 --- /dev/null +++ b/src/app/oauth/callback/route.ts @@ -0,0 +1,110 @@ +import { + ACCESS_TOKEN_KEY, + REFRESH_TOKEN_KEY, +} from '@/global/constants/cookies'; +import { NextRequest, NextResponse } from 'next/server'; + +export async function GET(request: NextRequest) { + const { searchParams } = new URL(request.url); + const protocol = request.headers.get('x-forwarded-proto') || 'https'; + const host = + request.headers.get('x-forwarded-host') || + request.headers.get('host') || + 'localhost:3000'; + const code = searchParams.get('code'); + const redirectParam = searchParams.get('redirect'); + let redirect: string | null = null; + + if (redirectParam) { + const decodedRedirect = decodeURIComponent(redirectParam); + try { + redirect = new URL(decodedRedirect, `${protocol}://${host}`).toString(); + } catch { + redirect = null; + } + } + + if (!code) { + return NextResponse.json( + { error: 'Authorization code is required' }, + { status: 400 }, + ); + } + + try { + const response = await fetch( + `${process.env.NEXT_PUBLIC_API_URL}/v1/auth/exchange?code=${code}`, + { + method: 'GET', + headers: { + 'Content-Type': 'application/json', + }, + }, + ); + + if (!response.ok) { + const errorText = await response.text(); + throw new Error(`요청 실패: ${response.status} ${errorText}`); + } + + const data = await response.json(); + + // ----- entry 쿠키 기준 서버 분기 ----- + const cookieDomain = + process.env.NODE_ENV === 'production' ? '.say-cheese.me' : undefined; + + // 쿠키 파싱 (Next.js 13+ 방식) + const entry = request.cookies.get('entry')?.value ?? null; + + let redirectPath = '/main'; + if (data.result.isOnboarded) { + if (entry === 'create-album') { + redirectPath = '/create-album'; + } + } else { + redirectPath = '/onboarding'; + } + const redirectUrl = new URL(redirectPath, `${protocol}://${host}`); + + if (!data.result.isOnboarded) { + redirectUrl.searchParams.set( + 'onboarding', + data.result.isOnboarded.toString(), + ); + redirectUrl.searchParams.set( + 'name', + encodeURIComponent(data.result.name), + ); + } + + // redirect 응답 객체 생성 + const res = NextResponse.redirect(redirect || redirectUrl); + res.cookies.set(ACCESS_TOKEN_KEY, data.result.accessToken, { + secure: process.env.NODE_ENV === 'production', + sameSite: 'lax', + domain: cookieDomain, + maxAge: 60 * 60 * 2, // 2시간 + path: '/', + }); + res.cookies.set(REFRESH_TOKEN_KEY, data.result.refreshToken, { + secure: process.env.NODE_ENV === 'production', + domain: cookieDomain, + sameSite: 'lax', + maxAge: 60 * 60 * 24 * 7, // 7일 + path: '/', + }); + // entry 쿠키 삭제 (만료일을 과거로) + res.cookies.set('entry', '', { + path: '/', + expires: new Date(0), + domain: cookieDomain, + }); + return res; + } catch (error) { + console.error('Auth callback error:', error); + return NextResponse.json( + { error: '서버사이드에서 토큰 처리 중 오류 발생' }, + { status: 500 }, + ); + } +} diff --git a/src/app/onboarding/complete/page.tsx b/src/app/onboarding/complete/page.tsx new file mode 100644 index 00000000..2651d4ea --- /dev/null +++ b/src/app/onboarding/complete/page.tsx @@ -0,0 +1,5 @@ +import ScreenOnboardingComplete from '@/feature/onboarding/components/ScreenOnBoardingComplete'; + +export default function page() { + return ; +} diff --git a/src/app/onboarding/page.tsx b/src/app/onboarding/page.tsx new file mode 100644 index 00000000..1674158a --- /dev/null +++ b/src/app/onboarding/page.tsx @@ -0,0 +1,11 @@ +import ScreenOnBoarding from '@/feature/onboarding/components/ScreenOnBoarding'; +import Spinner from '@/global/components/Spinner'; +import { Suspense } from 'react'; + +export default function OnBoardingPage() { + return ( + }> + + + ); +} diff --git a/src/app/page.tsx b/src/app/page.tsx new file mode 100644 index 00000000..33cd9e1a --- /dev/null +++ b/src/app/page.tsx @@ -0,0 +1,5 @@ +import ScreenRoot from '@/feature/root/components/ScreenRoot'; + +export default function Page() { + return ; +} diff --git a/src/app/photo/detail/[albumId]/page.tsx b/src/app/photo/detail/[albumId]/page.tsx new file mode 100644 index 00000000..5083f286 --- /dev/null +++ b/src/app/photo/detail/[albumId]/page.tsx @@ -0,0 +1,11 @@ +import ScreenPhotoDetail from '@/feature/photo-detail/components/ScreenPhotoDetail'; + +export default async function Page({ + params, +}: { + params: Promise<{ albumId: string }>; +}) { + const { albumId } = await params; + + return ; +} diff --git a/src/app/photo/entry/[albumId]/page.tsx b/src/app/photo/entry/[albumId]/page.tsx new file mode 100644 index 00000000..5362c84b --- /dev/null +++ b/src/app/photo/entry/[albumId]/page.tsx @@ -0,0 +1,12 @@ +import ScreenPhotoShareEntry from '@/feature/photo-entry/components/ScreenPhotoShareEntry'; +import { Suspense } from 'react'; + +export default function Page({ params }: { params: { albumId: string } }) { + const { albumId } = params; + + return ( + + + + ); +} diff --git a/src/app/sitemap.ts b/src/app/sitemap.ts new file mode 100644 index 00000000..800d9e9c --- /dev/null +++ b/src/app/sitemap.ts @@ -0,0 +1,32 @@ +import { MetadataRoute } from 'next'; + +export default function sitemap(): MetadataRoute.Sitemap { + const baseUrl = 'https://say-cheese.me'; + + return [ + { + url: baseUrl, + lastModified: new Date(), + changeFrequency: 'yearly', + priority: 1, + }, + { + url: `${baseUrl}/main`, + lastModified: new Date(), + changeFrequency: 'daily', + priority: 0.9, + }, + { + url: `${baseUrl}/login`, + lastModified: new Date(), + changeFrequency: 'monthly', + priority: 0.5, + }, + { + url: `${baseUrl}/create-album`, + lastModified: new Date(), + changeFrequency: 'monthly', + priority: 0.8, + }, + ]; +} diff --git a/src/app/term/page.tsx b/src/app/term/page.tsx new file mode 100644 index 00000000..fdda304d --- /dev/null +++ b/src/app/term/page.tsx @@ -0,0 +1,11 @@ +import ScreenTerm from '@/feature/term/ScreenTerm'; +import Spinner from '@/global/components/Spinner'; +import { Suspense } from 'react'; + +export default function TermPage() { + return ( + }> + + + ); +} diff --git a/src/app/token.json b/src/app/token.json new file mode 100644 index 00000000..8c83e003 --- /dev/null +++ b/src/app/token.json @@ -0,0 +1,1924 @@ +{ + "global": { + "primary": { + "25": { + "value": "#fffbeb", + "type": "color" + }, + "50": { + "value": "#fff7d6", + "type": "color" + }, + "100": { + "value": "#fff2c2", + "type": "color" + }, + "200": { + "value": "#ffe894", + "type": "color" + }, + "300": { + "value": "#ffdc5c", + "type": "color" + }, + "400": { + "value": "#ffcd14", + "type": "color" + }, + "500": { + "value": "#ffb700", + "type": "color" + }, + "600": { + "value": "#d18800", + "type": "color" + }, + "700": { + "value": "#996300", + "type": "color" + }, + "800": { + "value": "#664200", + "type": "color" + }, + "900": { + "value": "#332100", + "type": "color" + } + }, + "gray": { + "25": { + "value": "#f8f8f7", + "type": "color" + }, + "50": { + "value": "#f3f3f1", + "type": "color" + }, + "100": { + "value": "#e7e6e4", + "type": "color" + }, + "200": { + "value": "#cfcec9", + "type": "color" + }, + "300": { + "value": "#b7b5ae", + "type": "color" + }, + "400": { + "value": "#9f9d93", + "type": "color" + }, + "500": { + "value": "#827f73", + "type": "color" + }, + "600": { + "value": "#605d55", + "type": "color" + }, + "700": { + "value": "#494741", + "type": "color" + }, + "800": { + "value": "#31302b", + "type": "color" + }, + "900": { + "value": "#1b1a18", + "type": "color" + } + }, + "dropshadow-25-5": { + "value": { + "color": "#00000014", + "type": "dropShadow", + "x": 0, + "y": 0, + "blur": 25, + "spread": 5 + }, + "type": "boxShadow" + }, + "dropshadow-10-5": { + "value": { + "color": "#0000000d", + "type": "dropShadow", + "x": 0, + "y": 0, + "blur": 10, + "spread": 5 + }, + "type": "boxShadow" + }, + "fontWeights": { + "pretendard-0": { + "value": "Bold", + "type": "fontWeights" + }, + "pretendard-1": { + "value": "SemiBold", + "type": "fontWeights" + }, + "pretendard-2": { + "value": "Medium", + "type": "fontWeights" + }, + "pretendard-3": { + "value": "Regular", + "type": "fontWeights" + } + }, + "fontSize": { + "0": { + "value": 12, + "type": "fontSizes" + }, + "1": { + "value": 13, + "type": "fontSizes" + }, + "2": { + "value": 14, + "type": "fontSizes" + }, + "3": { + "value": 15, + "type": "fontSizes" + }, + "4": { + "value": 16, + "type": "fontSizes" + }, + "5": { + "value": 17, + "type": "fontSizes" + }, + "6": { + "value": 18, + "type": "fontSizes" + }, + "7": { + "value": 20, + "type": "fontSizes" + }, + "8": { + "value": 24, + "type": "fontSizes" + }, + "9": { + "value": 28, + "type": "fontSizes" + }, + "10": { + "value": 32, + "type": "fontSizes" + }, + "11": { + "value": 36, + "type": "fontSizes" + }, + "12": { + "value": 40, + "type": "fontSizes" + }, + "13": { + "value": 44, + "type": "fontSizes" + } + }, + "letterSpacing": { + "0": { + "value": 0, + "type": "letterSpacing" + } + }, + "paragraphSpacing": { + "0": { + "value": 0, + "type": "paragraphSpacing" + } + }, + "title": { + "lg": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.58}", + "fontSize": "{fontSize.13}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.58}", + "fontSize": "{fontSize.13}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "md": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.52}", + "fontSize": "{fontSize.12}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.52}", + "fontSize": "{fontSize.12}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "sm": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.50}", + "fontSize": "{fontSize.11}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.50}", + "fontSize": "{fontSize.11}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + } + }, + "heading": { + "2xl": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.48}", + "fontSize": "{fontSize.10}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.48}", + "fontSize": "{fontSize.10}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.48}", + "fontSize": "{fontSize.10}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.48}", + "fontSize": "{fontSize.10}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "1xl": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.42}", + "fontSize": "{fontSize.9}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.42}", + "fontSize": "{fontSize.9}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.42}", + "fontSize": "{fontSize.9}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.42}", + "fontSize": "{fontSize.9}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "lg": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.36}", + "fontSize": "{fontSize.8}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.36}", + "fontSize": "{fontSize.8}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.36}", + "fontSize": "{fontSize.8}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.36}", + "fontSize": "{fontSize.8}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "md": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.30}", + "fontSize": "{fontSize.7}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.30}", + "fontSize": "{fontSize.7}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.30}", + "fontSize": "{fontSize.7}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.30}", + "fontSize": "{fontSize.7}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "sm": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.28}", + "fontSize": "{fontSize.6}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.28}", + "fontSize": "{fontSize.6}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.28}", + "fontSize": "{fontSize.6}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.28}", + "fontSize": "{fontSize.6}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + } + }, + "body": { + "1xl": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.26}", + "fontSize": "{fontSize.5}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.26}", + "fontSize": "{fontSize.5}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.26}", + "fontSize": "{fontSize.5}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.26}", + "fontSize": "{fontSize.5}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "lg": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.4}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.4}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.4}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.4}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "md": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.3}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.3}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.3}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.24}", + "fontSize": "{fontSize.3}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "sm": { + "bold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-0}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.2}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "semibold": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-1}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.2}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.2}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.2}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + } + }, + "caption": { + "md": { + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.1}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.20}", + "fontSize": "{fontSize.1}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + }, + "sm": { + "medium": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-2}", + "lineHeight": "{font.line height.18}", + "fontSize": "{fontSize.0}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + }, + "regular": { + "value": { + "fontFamily": "{font.family.pretendard}", + "fontWeight": "{fontWeights.pretendard-3}", + "lineHeight": "{font.line height.18}", + "fontSize": "{fontSize.0}", + "letterSpacing": "{letterSpacing.0}", + "paragraphSpacing": "{paragraphSpacing.0}", + "paragraphIndent": "{paragraphIndent.0}", + "textCase": "{textCase.none}", + "textDecoration": "{textDecoration.none}" + }, + "type": "typography" + } + } + }, + "textCase": { + "none": { + "value": "none", + "type": "textCase" + } + }, + "textDecoration": { + "none": { + "value": "none", + "type": "textDecoration" + } + }, + "paragraphIndent": { + "0": { + "value": "0px", + "type": "dimension" + } + } + }, + "primitive/Mode 1": { + "color": { + "primary": { + "25": { + "value": "#fffbeb", + "type": "color" + }, + "50": { + "value": "#fff7d6", + "type": "color" + }, + "100": { + "value": "#fff2c2", + "type": "color" + }, + "200": { + "value": "#ffe894", + "type": "color" + }, + "300": { + "value": "#ffdc5c", + "type": "color" + }, + "400": { + "value": "#ffcd14", + "type": "color" + }, + "500": { + "value": "#ffb700", + "type": "color" + }, + "600": { + "value": "#e09900", + "type": "color" + }, + "700": { + "value": "#996300", + "type": "color" + }, + "800": { + "value": "#664200", + "type": "color" + }, + "900": { + "value": "#332100", + "type": "color" + } + }, + "accent": { + "25": { + "value": "#fff5f6", + "type": "color" + }, + "50": { + "value": "#ffe5ea", + "type": "color" + }, + "100": { + "value": "#ffccd4", + "type": "color" + }, + "200": { + "value": "#ff99aa", + "type": "color" + }, + "300": { + "value": "#ff6680", + "type": "color" + }, + "400": { + "value": "#ff3355", + "type": "color" + }, + "500": { + "value": "#ff002b", + "type": "color" + }, + "600": { + "value": "#cc0022", + "type": "color" + }, + "700": { + "value": "#99001a", + "type": "color" + }, + "800": { + "value": "#660011", + "type": "color" + }, + "900": { + "value": "#330009", + "type": "color" + } + }, + "success": { + "25": { + "value": "#ebffeb", + "type": "color" + }, + "50": { + "value": "#c7ffc7", + "type": "color" + }, + "100": { + "value": "#a8ffa8", + "type": "color" + }, + "200": { + "value": "#7aff7a", + "type": "color" + }, + "300": { + "value": "#1aff1a", + "type": "color" + }, + "400": { + "value": "#00e500", + "type": "color" + }, + "500": { + "value": "#00c400", + "type": "color" + }, + "600": { + "value": "#009900", + "type": "color" + }, + "700": { + "value": "#006600", + "type": "color" + }, + "800": { + "value": "#004d00", + "type": "color" + }, + "900": { + "value": "#002900", + "type": "color" + } + }, + "neutral": { + "0": { + "value": "#ffffff", + "type": "color" + }, + "25": { + "value": "#f7f7f8", + "type": "color" + }, + "50": { + "value": "#f1f2f3", + "type": "color" + }, + "100": { + "value": "#e5e5e7", + "type": "color" + }, + "200": { + "value": "#c9cacf", + "type": "color" + }, + "300": { + "value": "#afb0b7", + "type": "color" + }, + "400": { + "value": "#94969e", + "type": "color" + }, + "500": { + "value": "#747681", + "type": "color" + }, + "600": { + "value": "#56575f", + "type": "color" + }, + "700": { + "value": "#424349", + "type": "color" + }, + "800": { + "value": "#2c2c30", + "type": "color" + }, + "900": { + "value": "#18191b", + "type": "color" + } + }, + "alpha": { + "white": { + "white 100%": { + "value": "#ffffff", + "type": "color" + }, + "white 80%": { + "value": "#ffffffcc", + "type": "color" + }, + "white 50%": { + "value": "#ffffff80", + "type": "color" + }, + "white 20%": { + "value": "#ffffff33", + "type": "color" + }, + "white 0%": { + "value": "#ffffff00", + "type": "color" + } + }, + "black": { + "black 100%": { + "value": "#18191b", + "type": "color" + }, + "black 75%": { + "value": "#18191bcc", + "type": "color" + }, + "black 50%": { + "value": "#18191b80", + "type": "color" + }, + "black 20%": { + "value": "#18191b33", + "type": "color" + }, + "black 0%": { + "value": "#18191b00", + "type": "color" + }, + "black 10%": { + "value": "#18191b1a", + "type": "color" + } + }, + "primary": { + "yellow 100%": { + "value": "#ffcd14", + "type": "color" + }, + "yellow 80%": { + "value": "#ffcd14cc", + "type": "color" + }, + "yellow 50%": { + "value": "#ffcd1480", + "type": "color" + }, + "yellow 20%": { + "value": "#ffcd1433", + "type": "color" + }, + "yellow 0%": { + "value": "#ffcd1400", + "type": "color" + } + } + } + }, + "font": { + "family": { + "pretendard": { + "value": "pretendard", + "type": "text" + } + }, + "size": { + "12": { + "value": 12, + "type": "number" + }, + "13": { + "value": 13, + "type": "number" + }, + "14": { + "value": 14, + "type": "number" + }, + "15": { + "value": 15, + "type": "number" + }, + "16": { + "value": 16, + "type": "number" + }, + "17": { + "value": 17, + "type": "number" + }, + "18": { + "value": 18, + "type": "number" + }, + "20": { + "value": 20, + "type": "number" + }, + "24": { + "value": 24, + "type": "number" + }, + "28": { + "value": 28, + "type": "number" + }, + "32": { + "value": 32, + "type": "number" + }, + "36": { + "value": 36, + "type": "number" + }, + "40": { + "value": 40, + "type": "number" + }, + "44": { + "value": 44, + "type": "number" + } + }, + "line height": { + "18": { + "value": 18, + "type": "number" + }, + "20": { + "value": 20, + "type": "number" + }, + "24": { + "value": 24, + "type": "number" + }, + "26": { + "value": 26, + "type": "number" + }, + "28": { + "value": 28, + "type": "number" + }, + "30": { + "value": 30, + "type": "number" + }, + "36": { + "value": 36, + "type": "number" + }, + "42": { + "value": 42, + "type": "number" + }, + "48": { + "value": 48, + "type": "number" + }, + "50": { + "value": 50, + "type": "number" + }, + "52": { + "value": 52, + "type": "number" + }, + "58": { + "value": 58, + "type": "number" + } + }, + "weight": { + "400": { + "value": 400, + "type": "number" + }, + "500": { + "value": 500, + "type": "number" + }, + "600": { + "value": 600, + "type": "number" + }, + "700": { + "value": 700, + "type": "number" + } + } + }, + "number": { + "0": { + "value": 0, + "type": "number" + }, + "1": { + "value": 1, + "type": "number" + }, + "2": { + "value": 2, + "type": "number" + }, + "3": { + "value": 4, + "type": "number" + }, + "4": { + "value": 6, + "type": "number" + }, + "5": { + "value": 8, + "type": "number" + }, + "6": { + "value": 10, + "type": "number" + }, + "7": { + "value": 12, + "type": "number" + }, + "8": { + "value": 16, + "type": "number" + }, + "9": { + "value": 20, + "type": "number" + }, + "10": { + "value": 24, + "type": "number" + }, + "11": { + "value": 28, + "type": "number" + }, + "12": { + "value": 32, + "type": "number" + }, + "13": { + "value": 36, + "type": "number" + }, + "14": { + "value": 40, + "type": "number" + }, + "15": { + "value": 44, + "type": "number" + }, + "16": { + "value": 48, + "type": "number" + }, + "17": { + "value": 56, + "type": "number" + }, + "18": { + "value": 64, + "type": "number" + }, + "19": { + "value": 72, + "type": "number" + }, + "20": { + "value": 80, + "type": "number" + }, + "21": { + "value": 96, + "type": "number" + }, + "max": { + "value": 1000, + "type": "number" + } + } + }, + "semantic/Mode 1": { + "color": { + "button": { + "primary-fill": { + "value": "{color.primary.400}", + "type": "color" + }, + "primary-fill-pressed": { + "value": "{color.primary.500}", + "type": "color" + }, + "disabled-fill": { + "value": "{color.neutral.100}", + "type": "color" + }, + "secondary-fill-pressed": { + "value": "{color.primary.100}", + "type": "color" + }, + "secondary-fill": { + "value": "{color.primary.25}", + "type": "color" + }, + "secodnary-border": { + "value": "{color.primary.400}", + "type": "color" + }, + "disabled-border": { + "value": "{color.neutral.200}", + "type": "color" + }, + "tertiary-border": { + "value": "{color.neutral.600}", + "type": "color" + }, + "tertiary-fill-pressed": { + "value": "{color.neutral.50}", + "type": "color" + }, + "tertiary-fill": { + "value": "{color.neutral.25}", + "type": "color" + }, + "accent-fill": { + "value": "{color.accent.500}", + "type": "color" + }, + "accent-pressed": { + "value": "{color.accent.600}", + "type": "color" + } + }, + "background": { + "white": { + "value": "{color.neutral.0}", + "type": "color" + }, + "dim": { + "value": "{color.alpha.black.black 75%}", + "type": "color" + }, + "dim-dark": { + "value": "{color.alpha.black.black 20%}", + "type": "color" + }, + "dim-darker": { + "value": "{color.alpha.black.black 50%}", + "type": "color" + }, + "dim-darkest": { + "value": "{color.alpha.black.black 75%}", + "type": "color" + }, + "brand": { + "value": "{color.primary.25}", + "type": "color" + } + }, + "surface": { + "surface-default": { + "value": "{color.neutral.50}", + "type": "color" + }, + "surface-muted": { + "value": "{color.neutral.100}", + "type": "color" + }, + "surface-elevated": { + "value": "{color.neutral.25}", + "type": "color" + }, + "white": { + "value": "{color.neutral.0}", + "type": "color" + }, + "info": { + "value": "{color.alpha.black.black 75%}", + "type": "color" + }, + "inverse": { + "value": "{color.neutral.800}", + "type": "color" + }, + "inverse-default": { + "value": "{color.neutral.900}", + "type": "color" + } + }, + "text": { + "basic": { + "value": "{color.neutral.900}", + "type": "color" + }, + "subtle": { + "value": "{color.neutral.700}", + "type": "color" + }, + "disabled": { + "value": "{color.neutral.400}", + "type": "color" + }, + "basic-inverse": { + "value": "{color.neutral.0}", + "type": "color" + }, + "subtle-inverse": { + "value": "{color.neutral.300}", + "type": "color" + }, + "primary": { + "value": "{color.primary.900}", + "type": "color" + }, + "accent": { + "value": "{color.accent.700}", + "type": "color" + }, + "success": { + "value": "{color.success.700}", + "type": "color" + }, + "secondary": { + "value": "{color.primary.700}", + "type": "color" + }, + "subtler": { + "value": "{color.neutral.500}", + "type": "color" + }, + "brand": { + "value": "{color.primary.400}", + "type": "color" + }, + "error": { + "value": "{color.accent.500}", + "type": "color" + } + }, + "icon": { + "inverse": { + "value": "{color.neutral.0}", + "type": "color" + }, + "primary": { + "value": "{color.primary.400}", + "type": "color" + }, + "subtler": { + "value": "{color.neutral.500}", + "type": "color" + }, + "subtle": { + "value": "{color.neutral.500}", + "type": "color" + }, + "disabled": { + "value": "{color.neutral.400}", + "type": "color" + }, + "basic": { + "value": "{color.neutral.700}", + "type": "color" + }, + "gray": { + "value": "{color.neutral.200}", + "type": "color" + }, + "secondary": { + "value": "{color.primary.700}", + "type": "color" + }, + "tertiary": { + "value": "{color.primary.900}", + "type": "color" + } + }, + "brand": { + "primary": { + "value": "{color.primary.400}", + "type": "color" + } + }, + "element": { + "primary": { + "value": "{color.primary.400}", + "type": "color" + }, + "primary-light": { + "value": "{color.primary.100}", + "type": "color" + }, + "primary-lighter": { + "value": "{color.primary.50}", + "type": "color" + }, + "primary-alpha": { + "value": "{color.alpha.primary.yellow 20%}", + "type": "color" + }, + "gray-subtler": { + "value": "{color.neutral.25}", + "type": "color" + }, + "gray-subtle": { + "value": "{color.neutral.50}", + "type": "color" + }, + "gray": { + "value": "{color.neutral.100}", + "type": "color" + }, + "gray-lighter": { + "value": "{color.neutral.25}", + "type": "color" + }, + "gray-light": { + "value": "{color.neutral.50}", + "type": "color" + }, + "gray-dark": { + "value": "{color.neutral.200}", + "type": "color" + }, + "alpha-light": { + "value": "{color.alpha.white.white 80%}", + "type": "color" + }, + "white": { + "value": "{color.neutral.0}", + "type": "color" + }, + "alpha-dark": { + "value": "{color.alpha.black.black 50%}", + "type": "color" + }, + "primary-lightest": { + "value": "{color.primary.25}", + "type": "color" + }, + "gray-darker": { + "value": "{color.neutral.400}", + "type": "color" + }, + "disabled": { + "value": "{color.neutral.100}", + "type": "color" + }, + "accent-light": { + "value": "{color.accent.300}", + "type": "color" + }, + "letter": { + "value": "{color.primary.200}", + "type": "color" + } + }, + "border": { + "primary": { + "value": "{color.primary.400}", + "type": "color" + }, + "primary-light": { + "value": "{color.primary.200}", + "type": "color" + }, + "error": { + "value": "{color.accent.500}", + "type": "color" + }, + "gray": { + "value": "{color.neutral.300}", + "type": "color" + }, + "gray-light": { + "value": "{color.neutral.200}", + "type": "color" + }, + "gray-dark": { + "value": "{color.neutral.600}", + "type": "color" + }, + "gray-darker": { + "value": "{color.neutral.800}", + "type": "color" + }, + "primary-lighter": { + "value": "{color.primary.100}", + "type": "color" + }, + "gray-lighter": { + "value": "{color.neutral.100}", + "type": "color" + } + }, + "divider": { + "gray": { + "value": "{color.neutral.100}", + "type": "color" + }, + "gray-dark": { + "value": "{color.neutral.600}", + "type": "color" + }, + "gray-light": { + "value": "{color.neutral.25}", + "type": "color" + }, + "inverse": { + "value": "{color.neutral.0}", + "type": "color" + } + }, + "action": { + "secondary-pressed": { + "value": "{color.neutral.100}", + "type": "color" + }, + "secondary": { + "value": "{color.neutral.0}", + "type": "color" + }, + "primary-pressed": { + "value": "{color.primary.400}", + "type": "color" + }, + "primary": { + "value": "{color.primary.25}", + "type": "color" + } + } + }, + "size-height": { + "1": { + "value": "{number.5}", + "type": "number" + }, + "2": { + "value": "{number.8}", + "type": "number" + }, + "3": { + "value": "{number.9}", + "type": "number" + }, + "4": { + "value": "{number.10}", + "type": "number" + }, + "5": { + "value": "{number.12}", + "type": "number" + }, + "6": { + "value": "{number.14}", + "type": "number" + }, + "7": { + "value": "{number.16}", + "type": "number" + }, + "8": { + "value": "{number.17}", + "type": "number" + }, + "9": { + "value": "{number.18}", + "type": "number" + }, + "10": { + "value": "{number.19}", + "type": "number" + }, + "11": { + "value": "{number.20}", + "type": "number" + } + }, + "gap": { + "g1": { + "value": "{number.2}", + "type": "number" + }, + "g2": { + "value": "{number.3}", + "type": "number" + }, + "g3": { + "value": "{number.5}", + "type": "number" + }, + "g4": { + "value": "{number.7}", + "type": "number" + }, + "g5": { + "value": "{number.8}", + "type": "number" + }, + "g6": { + "value": "{number.9}", + "type": "number" + }, + "g7": { + "value": "{number.10}", + "type": "number" + }, + "g8": { + "value": "{number.12}", + "type": "number" + }, + "g9": { + "value": "{number.14}", + "type": "number" + }, + "g10": { + "value": "{number.16}", + "type": "number" + }, + "g11": { + "value": "{number.18}", + "type": "number" + }, + "g12": { + "value": "{number.20}", + "type": "number" + } + }, + "padding": { + "p1": { + "value": "{number.2}", + "type": "number" + }, + "p2": { + "value": "{number.3}", + "type": "number" + }, + "p3": { + "value": "{number.5}", + "type": "number" + }, + "p4": { + "value": "{number.6}", + "type": "number" + }, + "p5": { + "value": "{number.7}", + "type": "number" + }, + "p6": { + "value": "{number.8}", + "type": "number" + }, + "p7": { + "value": "{number.9}", + "type": "number" + }, + "p8": { + "value": "{number.10}", + "type": "number" + }, + "p9": { + "value": "{number.12}", + "type": "number" + }, + "p10": { + "value": "{number.14}", + "type": "number" + } + }, + "radius": { + "xsmall": { + "value": "{number.2}", + "type": "number" + }, + "small": { + "value": "{number.3}", + "type": "number" + }, + "medium": { + "value": "{number.4}", + "type": "number" + }, + "large": { + "value": "{number.5}", + "type": "number" + }, + "xlarge": { + "value": "{number.6}", + "type": "number" + }, + "2xlarge": { + "value": "{number.7}", + "type": "number" + }, + "max": { + "value": "{number.max}", + "type": "number" + }, + "3xlarge": { + "value": "{number.9}", + "type": "number" + } + } + }, + "$themes": [], + "$metadata": { + "tokenSetOrder": ["global", "primitive/Mode 1", "semantic/Mode 1"] + } +} diff --git a/src/components/ui/EmojiLoading.tsx b/src/components/ui/EmojiLoading.tsx new file mode 100644 index 00000000..aa8aa26c --- /dev/null +++ b/src/components/ui/EmojiLoading.tsx @@ -0,0 +1,99 @@ +'use client'; +import { EP } from '@/global/api/ep'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { motion } from 'framer-motion'; +import { useEffect, useState } from 'react'; + +import Toast from '@/global/components/toast/Toast'; +import BubbleTooltip from '@/global/components/tooltip/BubbleTooltip'; +import { useQueryClient } from '@tanstack/react-query'; + +interface EmojiLoadingProps { + duration?: number; + emoji?: string; + albumId?: string; +} +export default function EmojiLoading({ + duration = 3000, + emoji = 'U+1F60A', + albumId, +}: EmojiLoadingProps) { + const queryClient = useQueryClient(); + const [percent, setPercent] = useState(0); + + const displayEmoji = convertUnicodeToEmoji(emoji); + + useEffect(() => { + const timeout = setTimeout(() => { + let frame: number; + const startTime = performance.now(); + + const animate = (currentTime: number) => { + const elapsed = currentTime - startTime; + const progress = Math.min(elapsed / duration, 1); + setPercent(progress * 100); + + if (progress < 1) { + frame = requestAnimationFrame(animate); + } else { + // albumId를 props로 받아 invalidate + if (albumId) { + queryClient.invalidateQueries({ + queryKey: [EP.album.photos(albumId)], + }); + queryClient.invalidateQueries({ + queryKey: [EP.album.availableCount(albumId)], + }); + } + + const uploadedCount = useUploadingStore.getState().uploadedCount; + useUploadingStore.getState().reset(); + if (uploadedCount > 0) { + Toast.check(`총 ${uploadedCount}장을 앨범에 채웠어요.`); + } + } + }; + frame = requestAnimationFrame(animate); + return () => cancelAnimationFrame(frame); + }, 100); + + return () => clearTimeout(timeout); + }, [duration]); + + return ( +
+
+ + 0 + ? `conic-gradient(#FFCD14 0% ${percent}%, #FFE480 ${percent}%, white ${percent}% 100%)` + : 'white', + }} + /> +
+
+ {displayEmoji} +
+
+
+
+ ); +} diff --git a/src/components/ui/alert-dialog.tsx b/src/components/ui/alert-dialog.tsx new file mode 100644 index 00000000..6ceec312 --- /dev/null +++ b/src/components/ui/alert-dialog.tsx @@ -0,0 +1,157 @@ +'use client'; + +import * as AlertDialogPrimitive from '@radix-ui/react-alert-dialog'; +import * as React from 'react'; + +import { buttonVariants } from '@/components/ui/button'; +import { cn } from '@/lib/utils'; + +function AlertDialog({ + ...props +}: React.ComponentProps) { + return ; +} + +function AlertDialogTrigger({ + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogPortal({ + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogOverlay({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogContent({ + className, + ...props +}: React.ComponentProps) { + return ( + + + + + ); +} + +function AlertDialogHeader({ + className, + ...props +}: React.ComponentProps<'div'>) { + return ( +
+ ); +} + +function AlertDialogFooter({ + className, + ...props +}: React.ComponentProps<'div'>) { + return ( +
+ ); +} + +function AlertDialogTitle({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogDescription({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogAction({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function AlertDialogCancel({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +export { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogOverlay, + AlertDialogPortal, + AlertDialogTitle, + AlertDialogTrigger, +}; diff --git a/src/components/ui/button.tsx b/src/components/ui/button.tsx new file mode 100644 index 00000000..2f79011b --- /dev/null +++ b/src/components/ui/button.tsx @@ -0,0 +1,60 @@ +import { Slot } from '@radix-ui/react-slot'; +import { cva, type VariantProps } from 'class-variance-authority'; +import * as React from 'react'; + +import { cn } from '@/lib/utils'; + +const buttonVariants = cva( + "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive", + { + variants: { + variant: { + default: 'bg-primary text-primary-foreground hover:bg-primary/90', + destructive: + 'bg-destructive text-white hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60', + outline: + 'border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50', + secondary: + 'bg-secondary text-secondary-foreground hover:bg-secondary/80', + ghost: + 'hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50', + link: 'text-primary underline-offset-4 hover:underline', + }, + size: { + default: 'h-9 px-4 py-2 has-[>svg]:px-3', + sm: 'h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5', + lg: 'h-10 rounded-md px-6 has-[>svg]:px-4', + icon: 'size-9', + 'icon-sm': 'size-8', + 'icon-lg': 'size-10', + }, + }, + defaultVariants: { + variant: 'default', + size: 'default', + }, + }, +); + +function Button({ + className, + variant, + size, + asChild = false, + ...props +}: React.ComponentProps<'button'> & + VariantProps & { + asChild?: boolean; + }) { + const Comp = asChild ? Slot : 'button'; + + return ( + + ); +} + +export { Button, buttonVariants }; diff --git a/src/components/ui/calendar.tsx b/src/components/ui/calendar.tsx new file mode 100644 index 00000000..25042200 --- /dev/null +++ b/src/components/ui/calendar.tsx @@ -0,0 +1,222 @@ +'use client'; + +import { + ChevronDownIcon, + ChevronLeftIcon, + ChevronRightIcon, +} from 'lucide-react'; +import * as React from 'react'; +import { DayButton, DayPicker, getDefaultClassNames } from 'react-day-picker'; + +import { Button, buttonVariants } from '@/components/ui/button'; +import { cn } from '@/lib/utils'; + +function Calendar({ + className, + classNames, + showOutsideDays = true, + captionLayout = 'label', + buttonVariant = 'ghost', + formatters, + fromDate, + toDate, + components, + ...props +}: React.ComponentProps & { + buttonVariant?: React.ComponentProps['variant']; + fromDate?: Date; + toDate?: Date; +}) { + const defaultClassNames = getDefaultClassNames(); + + return ( + svg]:rotate-180`, + String.raw`rtl:**:[.rdp-button\_previous>svg]:rotate-180`, + className, + )} + captionLayout={captionLayout} + formatters={{ + formatMonthDropdown: (date) => + date.toLocaleString('default', { month: 'short' }), + ...formatters, + }} + classNames={{ + root: cn('w-fit', defaultClassNames.root), + months: cn( + 'flex gap-4 flex-col md:flex-row relative', + defaultClassNames.months, + ), + month: cn('flex flex-col w-full gap-4', defaultClassNames.month), + nav: cn( + 'flex items-center gap-1 w-full absolute top-0 inset-x-0 justify-between', + defaultClassNames.nav, + ), + button_previous: cn( + buttonVariants({ variant: buttonVariant }), + 'size-(--cell-size) aria-disabled:opacity-50 p-0 select-none', + defaultClassNames.button_previous, + ), + button_next: cn( + buttonVariants({ variant: buttonVariant }), + 'size-(--cell-size) aria-disabled:opacity-50 p-0 select-none', + defaultClassNames.button_next, + ), + month_caption: cn( + 'flex items-center justify-center h-(--cell-size) w-full px-(--cell-size)', + defaultClassNames.month_caption, + ), + dropdowns: cn( + 'w-full flex items-center text-sm font-medium justify-center h-(--cell-size) gap-1.5', + defaultClassNames.dropdowns, + ), + dropdown_root: cn( + 'relative has-focus:border-ring border border-input shadow-xs has-focus:ring-ring/50 has-focus:ring-[3px] rounded-md', + defaultClassNames.dropdown_root, + ), + dropdown: cn( + 'absolute bg-popover inset-0 opacity-0', + defaultClassNames.dropdown, + ), + caption_label: cn( + 'select-none font-medium', + captionLayout === 'label' + ? 'text-sm' + : 'rounded-md pl-2 pr-1 flex items-center gap-1 text-sm h-8 [&>svg]:text-muted-foreground [&>svg]:size-3.5', + defaultClassNames.caption_label, + ), + table: 'w-full border-collapse', + weekdays: cn('flex', defaultClassNames.weekdays), + weekday: cn( + 'text-muted-foreground rounded-md flex-1 font-normal text-[0.8rem] select-none', + defaultClassNames.weekday, + ), + week: cn('flex w-full mt-2', defaultClassNames.week), + week_number_header: cn( + 'select-none w-(--cell-size)', + defaultClassNames.week_number_header, + ), + week_number: cn( + 'text-[0.8rem] select-none text-muted-foreground', + defaultClassNames.week_number, + ), + day: cn( + 'relative w-full h-full p-0 text-center [&:last-child[data-selected=true]_button]:rounded-r-md group/day aspect-square select-none', + props.showWeekNumber + ? '[&:nth-child(2)[data-selected=true]_button]:rounded-l-md' + : '[&:first-child[data-selected=true]_button]:rounded-l-md', + defaultClassNames.day, + ), + range_start: cn( + 'rounded-l-md bg-accent', + defaultClassNames.range_start, + ), + range_middle: cn('rounded-none', defaultClassNames.range_middle), + range_end: cn('rounded-r-md bg-accent', defaultClassNames.range_end), + today: cn( + 'bg-accent text-accent-foreground rounded-md data-[selected=true]:rounded-none', + defaultClassNames.today, + ), + outside: cn( + 'text-muted-foreground aria-selected:text-muted-foreground', + defaultClassNames.outside, + ), + disabled: cn( + 'text-muted-foreground opacity-50', + defaultClassNames.disabled, + ), + hidden: cn('invisible', defaultClassNames.hidden), + ...classNames, + }} + components={{ + Root: ({ className, rootRef, ...props }) => { + return ( +
+ ); + }, + Chevron: ({ className, orientation, ...props }) => { + if (orientation === 'left') { + return ( + + ); + } + + if (orientation === 'right') { + return ( + + ); + } + + return ( + + ); + }, + DayButton: CalendarDayButton, + WeekNumber: ({ children, ...props }) => { + return ( + +
+ {children} +
+ + ); + }, + ...components, + }} + {...props} + /> + ); +} + +function CalendarDayButton({ + className, + day, + modifiers, + ...props +}: React.ComponentProps) { + const defaultClassNames = getDefaultClassNames(); + + const ref = React.useRef(null); + React.useEffect(() => { + if (modifiers.focused) ref.current?.focus(); + }, [modifiers.focused]); + + return ( + + ); +} + +function CarouselNext({ + className, + variant = 'outline', + size = 'icon', + ...props +}: React.ComponentProps) { + const { orientation, scrollNext, canScrollNext } = useCarousel(); + + return ( + + ); +} + +export { + type CarouselApi, + Carousel, + CarouselContent, + CarouselItem, + CarouselPrevious, + CarouselNext, +}; diff --git a/src/components/ui/drawer.tsx b/src/components/ui/drawer.tsx new file mode 100644 index 00000000..37f97bf7 --- /dev/null +++ b/src/components/ui/drawer.tsx @@ -0,0 +1,162 @@ +'use client'; + +import { useEffect } from 'react'; +import { Drawer as DrawerPrimitive } from 'vaul'; + +import { cn } from '@/lib/utils'; +function Drawer({ + ...props +}: React.ComponentProps) { + useEffect(() => { + const cleanup = () => { + document.body.removeAttribute('data-scroll-locked'); + document.body.style.paddingLeft = ''; + document.body.style.paddingRight = ''; + document.body.style.marginLeft = 'auto'; + document.body.style.marginRight = 'auto'; + document.body.style.overflow = ''; + }; + + const observer = new MutationObserver(() => { + if (document.body.hasAttribute('data-scroll-locked')) { + cleanup(); + } + }); + + observer.observe(document.body, { attributes: true }); + + return () => { + observer.disconnect(); + cleanup(); + }; + }, []); + return ; +} + +function DrawerTrigger({ + ...props +}: React.ComponentProps) { + return ; +} + +function DrawerPortal({ + ...props +}: React.ComponentProps) { + return ; +} + +function DrawerClose({ + ...props +}: React.ComponentProps) { + return ; +} + +function DrawerOverlay({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function DrawerContent({ + className, + children, + showHandle = true, + ...props +}: React.ComponentProps & { + showHandle?: boolean; +}) { + return ( + + + + {showHandle && ( +
+ )} + {children} + + + ); +} + +function DrawerHeader({ className, ...props }: React.ComponentProps<'div'>) { + return ( +
+ ); +} + +function DrawerFooter({ className, ...props }: React.ComponentProps<'div'>) { + return ( +
+ ); +} + +function DrawerTitle({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +function DrawerDescription({ + className, + ...props +}: React.ComponentProps) { + return ( + + ); +} + +export { + Drawer, + DrawerClose, + DrawerContent, + DrawerDescription, + DrawerFooter, + DrawerHeader, + DrawerOverlay, + DrawerPortal, + DrawerTitle, + DrawerTrigger, +}; diff --git a/src/components/ui/popover.tsx b/src/components/ui/popover.tsx new file mode 100644 index 00000000..841c81d1 --- /dev/null +++ b/src/components/ui/popover.tsx @@ -0,0 +1,34 @@ +// shadcn/ui Popover 컴포넌트 +'use client'; + +import * as RadixPopover from '@radix-ui/react-popover'; +import * as React from 'react'; + +import { cn } from '@/lib/utils'; + +const Popover = RadixPopover.Root; + +const PopoverTrigger = RadixPopover.Trigger; + +const PopoverAnchor = RadixPopover.Anchor; + +const PopoverContent = React.forwardRef< + React.ElementRef, + React.ComponentPropsWithoutRef +>(({ className, align = 'center', sideOffset = 4, ...props }, ref) => ( + + + +)); +PopoverContent.displayName = RadixPopover.Content.displayName; + +export { Popover, PopoverAnchor, PopoverContent, PopoverTrigger }; diff --git a/src/feature/album-entry/components/FullSizeLetter.tsx b/src/feature/album-entry/components/FullSizeLetter.tsx new file mode 100644 index 00000000..0b683db3 --- /dev/null +++ b/src/feature/album-entry/components/FullSizeLetter.tsx @@ -0,0 +1,132 @@ +interface FullSizeLetterProps { + children: React.ReactNode; +} + +export default function FullSizeLetter({ children }: FullSizeLetterProps) { + return ( + <> + {/* 편지지 */} +
+ {children} +
+ + {/* 뒷편지봉투 svg */} + + + + + {/* 앞편지봉투 svg */} + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + ); +} diff --git a/src/feature/album-entry/components/LetterContent.tsx b/src/feature/album-entry/components/LetterContent.tsx new file mode 100644 index 00000000..33a95889 --- /dev/null +++ b/src/feature/album-entry/components/LetterContent.tsx @@ -0,0 +1,80 @@ +'use client'; +import { useGetAlbumInvitation } from '@/feature/album/detail/hooks/useGetAlbumInvitation'; +import Toast from '@/global/components/toast/Toast'; +import { useCheckAuth } from '@/global/hooks/useCheckAuth'; +import { buildQuery } from '@/global/utils/buildQuery'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import { formatExpirationTime } from '@/global/utils/time/formatExpirationTime'; +import Image from 'next/image'; +import { useRouter } from 'next/navigation'; + +interface LetterContentProps { + albumId: string; +} + +export default function LetterContent({ albumId }: LetterContentProps) { + const router = useRouter(); + const { data, isPending, isError } = useGetAlbumInvitation(albumId); + const { isAuthed } = useCheckAuth(); + + if (isPending) return null; + if (isError) return null; + if (!data) return null; + + const handleInviteAccept = async () => { + try { + if (isAuthed) { + router.push(`/photo/entry/${albumId}${buildQuery({ isInvite: true })}`); + } else { + router.push( + `/login${buildQuery({ redirect: encodeURIComponent(`/photo/entry/${albumId}${buildQuery({ isInvite: true })}`) })}`, + ); + } + } catch (error) { + Toast.alert('앨범 입장에 실패하였습니다'); + } + }; + + return ( + <> +
+ {data.makerName} + + {data.makerName} + +
+
+
+ {convertUnicodeToEmoji(data.themeEmoji)} +
+ +

+ {data.title} +

+ +

+ {data.eventDate} +

+ {!data.isExpired && ( + + 앨범 소멸까지 {formatExpirationTime(data.expiredAt)} + + )} + + +
+ + ); +} diff --git a/src/feature/album-entry/components/ScreenAlbumEntry.tsx b/src/feature/album-entry/components/ScreenAlbumEntry.tsx new file mode 100644 index 00000000..f85746af --- /dev/null +++ b/src/feature/album-entry/components/ScreenAlbumEntry.tsx @@ -0,0 +1,24 @@ +import LogoHeader from '@/global/components/header/LogoHeader'; +import FullSizeLetter from './FullSizeLetter'; +import LetterContent from './LetterContent'; + +interface ScreenAlbumEntryProps { + albumId: string; +} + +export default function ScreenAlbumEntry({ albumId }: ScreenAlbumEntryProps) { + return ( +
+ + +
+ + + + +
+ ); +} diff --git a/src/feature/album-entry/hooks/useAlbumEnterMutation.ts b/src/feature/album-entry/hooks/useAlbumEnterMutation.ts new file mode 100644 index 00000000..7770ee90 --- /dev/null +++ b/src/feature/album-entry/hooks/useAlbumEnterMutation.ts @@ -0,0 +1,25 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +const fetchData = async (albumId: string, redirectUrlOnAuthError?: string) => { + const res = await api.post({ + path: EP.album.enter(albumId), + ...(redirectUrlOnAuthError && { redirectUrlOnAuthError }), + }); + return res.result; +}; + +interface AlbumEnterProps { + albumId: string; + redirectUrlOnAuthError?: string; +} + +export function useAlbumEnterMutation() { + const mutation = useMutation({ + mutationFn: ({ albumId, redirectUrlOnAuthError }: AlbumEnterProps) => + fetchData(albumId, redirectUrlOnAuthError), + }); + + return mutation; +} diff --git a/src/feature/album-select/components/SelectAlbumBody.tsx b/src/feature/album-select/components/SelectAlbumBody.tsx new file mode 100644 index 00000000..74cea9dd --- /dev/null +++ b/src/feature/album-select/components/SelectAlbumBody.tsx @@ -0,0 +1,233 @@ +'use client'; +import { useCheckImages } from '@/feature/create-album/hook/useCheckImages'; +import { getFilesWithCaptureTime } from '@/feature/create-album/utils/getFilesWithCaptureTime'; +import { validateImages } from '@/feature/create-album/utils/validateImages'; +import LongButton from '@/global/components/LongButton'; +import PhotoBox from '@/global/components/photo/PhotoBox'; +import Toast from '@/global/components/toast/Toast'; +import { usePresignedAndUploadToNCP } from '@/global/hooks/usePresignedAndUploadToNCP'; +import { useReportFailed } from '@/global/hooks/useReportFailed'; +import { useImageStore } from '@/store/useImageStore'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { useParams, useRouter } from 'next/navigation'; +import { useEffect, useMemo, useState } from 'react'; + +type ImageWithUrl = { + id: string; + file: File; + url: string; + isOversized: boolean; +}; + +export default function SelectAlbumBody() { + const isUploaded = useUploadingStore((state) => state.isUploaded); + const handleUpload = async () => { + const selectedFiles = processedImages.filter((img) => + selectedIds.has(img.id), + ); + const files = selectedFiles.map((img) => img.file); + const filesWithCapture = await getFilesWithCaptureTime(files); + const fileInfos = filesWithCapture.map(({ file, captureTime }) => ({ + fileName: file.name, + fileSize: file.size, + contentType: file.type, + captureTime, + })); + + uploadMutate({ + albumCode: albumId, + fileInfos, + files, + }); + }; + const { albumId } = useParams() as { albumId: string }; + const { images } = useImageStore(); + const router = useRouter(); + + useEffect(() => { + if (images.length === 0 && albumId) { + router.push(`/album/upload/${albumId}`); + } + }, [images, albumId, router]); + const [selectedIds, setSelectedIds] = useState>(new Set()); + const [availableCount, setAvailableCount] = useState(null); + const { mutate: checkImagesMutate } = useCheckImages(); + + const revokeAllObjectUrls = () => { + processedImages.forEach((img) => { + URL.revokeObjectURL(img.url); + }); + }; + + const { mutate: reportFailed } = useReportFailed(); + + const { mutate: uploadMutate } = usePresignedAndUploadToNCP({ + onSuccess: (result) => { + if (result.failed > 0) { + Toast.alert(`${result.failed}개 파일 업로드에 실패했어요`); + if ( + Array.isArray(result.failedPhotoIds) && + result.failedPhotoIds.length > 0 + ) { + reportFailed(result.failedPhotoIds); + } + } else { + revokeAllObjectUrls(); + useUploadingStore.getState().setUploaded(true); + useUploadingStore.getState().setUploadedCount(result.success); + router.replace(`/album/detail/${albumId}`); + } + }, + onError: (e) => { + revokeAllObjectUrls(); + console.error('에러 발생', e); + alert('사진을 업로드하는 중 오류가 발생했습니다. 다시 시도해주세요.'); + }, + }); + + const processedImages = useMemo(() => { + const validation = validateImages(images.map((img) => img.file)); + const oversizedSet = new Set(validation.oversizedFiles); + + return images.map((img) => ({ + id: img.id, + file: img.file, + url: URL.createObjectURL(img.file), + isOversized: oversizedSet.has(img.file.name), + })); + }, [images]); + + // 뒤로가기(나가기) 또는 페이지 이탈 시 object URL 해제 + useEffect(() => { + const handleRevoke = () => { + revokeAllObjectUrls(); + }; + window.addEventListener('pagehide', handleRevoke); + window.addEventListener('popstate', handleRevoke); + return () => { + window.removeEventListener('pagehide', handleRevoke); + window.removeEventListener('popstate', handleRevoke); + revokeAllObjectUrls(); + }; + }, [processedImages]); + + const validImages = useMemo( + () => processedImages.filter((img) => !img.isOversized), + [processedImages], + ); + + useEffect(() => { + if (!validImages.length) { + setSelectedIds(new Set()); + return; + } + if (!availableCount || validImages.length <= availableCount) { + setSelectedIds(new Set(validImages.map((img) => img.id))); + } else { + setSelectedIds( + new Set(validImages.slice(0, availableCount).map((img) => img.id)), + ); + } + }, [validImages, availableCount]); + + useEffect(() => { + const files = images.map((img) => img.file); + if (!files.length || !albumId) return; + + checkImagesMutate( + { files, albumId }, + { + onSuccess: ({ oversizedFiles, availableCount }) => { + setAvailableCount(availableCount); + const msgs: string[] = []; + if (oversizedFiles.length > 0) { + msgs.push( + `6MB를 초과한 사진 ${oversizedFiles.length}장이 제외되었어요.`, + ); + } + if (files.length > availableCount) { + msgs.push('지금 앨범에 담을 수 있는 만큼만 선택되었어요.'); + } + if (msgs.length) Toast.alert(msgs.join('\n')); + }, + onError: () => { + Toast.alert('이미지 검증 중 오류가 발생했어요.'); + }, + }, + ); + }, [images, albumId, checkImagesMutate]); + + const toggleSelect = ( + id: string, + isOversized: boolean, + nextSelected?: boolean, + ) => { + if (isOversized) return; // 6MB 초과는 선택 불가 + + setSelectedIds((prev) => { + const updated = new Set(prev); + if (typeof nextSelected === 'boolean') { + if (nextSelected) updated.add(id); + else updated.delete(id); + return updated; + } + if (updated.has(id)) updated.delete(id); + else updated.add(id); + return updated; + }); + }; + + const isOverCount = + availableCount !== null && selectedIds.size > availableCount; + + return ( +
+
+ + 총 {images.length}장 + + + {selectedIds.size}/{availableCount} + +
+
+ {processedImages.map((img) => { + const isSelected = selectedIds.has(img.id); + return ( +
+ { + toggleSelect(img.id, img.isOversized, next); + }} + onDisabledPress={() => { + Toast.alert('사진이 6MB를 초과해 업로드할 수 없어요.'); + }} + /> +
+ ); + })} +
+ +
+ ); +} diff --git a/src/feature/album/4cut/components/Capture4CutPortal.tsx b/src/feature/album/4cut/components/Capture4CutPortal.tsx new file mode 100644 index 00000000..8deff1da --- /dev/null +++ b/src/feature/album/4cut/components/Capture4CutPortal.tsx @@ -0,0 +1,42 @@ +'use client'; + +import { BodyPortal } from '@/global/components/portal/BodyPortal'; +import type { RefObject } from 'react'; +import Container4Cut from './Container4Cut'; + +interface Capture4CutPortalProps { + captureRef: RefObject; + visible: boolean; + albumId: string; + eventName?: string; + eventDate?: string; +} + +const Capture4CutPortal = ({ + captureRef, + visible, + albumId, + eventName, + eventDate, +}: Capture4CutPortalProps) => ( + +
+ +
+
+); + +export default Capture4CutPortal; diff --git a/src/feature/album/4cut/components/Container4Cut.tsx b/src/feature/album/4cut/components/Container4Cut.tsx new file mode 100644 index 00000000..b3a71ca3 --- /dev/null +++ b/src/feature/album/4cut/components/Container4Cut.tsx @@ -0,0 +1,109 @@ +import { useBase64Images } from '@/global/hooks/useBase64Images'; +import { useMemo } from 'react'; +import { use4CutPreviewQuery } from '../hooks/use4CutPreviewQuery'; +import Svg4Cut from '../svg/Svg4Cut'; + +interface Container4CutProps { + albumId: string; + eventName?: string; + eventDate?: string; + scale?: number; + width?: number; +} + +const BASE_WIDTH = 216; +const BASE_HEIGHT = 384; +const BASE_ASPECT_RATIO = BASE_HEIGHT / BASE_WIDTH; +const BASE_FONT_SIZE = 7.963; +const BASE_NAME_POSITION = { + bottom: 7.4, + left: 9.6, +}; +const BASE_DATE_POSITION = { + bottom: 7.4, + right: 10.4, +}; + +export default function Container4Cut({ + albumId, + eventDate, + eventName, + scale = 1, + width, +}: Container4CutProps) { + // TODO : openapi type이 이상해서 임시 any처리. 백엔드랑 협의 필요 + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const { data }: any = use4CutPreviewQuery(albumId); + + const images = useMemo(() => { + return ( + // eslint-disable-next-line @typescript-eslint/no-explicit-any + data?.previewPhotos?.map((item: any) => item.imageUrl) ?? + // eslint-disable-next-line @typescript-eslint/no-explicit-any + data?.photos?.map((item: any) => item.imageUrl) ?? + [] + ); + }, [data]); + + const { base64List } = useBase64Images({ imageUrls: images }); + + const calculatedWidth = width ?? BASE_WIDTH * scale; + const calculatedHeight = calculatedWidth * BASE_ASPECT_RATIO; + const calculatedScale = calculatedWidth / BASE_WIDTH; + + const scaledFontSize = BASE_FONT_SIZE * calculatedScale; + const scaledNamePosition = { + bottom: `${BASE_NAME_POSITION.bottom * calculatedScale}px`, + left: `${BASE_NAME_POSITION.left * calculatedScale}px`, + }; + const scaledDatePosition = { + bottom: `${BASE_DATE_POSITION.bottom * calculatedScale}px`, + right: `${BASE_DATE_POSITION.right * calculatedScale}px`, + }; + + return ( +
+ + + {eventName && ( + + {eventName} + + )} + + {eventDate && ( + + {eventDate} + + )} + + {/* 이 컴포넌트에서만 사용되는 폰트 */} + +
+ ); +} diff --git a/src/feature/album/4cut/components/ScreenAlbum4Cut.tsx b/src/feature/album/4cut/components/ScreenAlbum4Cut.tsx new file mode 100644 index 00000000..6fafc0b7 --- /dev/null +++ b/src/feature/album/4cut/components/ScreenAlbum4Cut.tsx @@ -0,0 +1,276 @@ +'use client'; +import { useGetUserMe } from '@/feature/main/hooks/useGetUserMe'; +import { EP } from '@/global/api/ep'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import LongButton from '@/global/components/LongButton'; +import ConfirmModal from '@/global/components/modal/ConfirmModal'; +import Toast from '@/global/components/toast/Toast'; +import BubbleTooltip from '@/global/components/tooltip/BubbleTooltip'; +import PersonSvg from '@/global/svg/PersonSvg'; +import { downloadFile } from '@/global/utils/downloadFile'; +import { getDeviceType } from '@/global/utils/getDeviceType'; +import { extractHtmlToBlob } from '@/global/utils/image/extractHtmlToBlob'; +import { shareImage } from '@/global/utils/image/shareImage'; +import { shareViaNavigator } from '@/global/utils/shareNavigator'; +import { useQueryClient } from '@tanstack/react-query'; +import { Download, Loader2, LucideIcon, Menu, Send } from 'lucide-react'; +import dynamic from 'next/dynamic'; +import Link from 'next/link'; +import { useRouter } from 'next/navigation'; +import { useRef, useState } from 'react'; +import { useGetAlbumInfo } from '../../detail/hooks/useGetAlbumInfo'; +import { use4CutFixed } from '../hooks/use4CutFixed'; +import { use4CutPreviewQuery } from '../hooks/use4CutPreviewQuery'; +import Container4Cut from './Container4Cut'; +const Capture4CutPortal = dynamic(() => import('./Capture4CutPortal'), { + ssr: false, +}); + +interface ScreenAlbum4CutProps { + albumId: string; +} + +export default function ScreenAlbum4Cut({ albumId }: ScreenAlbum4CutProps) { + const router = useRouter(); + const queryClient = useQueryClient(); + const [isConfirmed, setIsConfirmed] = useState(false); + const [isCaptureVisible, setIsCaptureVisible] = useState(false); + const [isDownloading, setIsDownloading] = useState(false); + const captureRef = useRef(null); + const { data } = useGetAlbumInfo(albumId); + const { data: { name } = {} } = useGetUserMe(); + + // TODO : openapi type이 이상해서 임시 any처리. 백엔드랑 협의 필요 + + const { + data: { myRole, previewPhotos, isFinalized } = {}, + isPending: is4CutPreviewPending, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + }: any = use4CutPreviewQuery(albumId); + const { mutateAsync } = use4CutFixed(); + + const isMaker = myRole === 'MAKER'; + + const showCaptureNode = async () => + new Promise((resolve) => { + setIsCaptureVisible(true); + requestAnimationFrame(() => resolve()); + }); + + const handleConfirm = async (): Promise => { + await mutateAsync({ + albumId, + photoIds: previewPhotos.map( + (photo: { photoId: number; imageUrl: string; photoRank: number }) => + photo.photoId, + ), + }); + queryClient.invalidateQueries({ + queryKey: [EP.cheese4cut.preview(albumId)], + }); + setIsConfirmed(true); + }; + + const handleDownload = async () => { + const deviceType = getDeviceType(); + + if (!captureRef.current) { + Toast.alert( + '다운로드할 이미지를 찾지 못했어요. 잠시 후 다시 시도해주세요.', + ); + return; + } + + try { + setIsDownloading(true); + await showCaptureNode(); + + const fileName = data?.title + ? `${data.title}-cheese-4cut.png` + : 'cheese-4cut.png'; + const blob = await extractHtmlToBlob(captureRef.current); + + if (deviceType === 'ios') { + await shareImage({ + imageBlobs: blob, + imageTitle: fileName, + onError: () => { + downloadFile(blob, fileName); + }, + }); + } else { + downloadFile(blob, fileName); + } + } catch (error) { + console.error(error); + Toast.alert('이미지를 다운로드하지 못했습니다. 다시 시도해주세요.'); + } finally { + setIsCaptureVisible(false); + setIsDownloading(false); + } + }; + + const handleShare = async () => { + if (!captureRef.current) { + Toast.alert('공유할 이미지를 찾지 못했어요. 잠시 후 다시 시도해주세요.'); + return; + } + + try { + await showCaptureNode(); + + const fileName = data?.title + ? `${data.title}-cheese-4cut.png` + : 'cheese-4cut.png'; + const blob = await extractHtmlToBlob(captureRef.current); + + await shareImage({ + imageBlobs: blob, + imageTitle: fileName, + onError: () => { + downloadFile(blob, fileName); + }, + }); + } catch (error) { + console.error('Failed to share 4cut preview:', error); + Toast.alert('이미지를 생성하지 못했습니다. 다시 시도해주세요.'); + } finally { + setIsCaptureVisible(false); + } + }; + + return ( + <> + + + + +
+ } + /> +
+ {!isFinalized && ( +
현재 TOP 4 사진
+ )} +
+ +
+
+ + {!is4CutPreviewPending && ( +
+ {isMaker || isFinalized ? ( + <> + {isFinalized ? ( +
+ + +
+ ) : ( + <> +
+ 띱 진행상황 +
+ + + + + {`${data?.currentParticipant} / ${data?.participant}`}{' '} + 명 + +
+
+ } + title='이대로 확정하시겠어요?' + description='예쁜 치즈네컷을 만들어드릴게요' + confirmText='확정하기' + onConfirm={handleConfirm} + /> + + )} + + ) : ( +
+ + { + if (!data) return; + + await shareViaNavigator({ + data: { + title: `'${data.title}'앨범에 대한 치즈네컷을 선정해주세요`, + text: `${name}님이 메이커님에게 조르기를 요청했어요!`, + url: `https://say-cheese.me/album/4cut/${albumId}`, + }, + errorMessage: + '공유에 실패하였습니다. 다시한번 시도해주세요.', + }); + }} + /> +
+ )} +
+ )} + {isDownloading && ( +
+
+ + + 다운로드 중... + +
+
+ )} + + + ); +} + +interface ActionButtonProps { + icon: LucideIcon; + text: string; + onClick: () => void; +} + +export const ActionButton = ({ + icon: Icon, + text, + onClick, +}: ActionButtonProps) => ( + +); diff --git a/src/feature/album/4cut/hooks/use4CutFixed.ts b/src/feature/album/4cut/hooks/use4CutFixed.ts new file mode 100644 index 00000000..d5b6eeee --- /dev/null +++ b/src/feature/album/4cut/hooks/use4CutFixed.ts @@ -0,0 +1,25 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +interface Cheese4CutFixedProps { + albumId: string; + photoIds: number[]; +} + +const fetchData = async ({ albumId, photoIds }: Cheese4CutFixedProps) => { + const res = await api.post({ + path: EP.cheese4cut.finalize(albumId), + body: { photoIds }, + }); + return res.result; +}; + +export function use4CutFixed() { + const mutation = useMutation({ + mutationFn: ({ albumId, photoIds }: Cheese4CutFixedProps) => + fetchData({ albumId, photoIds }), + }); + + return mutation; +} diff --git a/src/feature/album/4cut/hooks/use4CutPreviewQuery.ts b/src/feature/album/4cut/hooks/use4CutPreviewQuery.ts new file mode 100644 index 00000000..00607614 --- /dev/null +++ b/src/feature/album/4cut/hooks/use4CutPreviewQuery.ts @@ -0,0 +1,24 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const fetchData = async (albumId: string) => { + const response = await api.get({ + path: EP.cheese4cut.preview(albumId), + }); + + return response.result; +}; + +export function use4CutPreviewQuery( + albumId: string, + options?: UseQueryOptions, +) { + const query = useQuery({ + queryKey: [EP.cheese4cut.preview(albumId)], + queryFn: () => fetchData(albumId), + ...options, + }); + + return query; +} diff --git a/src/feature/album/4cut/svg/Svg4Cut.tsx b/src/feature/album/4cut/svg/Svg4Cut.tsx new file mode 100644 index 00000000..e2c83675 --- /dev/null +++ b/src/feature/album/4cut/svg/Svg4Cut.tsx @@ -0,0 +1,71 @@ +const PHOTO_SLOTS = [ + { x: 65, y: 78.5 }, + { x: 552, y: 78.5 }, + { x: 552, y: 788.5 }, + { x: 65, y: 788.5 }, +] as const; + +type PhotoUrl = string | null | undefined; + +interface Svg4CutProps { + width: number | `${number}`; + height: number | `${number}`; + photos?: ReadonlyArray; +} + +export default function Svg4Cut({ photos, height, width }: Svg4CutProps) { + const photoSources = photos ?? []; + + return ( + + {PHOTO_SLOTS.map((pos, i) => { + if (!photoSources[i]) return null; + + return ( + + ); + })} + + + + + + + + + + + ); +} diff --git a/src/feature/album/detail/api/getPhotoListByAlbumId.server.ts b/src/feature/album/detail/api/getPhotoListByAlbumId.server.ts new file mode 100644 index 00000000..725505f6 --- /dev/null +++ b/src/feature/album/detail/api/getPhotoListByAlbumId.server.ts @@ -0,0 +1,49 @@ +import { PhotoSorting } from '@/global/api/ep'; +import { serverApi } from '@/global/utils/serverApi'; + +export interface GetPhotoListParams { + page?: number; + size?: number; + sorting?: PhotoSorting; +} + +export interface Photo { + photoId: number; + thumbnailUrl: string; + likesCnt: number; + isLiked: boolean; + isDownloaded: boolean; + isRecentlyDownloaded: boolean; +} + +export interface PhotoListResult { + responses: Photo[]; + listSize: number; + isFirst: boolean; + isLast: boolean; + hasNext: boolean; +} + +/** + * 앨범의 사진 리스트를 가져오는 API (서버/클라이언트 공용) + * @param albumId 앨범 코드(string) + * @param params page, size, sorting 쿼리 파라미터 + * @returns 사진 리스트 + */ +export async function getPhotoListByAlbumId( + albumId: string, + params?: GetPhotoListParams, +): Promise { + if (!albumId) throw new Error('albumId가 필요합니다'); + const queryParams: Record = {}; + if (params) { + if (params.page !== undefined) queryParams.page = params.page; + if (params.size !== undefined) queryParams.size = params.size; + if (params.sorting !== undefined) queryParams.sorting = params.sorting; + } + const res = await serverApi.get({ + path: `/v1/album/${albumId}/photos`, + params: queryParams, + }); + return res.result as PhotoListResult; +} diff --git a/src/feature/album/detail/components/AlbumBestCut.tsx b/src/feature/album/detail/components/AlbumBestCut.tsx new file mode 100644 index 00000000..bcc55d49 --- /dev/null +++ b/src/feature/album/detail/components/AlbumBestCut.tsx @@ -0,0 +1,37 @@ +import LongButton from '@/global/components/LongButton'; +import { useRouter } from 'next/navigation'; +import AlbumBestCutPhotoList from './AlbumBestCutPhotoList'; + +interface AlbumBestCutProps { + albumId: string; + photoCount?: number; +} + +export default function AlbumBestCut({ + albumId, + photoCount, +}: AlbumBestCutProps) { + const router = useRouter(); + + if (photoCount === undefined || photoCount === 0) return null; + + return ( +
+

+ 앨범 베스트컷 +

+ +
+ +
+ + router.push(`/album/4cut/${albumId}`)} + noFixed + disabled={photoCount < 4} + height={48} + /> +
+ ); +} diff --git a/src/feature/album/detail/components/AlbumBestCutPhotoList.tsx b/src/feature/album/detail/components/AlbumBestCutPhotoList.tsx new file mode 100644 index 00000000..533c3614 --- /dev/null +++ b/src/feature/album/detail/components/AlbumBestCutPhotoList.tsx @@ -0,0 +1,46 @@ +import { useAlbumPhotosInfiniteQuery } from '@/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery'; +import PhotoBox from '@/global/components/photo/PhotoBox'; +import { buildQuery } from '@/global/utils/buildQuery'; +import { useRouter } from 'next/navigation'; + +interface AlbumBestCutPhotoListProps { + albumId: string; +} + +export default function AlbumBestCutPhotoList({ + albumId, +}: AlbumBestCutPhotoListProps) { + const router = useRouter(); + const { isPending, isError, items } = useAlbumPhotosInfiniteQuery({ + code: albumId, + size: 4, + sorting: 'POPULAR', + // 좋아요 누른것 실시간으로 반영되게 매번 호출 + refetchOnMount: 'always', + }); + + // Layout Shifting 방지 위해 height 고정 + if (isPending) return
; + if (isError) return
; + + return ( +
+ {items.map(({ thumbnailUrl, photoId, likeCnt, isLiked }) => { + if (!photoId) return null; + + return ( + + router.push(`/photo/detail/${albumId}${buildQuery({ photoId })}`) + } + imageSrc={thumbnailUrl} + likeCount={likeCnt} + liked={isLiked} + /> + ); + })} +
+ ); +} diff --git a/src/feature/album/detail/components/AlbumBottomActions.tsx b/src/feature/album/detail/components/AlbumBottomActions.tsx new file mode 100644 index 00000000..3d532ff1 --- /dev/null +++ b/src/feature/album/detail/components/AlbumBottomActions.tsx @@ -0,0 +1,43 @@ +import DownloadActionBar from './DownloadActionBar'; +import NavBarAlbumDetail from './NavBarAlbumDetail'; +import { AlbumDetailMode } from './ScreenAlbumDetail'; +import UploadButtonInDetail from './UploadButtonInDetail'; + +interface AlbumBottomActionsProps { + mode: AlbumDetailMode; + albumId: string; + changeAlbumMode: (newMode: AlbumDetailMode) => void; + selectedCount: number; + totalPhotoCount?: number; + isLoading: boolean; +} + +export default function AlbumBottomActions({ + mode, + albumId, + changeAlbumMode, + selectedCount, + totalPhotoCount, + isLoading, +}: AlbumBottomActionsProps) { + if (isLoading) return null; + + if (!totalPhotoCount) + return ; + + if (mode === 'default') { + return ; + } + + if (mode === 'select') { + return ( + + ); + } + + return null; +} diff --git a/src/feature/album/detail/components/AlbumInfoSummary.tsx b/src/feature/album/detail/components/AlbumInfoSummary.tsx new file mode 100644 index 00000000..59502b62 --- /dev/null +++ b/src/feature/album/detail/components/AlbumInfoSummary.tsx @@ -0,0 +1,42 @@ +import { AlbumInvitationResponseSchema } from '@/global/api/ep'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; + +// 😀 <- 이 이모지 +const DEFAULT_EMOJI = 'U+1F600'; + +interface AlbumInfoSummaryProps { + albumInfo?: AlbumInvitationResponseSchema; + isLoading: boolean; + isError: boolean; +} + +export function AlbumInfoSummary({ + albumInfo, + isLoading, + isError, +}: AlbumInfoSummaryProps) { + if (isLoading) { + return
; + } + if (isError || !albumInfo) { + return
; + } + + const emoji = convertUnicodeToEmoji(albumInfo.themeEmoji ?? DEFAULT_EMOJI); + + return ( +
+
+ {emoji} +
+
+

+ {albumInfo.title} +

+ + {albumInfo.eventDate} + +
+
+ ); +} diff --git a/src/feature/album/detail/components/AlbumInfos.tsx b/src/feature/album/detail/components/AlbumInfos.tsx new file mode 100644 index 00000000..5e7c6242 --- /dev/null +++ b/src/feature/album/detail/components/AlbumInfos.tsx @@ -0,0 +1,43 @@ +'use client'; + +import { AlbumInvitationResponseSchema } from '@/global/api/ep'; +import { useAlbumTypeStore } from '@/store/useAlbumTypeStore'; +import { forwardRef } from 'react'; +import { useShallow } from 'zustand/shallow'; +import AlbumBestCut from './AlbumBestCut'; +import { AlbumInfoSummary } from './AlbumInfoSummary'; + +interface AlbumInfosProps { + albumId: string; + albumInfo?: AlbumInvitationResponseSchema; + isLoading: boolean; + isError: boolean; + photoCount?: number; +} + +const AlbumInfos = forwardRef( + ({ albumId, photoCount, ...rest }, ref) => { + const { albumType, setAlbumType } = useAlbumTypeStore( + useShallow((state) => ({ + albumType: state.albumType, + setAlbumType: state.setAlbumType, + })), + ); + + if (albumType === 'deep') return null; + + return ( +
+ + +
+ ); + }, +); + +AlbumInfos.displayName = 'AlbumInfos'; + +export default AlbumInfos; diff --git a/src/feature/album/detail/components/AlbumPhotoSection.tsx b/src/feature/album/detail/components/AlbumPhotoSection.tsx new file mode 100644 index 00000000..20d1ee9f --- /dev/null +++ b/src/feature/album/detail/components/AlbumPhotoSection.tsx @@ -0,0 +1,81 @@ +import { PhotoListResponseSchema } from '@/global/api/ep'; +import Spinner from '@/global/components/Spinner'; +import { useAlbumTypeStore } from '@/store/useAlbumTypeStore'; +import { + FetchNextPageOptions, + InfiniteQueryObserverResult, +} from '@tanstack/react-query'; +import { useShallow } from 'zustand/shallow'; +import NoPhotoBody from './NoPhotoBody'; +import PhotoList from './PhotoList'; +import { AlbumDetailMode } from './ScreenAlbumDetail'; + +interface AlbumPhotoSectionProps { + isLoading: boolean; + photos: PhotoListResponseSchema[]; + selectionResetKey: number; + albumId: string; + mode: AlbumDetailMode; + onChangeMode: (mode: AlbumDetailMode) => void; + fetchNextPage: ( + options?: FetchNextPageOptions, + ) => Promise; + hasNextPage: boolean; + isFetchingNextPage: boolean; + totalPhotoCount?: number; +} + +export default function AlbumPhotoSection({ + isLoading, + photos, + selectionResetKey, + albumId, + mode, + onChangeMode, + fetchNextPage, + hasNextPage, + isFetchingNextPage, + totalPhotoCount, +}: AlbumPhotoSectionProps) { + const { albumType } = useAlbumTypeStore( + useShallow((state) => ({ + albumType: state.albumType, + setAlbumType: state.setAlbumType, + })), + ); + + if (isLoading) + return ( +
+ +
+ ); + + if (photos.length === 0) { + return ( + + ); + } + + return ( + + ); +} diff --git a/src/feature/album/detail/components/DownloadActionBar.tsx b/src/feature/album/detail/components/DownloadActionBar.tsx new file mode 100644 index 00000000..c48aae5c --- /dev/null +++ b/src/feature/album/detail/components/DownloadActionBar.tsx @@ -0,0 +1,79 @@ +import { usePhotoDownloadMutation } from '@/feature/photo-detail/hooks/usePhotoDownloadMutation'; +import Toast from '@/global/components/toast/Toast'; +import { downloadFile } from '@/global/utils/downloadFile'; +import { getDeviceType } from '@/global/utils/getDeviceType'; +import { shareImage } from '@/global/utils/image/shareImage'; +import { useSelectedPhotosStore } from '@/store/useSelectedPhotosStore'; +import { useShallow } from 'zustand/shallow'; +import { AlbumDetailMode } from './ScreenAlbumDetail'; + +interface DownloadActionBarProps { + albumId: string; + selectedCount: number; + changeAlbumMode: (newMode: AlbumDetailMode) => void; +} + +export default function DownloadActionBar({ + albumId, + selectedCount, + changeAlbumMode, +}: DownloadActionBarProps) { + const { mutateAsync } = usePhotoDownloadMutation(); + const { selectedPhotos, clearSelectedPhotos } = useSelectedPhotosStore( + useShallow((state) => ({ + selectedPhotos: state.selectedPhotos, + clearSelectedPhotos: state.clearSelectedPhotos, + })), + ); + + const handleDownload = async () => { + if (selectedCount === 0) return; + + const photoUrls = selectedPhotos.map((photo) => photo.url); + const photoIds = selectedPhotos.map((photo) => photo.id); + try { + const deviceType = getDeviceType(); + + if (deviceType === 'ios') { + shareImage({ + imageUrls: photoUrls, + onSuccess: () => { + changeAlbumMode('default'); + clearSelectedPhotos(); + window.scrollTo({ top: 0, behavior: 'smooth' }); + mutateAsync({ albumId, photoIds }); + }, + onError: () => { + Toast.alert('사진을 준비하는 중 오류가 발생했습니다.'); + }, + }); + } else { + await downloadFile(photoUrls); + changeAlbumMode('default'); + clearSelectedPhotos(); + window.scrollTo({ top: 0, behavior: 'smooth' }); + mutateAsync({ albumId, photoIds }); + } + } catch (e) { + console.error('사진 다운로드 처리 중 오류 발생:', e); + } + }; + + const isDisabled = selectedCount === 0; + + return ( +
+
+ {selectedCount}장의 사진이 선택됨 +
+ +
+ ); +} diff --git a/src/feature/album/detail/components/NavBarAlbumDetail.tsx b/src/feature/album/detail/components/NavBarAlbumDetail.tsx new file mode 100644 index 00000000..2ead600e --- /dev/null +++ b/src/feature/album/detail/components/NavBarAlbumDetail.tsx @@ -0,0 +1,103 @@ +'use client'; +import { handleFileUpload } from '@/feature/create-album/utils/handleFileUpload'; +import ToggleAlbumType from '@/feature/main/components/open-album/ToggleAlbumType'; +import BottomSheetModal from '@/global/components/modal/BottomSheetModal'; +import Toast from '@/global/components/toast/Toast'; +import { useAlbumSortStore } from '@/store/useAlbumSortStore'; +import { useAlbumTypeStore } from '@/store/useAlbumTypeStore'; +import { ArrowDownUp, Plus } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import { useRef } from 'react'; +import { useShallow } from 'zustand/shallow'; +import SelectPhotoSortType from './SelectPhotoSortType'; + +export type AlbumType = 'all' | 'deep'; + +interface NavBarAlbumDetailProps { + albumId: string; +} + +export default function NavBarAlbumDetail({ albumId }: NavBarAlbumDetailProps) { + const router = useRouter(); + const fileInputRef = useRef(null); + const { sortType, setSortType } = useAlbumSortStore( + useShallow((state) => ({ + sortType: state.sortType, + setSortType: state.setSortType, + })), + ); + const { albumType, setAlbumType } = useAlbumTypeStore( + useShallow((state) => ({ + albumType: state.albumType, + setAlbumType: state.setAlbumType, + })), + ); + + const handleToggleChange = (value: AlbumType): void => { + setAlbumType(value); + }; + + const handleButtonClick = (): void => { + fileInputRef.current?.click(); + }; + + const onFileChange = async ( + e: React.ChangeEvent, + ): Promise => { + try { + const result = await handleFileUpload(e, albumId, router, { + stay: true, + }); + const success = result?.success ?? 0; + } catch (error: unknown) { + console.error('error', typeof error); + } + }; + + return ( +
+ + + + } + > + setSortType(newType)} + /> + + + + + +
+ ); +} diff --git a/src/feature/album/detail/components/NoPhotoBody.tsx b/src/feature/album/detail/components/NoPhotoBody.tsx new file mode 100644 index 00000000..5b355905 --- /dev/null +++ b/src/feature/album/detail/components/NoPhotoBody.tsx @@ -0,0 +1,34 @@ +import { useQueryClient } from '@tanstack/react-query'; +import Image from 'next/image'; + +interface NoPhotoBodyProps { + text: string; + isRefresh?: boolean; +} + +export default function NoPhotoBody({ text, isRefresh }: NoPhotoBodyProps) { + const queryClient = useQueryClient(); + const handleRefresh = () => { + queryClient.invalidateQueries(); + }; + return ( +
+ 사진 없음 + {text} + {isRefresh && ( + + )} +
+ ); +} diff --git a/src/feature/album/detail/components/PhotoList.tsx b/src/feature/album/detail/components/PhotoList.tsx new file mode 100644 index 00000000..882abc22 --- /dev/null +++ b/src/feature/album/detail/components/PhotoList.tsx @@ -0,0 +1,230 @@ +'use client'; +import { PhotoListResponseSchema } from '@/global/api/ep'; +import PhotoBox from '@/global/components/photo/PhotoBox'; +import Toast from '@/global/components/toast/Toast'; +import { buildQuery } from '@/global/utils/buildQuery'; +import { useAlbumSortStore } from '@/store/useAlbumSortStore'; +import { useAlbumTypeStore } from '@/store/useAlbumTypeStore'; +import { useSelectedPhotosStore } from '@/store/useSelectedPhotosStore'; +import { + type FetchNextPageOptions, + type InfiniteQueryObserverResult, +} from '@tanstack/react-query'; +import { useRouter } from 'next/navigation'; +import { useEffect, useRef } from 'react'; +import { useShallow } from 'zustand/shallow'; +import { AlbumDetailMode } from './ScreenAlbumDetail'; + +const SELECT_MODE_MIN_HEIGHT = '800px'; + +export const ID_PHOTO_LIST = 'photo-list'; +export const ID_PHOTO_LIST_ANCHOR = 'photo-list-anchor'; + +interface PhotoListProps { + albumId: string; + selectable?: boolean; + changeMode: (newMode: AlbumDetailMode) => void; + mode: AlbumDetailMode; + photos: PhotoListResponseSchema[]; + fetchNextPage: ( + options?: FetchNextPageOptions, + ) => Promise; + hasNextPage: boolean; + isFetchingNextPage: boolean; + totalPhotoCount?: number; +} + +export default function PhotoList({ + albumId, + selectable = false, + changeMode, + mode, + photos, + fetchNextPage, + hasNextPage, + isFetchingNextPage, + totalPhotoCount, +}: PhotoListProps) { + const router = useRouter(); + const photoListRef = useRef(null); + const anchorRef = useRef(null); + const loadMoreRef = useRef(null); + const { addSelectedPhoto, deleteSelectedPhoto, isSelected } = + useSelectedPhotosStore( + useShallow((state) => ({ + selectedPhotos: state.selectedPhotos, + addSelectedPhoto: state.addSelectedPhoto, + deleteSelectedPhoto: state.deleteSelectedPhoto, + clearSelectedPhotos: state.clearSelectedPhotos, + isSelected: state.isSelected, + })), + ); + const { sortType, setSortType } = useAlbumSortStore( + useShallow((state) => ({ + sortType: state.sortType, + setSortType: state.setSortType, + })), + ); + const { albumType, setAlbumType } = useAlbumTypeStore( + useShallow((state) => ({ + albumType: state.albumType, + setAlbumType: state.setAlbumType, + })), + ); + + useEffect(() => { + if (!hasNextPage) return; + const target = loadMoreRef.current; + if (!target) return; + + const observer = new IntersectionObserver( + (entries) => { + const entry = entries[0]; + if (entry.isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } + }, + { + rootMargin: '200px 0px', + }, + ); + + observer.observe(target); + + return () => { + observer.disconnect(); + }; + }, [fetchNextPage, hasNextPage, isFetchingNextPage]); + + const handlePhotoPress = ({ + photoId, + photoUrl, + }: { + photoId: number; + photoUrl: string; + }): void => { + if (!selectable) return; + + if (isSelected(photoId)) { + deleteSelectedPhoto(photoId); + } else { + addSelectedPhoto({ id: photoId, url: photoUrl }); + } + }; + + const handleChangeMode = (nextMode: AlbumDetailMode): void => { + const photoListEl = photoListRef.current; + const anchorEl = anchorRef.current; + + if (nextMode === 'select') { + if (photoListEl) { + photoListEl.style.minHeight = SELECT_MODE_MIN_HEIGHT; + } + + if (anchorEl) { + anchorEl.scrollIntoView({ behavior: 'smooth', block: 'start' }); + } + } else { + window.scrollTo({ top: 0, behavior: 'smooth' }); + + // behavior:smooth동작 끝나면 min-height 원복 + setTimeout(() => { + if (photoListEl) { + photoListEl.style.minHeight = ''; + } + }, 300); + } + + changeMode(nextMode); + }; + + return ( +
+
+
+ + 총 {(albumType === 'all' ? totalPhotoCount : photos.length) || 0}장 + + {mode === 'default' && ( + + )} + {mode === 'select' && ( + + )} +
+
+ {photos.map( + ({ + photoId, + likeCnt, + isLiked, + thumbnailUrl, + imageUrl, + isDownloaded, + isRecentlyDownloaded, + }) => { + if (!photoId || !thumbnailUrl || !imageUrl) { + return null; + } + + return ( + { + if (mode === 'default') { + router.push( + `/photo/detail/${albumId}${buildQuery({ photoId: photoId })}`, + ); + } else { + handlePhotoPress({ photoId, photoUrl: imageUrl }); + } + }} + onDisabledPress={() => { + if (mode === 'default') { + router.push( + `/photo/detail/${albumId}${buildQuery({ photoId: photoId })}`, + ); + } else { + Toast.alert( + `금방 다운받은 사진이에요.\n1시간 뒤에 다시 시도하세요.`, + ); + } + }} + /> + ); + }, + )} +
+
+
+ ); +} diff --git a/src/feature/album/detail/components/ScreenAlbumDetail.tsx b/src/feature/album/detail/components/ScreenAlbumDetail.tsx new file mode 100644 index 00000000..0c4485bf --- /dev/null +++ b/src/feature/album/detail/components/ScreenAlbumDetail.tsx @@ -0,0 +1,227 @@ +'use client'; + +import EmojiLoading from '@/components/ui/EmojiLoading'; +import { useAlbumPhotosInfiniteQuery } from '@/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery'; +import { + useAlbumPhotosLikedInfiniteQuery, + type AlbumPhotosLikedItem, +} from '@/feature/photo-detail/hooks/useAlbumPhotosLikedInfiniteQuery'; +import { PhotoListResponseSchema } from '@/global/api/ep'; +import CustomHeader, { + HEADER_HEIGHT, +} from '@/global/components/header/CustomHeader'; +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; +import { useAlbumSortStore } from '@/store/useAlbumSortStore'; +import { useAlbumTypeStore } from '@/store/useAlbumTypeStore'; +import { useSelectedPhotosStore } from '@/store/useSelectedPhotosStore'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { Menu } from 'lucide-react'; +import Link from 'next/link'; +import { useRouter } from 'next/navigation'; +import { useEffect, useMemo, useRef, useState } from 'react'; +import { useShallow } from 'zustand/shallow'; +import { photoSortToApiSorting } from '../constants/photoSort'; +import { useGetAlbumAvailableCount } from '../hooks/useGetAlbumAvailableCount'; +import { useGetAlbumInvitation } from '../hooks/useGetAlbumInvitation'; +import AlbumBottomActions from './AlbumBottomActions'; +import AlbumInfos from './AlbumInfos'; +import AlbumPhotoSection from './AlbumPhotoSection'; + +export type AlbumDetailMode = 'select' | 'default'; + +interface ScreenAlbumDetailProps { + albumId: string; +} + +const LOADING_MODAL_DURATION = 5000; + +export default function ScreenAlbumDetail({ albumId }: ScreenAlbumDetailProps) { + const router = useRouter(); + const albumInfosRef = useRef(null); + const [mode, setMode] = useState('default'); + const [isAlbumInfosHidden, setIsAlbumInfosHidden] = useState(false); + const [selectionResetKey, setSelectionResetKey] = useState(0); + const { sortType, setSortType } = useAlbumSortStore( + useShallow((state) => ({ + sortType: state.sortType, + setSortType: state.setSortType, + })), + ); + const { albumType, setAlbumType } = useAlbumTypeStore( + useShallow((state) => ({ + albumType: state.albumType, + setAlbumType: state.setAlbumType, + })), + ); + const sorting = photoSortToApiSorting[sortType]; + const { + selectedPhotos, + addSelectedPhoto, + deleteSelectedPhoto, + clearSelectedPhotos, + } = useSelectedPhotosStore( + useShallow((state) => ({ + selectedPhotos: state.selectedPhotos, + addSelectedPhoto: state.addSelectedPhoto, + deleteSelectedPhoto: state.deleteSelectedPhoto, + clearSelectedPhotos: state.clearSelectedPhotos, + })), + ); + const { isUploaded, setUploaded } = useUploadingStore( + useShallow((state) => ({ + isUploaded: state.isUploaded, + setUploaded: state.setUploaded, + })), + ); + + const { + data: invitationData, + isLoading: isInvitationLoading, + isError: isInvitationError, + } = useGetAlbumInvitation(albumId); + const isDeepAlbumType = albumType === 'deep'; + const { data } = useGetAlbumAvailableCount(albumId); + const totalPhotoCount = data?.currentPhotoCount; + const defaultPhotosQuery = useAlbumPhotosInfiniteQuery({ + code: albumId, + sorting, + enabled: albumType === 'all', + // 좋아요 누른것 실시간으로 반영되게 매번 호출 + refetchOnMount: 'always', + }); + + const likedPhotosQuery = useAlbumPhotosLikedInfiniteQuery({ + code: albumId, + enabled: isDeepAlbumType, + // 좋아요 누른것 실시간으로 반영되게 매번 호출 + refetchOnMount: 'always', + }); + + const likedPhotos = useMemo( + () => mapLikedPhotosToPhotoList(likedPhotosQuery.items), + [likedPhotosQuery.items], + ); + + const photos: PhotoListResponseSchema[] = isDeepAlbumType + ? likedPhotos + : defaultPhotosQuery.items; + const fetchNextPage = isDeepAlbumType + ? likedPhotosQuery.fetchNextPage + : defaultPhotosQuery.fetchNextPage; + const hasNextPage = isDeepAlbumType + ? likedPhotosQuery.hasNextPage + : defaultPhotosQuery.hasNextPage; + const isFetchingNextPage = isDeepAlbumType + ? likedPhotosQuery.isFetchingNextPage + : defaultPhotosQuery.isFetchingNextPage; + const isLoading = defaultPhotosQuery.isLoading; + + useEffect(() => { + const target = albumInfosRef.current; + if (!target) return; + + const observer = new IntersectionObserver( + ([entry]) => { + setIsAlbumInfosHidden(!entry.isIntersecting); + }, + { + rootMargin: `-${HEADER_HEIGHT}px 0px 0px 0px`, + }, + ); + + observer.observe(target); + + return () => { + observer.disconnect(); + }; + }, []); + + useEffect(() => { + if (mode === 'select') return; + if (selectedPhotos.length === 0) return; + + clearSelectedPhotos(); + setSelectionResetKey((prev) => prev + 1); + }, [clearSelectedPhotos, photos.length, mode, selectedPhotos.length]); + + useEffect(() => { + return () => { + clearSelectedPhotos(); + }; + }, [clearSelectedPhotos]); + + const handleChangeMode = (newMode: AlbumDetailMode) => setMode(newMode); + + const emoji = invitationData?.themeEmoji; + + return ( + <> + {isUploaded && ( + + )} + router.replace('/main')} + isHidden={mode === 'select'} + title={isAlbumInfosHidden ? (invitationData?.title ?? '') : ''} + rightContent={ +
+ + + +
+ } + /> +
+ + +
+ + + ); +} + +function mapLikedPhotosToPhotoList( + items: AlbumPhotosLikedItem[], +): PhotoListResponseSchema[] { + return items.map((item) => ({ + name: item.name, + photoId: item.photoId, + imageUrl: item.imageUrl, + thumbnailUrl: item.thumbnailUrl, + profileImage: DEFAULT_PROFILE_IMAGE, // 사용되지 않는필드. 타입을위해 임시 DEFAULT 프사 넣음. + likeCnt: item.likeCnt ?? 0, + isLiked: item.isLiked ?? false, + isDownloaded: item.isDownloaded, + isRecentlyDownloaded: item.isRecentlyDownloaded, + })); +} diff --git a/src/feature/album/detail/components/SelectPhotoSortType.tsx b/src/feature/album/detail/components/SelectPhotoSortType.tsx new file mode 100644 index 00000000..e1ccd0b3 --- /dev/null +++ b/src/feature/album/detail/components/SelectPhotoSortType.tsx @@ -0,0 +1,48 @@ +import { DrawerClose } from '@/components/ui/drawer'; +import { Check } from 'lucide-react'; +import type { PhotoSortType } from '../constants/photoSort'; +import { photoSortOptions } from '../constants/photoSort'; + +interface SelectPhotoSortTypeProps { + sort: PhotoSortType; + onChange: (newType: PhotoSortType) => void; +} + +export default function SelectPhotoSortType({ + sort, + onChange, +}: SelectPhotoSortTypeProps) { + return ( +
+
    + {photoSortOptions.map((item) => { + const isActive = sort === item.value; + return ( +
  • + { + onChange(item.value); + }} + className='flex w-full items-center gap-2.5 py-4' + > + + {isActive && ( + + )} + + + {item.label} + + +
  • + ); + })} +
+
+ ); +} diff --git a/src/feature/album/detail/components/UploadButtonInDetail.tsx b/src/feature/album/detail/components/UploadButtonInDetail.tsx new file mode 100644 index 00000000..b243ce57 --- /dev/null +++ b/src/feature/album/detail/components/UploadButtonInDetail.tsx @@ -0,0 +1,68 @@ +'use client'; + +import { handleFileUpload } from '@/feature/create-album/utils/handleFileUpload'; +import { EP } from '@/global/api/ep'; +import LongButton from '@/global/components/LongButton'; +import Toast from '@/global/components/toast/Toast'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { useQueryClient } from '@tanstack/react-query'; +import { useParams, useRouter } from 'next/navigation'; +import { useRef } from 'react'; + +interface UploadButtonInDetailProps { + buttonText?: string; +} + +export default function UploadButtonInDetail({ + buttonText, +}: UploadButtonInDetailProps) { + const fileInputRef = useRef(null); + const isUploaded = useUploadingStore((state) => state.isUploaded); + const router = useRouter(); + const params = useParams(); + const queryClient = useQueryClient(); + const albumId = + typeof params.albumId === 'string' + ? params.albumId + : Array.isArray(params.albumId) + ? params.albumId[0] + : ''; + + const onFileChange = async (e: React.ChangeEvent) => { + if (!e.target.files || e.target.files.length === 0) return; + const { success } = await handleFileUpload(e, albumId, router, { + stay: true, + }); + + await queryClient.invalidateQueries({ + queryKey: [EP.album.photos(albumId)], + }); + + if (success) { + setTimeout(() => Toast.check(`총 ${success}장을 앨범에 채웠어요.`), 2000); + } + }; + + const handleButtonClick = () => { + if (!isUploaded) fileInputRef.current?.click(); + }; + + return ( + <> + + + + ); +} diff --git a/src/feature/album/detail/constants/photoSort.ts b/src/feature/album/detail/constants/photoSort.ts new file mode 100644 index 00000000..60e3e73e --- /dev/null +++ b/src/feature/album/detail/constants/photoSort.ts @@ -0,0 +1,15 @@ +import { PhotoSorting } from '@/global/api/ep'; + +export type PhotoSortType = 'uploaded' | 'shot' | 'liked'; + +export const photoSortOptions: { value: PhotoSortType; label: string }[] = [ + { value: 'uploaded', label: '최근 업로드된 사진순' }, + { value: 'shot', label: '최근 촬영한 시간순' }, + { value: 'liked', label: '띱 많은 순' }, +]; + +export const photoSortToApiSorting: Record = { + uploaded: 'CREATED_AT', + shot: 'CAPTURED_AT', + liked: 'POPULAR', +}; diff --git a/src/feature/album/detail/hooks/useGet4CutPreview.ts b/src/feature/album/detail/hooks/useGet4CutPreview.ts new file mode 100644 index 00000000..eb404159 --- /dev/null +++ b/src/feature/album/detail/hooks/useGet4CutPreview.ts @@ -0,0 +1,25 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const fetchData = async (albumId: string) => { + const response = await api.get({ + path: EP.cheese4cut.preview(albumId), + }); + + return response.result; +}; + +export function useGet4CutPreview( + albumId: string, + options?: UseQueryOptions, +) { + const query = useQuery({ + queryKey: [EP.cheese4cut.preview(albumId)], + queryFn: () => fetchData(albumId), + enabled: !!albumId, + ...options, + }); + + return query; +} diff --git a/src/feature/album/detail/hooks/useGetAlbumAvailableCount.ts b/src/feature/album/detail/hooks/useGetAlbumAvailableCount.ts new file mode 100644 index 00000000..911c13c9 --- /dev/null +++ b/src/feature/album/detail/hooks/useGetAlbumAvailableCount.ts @@ -0,0 +1,25 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const fetchData = async (albumId: string) => { + const response = await api.get({ + path: EP.album.availableCount(albumId), + }); + + return response.result; +}; + +export function useGetAlbumAvailableCount( + albumId: string, + options?: UseQueryOptions, +) { + const query = useQuery({ + queryKey: [EP.album.availableCount(albumId)], + queryFn: () => fetchData(albumId), + enabled: !!albumId, + ...options, + }); + + return query; +} diff --git a/src/feature/album/detail/hooks/useGetAlbumInfo.ts b/src/feature/album/detail/hooks/useGetAlbumInfo.ts new file mode 100644 index 00000000..91a30942 --- /dev/null +++ b/src/feature/album/detail/hooks/useGetAlbumInfo.ts @@ -0,0 +1,25 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const fetchData = async (albumId: string) => { + const response = await api.get({ + path: EP.album.albumInfo(albumId), + }); + + return response.result; +}; + +export function useGetAlbumInfo( + albumId: string, + options?: UseQueryOptions, +) { + const query = useQuery({ + queryKey: [EP.album.albumInfo(albumId)], + queryFn: () => fetchData(albumId), + enabled: !!albumId, + ...options, + }); + + return query; +} diff --git a/src/feature/album/detail/hooks/useGetAlbumInvitation.ts b/src/feature/album/detail/hooks/useGetAlbumInvitation.ts new file mode 100644 index 00000000..cc37e5db --- /dev/null +++ b/src/feature/album/detail/hooks/useGetAlbumInvitation.ts @@ -0,0 +1,22 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const getAlbumInvitation = async (code: string) => { + const response = await api.get({ + path: EP.album.invitation(code), + }); + return response.result; +}; + +export function useGetAlbumInvitation( + albumId: string, + options?: UseQueryOptions, +) { + return useQuery({ + queryKey: [EP.album.invitation(albumId)], + queryFn: () => getAlbumInvitation(albumId), + enabled: !!albumId, + ...options, + }); +} diff --git a/src/feature/album/detail/hooks/useGetPhotoListByAlbumId.server.ts b/src/feature/album/detail/hooks/useGetPhotoListByAlbumId.server.ts new file mode 100644 index 00000000..83901d93 --- /dev/null +++ b/src/feature/album/detail/hooks/useGetPhotoListByAlbumId.server.ts @@ -0,0 +1,22 @@ +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; +import { + getPhotoListByAlbumId, + GetPhotoListParams, +} from '../api/getPhotoListByAlbumId.server'; + +export type PhotoListResponse = Awaited< + ReturnType +>; + +export function useGetPhotoListByAlbumId( + albumId: string, + params?: GetPhotoListParams, + options?: UseQueryOptions, +) { + return useQuery({ + queryKey: ['albumPhotos', albumId, params], + queryFn: () => getPhotoListByAlbumId(albumId, params), + enabled: !!albumId, + ...options, + }); +} diff --git a/src/feature/album/detail/sidebar/components/AlbumParticipants.tsx b/src/feature/album/detail/sidebar/components/AlbumParticipants.tsx new file mode 100644 index 00000000..4971fe47 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/AlbumParticipants.tsx @@ -0,0 +1,54 @@ +import { useGetAlbumInform } from '@/feature/upload/hooks/useGetAlbumInform'; +import BottomSheetModal from '@/global/components/modal/BottomSheetModal'; +import BottomSheetContentShare from './BottomSheetContentShare'; +import ItemParticipant from './ItemParticipant'; + +interface AlbumParticipantsProps { + albumId: string; +} + +export default function AlbumParticipants({ albumId }: AlbumParticipantsProps) { + const { data, isPending, isError } = useGetAlbumInform({ code: albumId }); + + if (isPending) return null; + if (isError) return null; + if (!data) return null; + + return ( +
+
+
+

+ 앨범 참가자 + {`${data.currentParticipantCount}/${data.maxParticipantCount}`} +

+
+ {!data.isExpired && ( + + 친구 초대 + + } + > + + + )} +
+
+ {data.participants?.map(({ isMe, name, profileImage, role }, index) => ( + + ))} +
+
+ ); +} diff --git a/src/feature/album/detail/sidebar/components/BottomSheetContentShare.tsx b/src/feature/album/detail/sidebar/components/BottomSheetContentShare.tsx new file mode 100644 index 00000000..ed1c9141 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/BottomSheetContentShare.tsx @@ -0,0 +1,31 @@ +import CopyShareButton from './CopyShareButton'; +import KakaoShareButton from './KakaoShareButton'; +import MoreShareButton from './MoreShareButton'; +import QrcodeShareButton from './QrcodeShareButton'; + +interface BottomSheetContentShareProps { + albumId: string; +} + +export default function BottomSheetContentShare({ + albumId, +}: BottomSheetContentShareProps) { + return ( +
+
+

+ 친구 초대하기 +

+ + 사진이 채워지는 동안 친구에게 앨범을 공유해보세요. + +
+
+ + + + +
+
+ ); +} diff --git a/src/feature/album/detail/sidebar/components/CopyShareButton.tsx b/src/feature/album/detail/sidebar/components/CopyShareButton.tsx new file mode 100644 index 00000000..ce35018b --- /dev/null +++ b/src/feature/album/detail/sidebar/components/CopyShareButton.tsx @@ -0,0 +1,29 @@ +import Toast from '@/global/components/toast/Toast'; +import { copyToClipboard } from '@/global/utils/copyToClipboard'; +import { Copy } from 'lucide-react'; + +interface CopyShareButtonProps { + albumId: string; +} + +export default function CopyShareButton({ albumId }: CopyShareButtonProps) { + const handleClick = (): void => { + copyToClipboard( + `${process.env.NEXT_PUBLIC_CLIENT_URL}/album/entry/${albumId}`, + ); + Toast.check('링크를 복사했어요'); + }; + + return ( + + ); +} diff --git a/src/feature/album/detail/sidebar/components/ItemParticipant.tsx b/src/feature/album/detail/sidebar/components/ItemParticipant.tsx new file mode 100644 index 00000000..55a983f9 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/ItemParticipant.tsx @@ -0,0 +1,42 @@ +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; + +interface ItemParticipantProps { + name: string; + profileImage?: string; + role?: 'MAKER' | 'GUEST' | 'BLACK'; + isMe?: boolean; +} + +export default function ItemParticipant({ + profileImage, + name, + isMe, + role, +}: ItemParticipantProps) { + return ( +
+
+ 프로필사진 +
+ {name} + {role === 'MAKER' && ( + + 메이커 + + )} + {isMe && ( + + 나 + + )} +
+
+
+ ); +} diff --git a/src/feature/album/detail/sidebar/components/KakaoShareButton.tsx b/src/feature/album/detail/sidebar/components/KakaoShareButton.tsx new file mode 100644 index 00000000..4195888d --- /dev/null +++ b/src/feature/album/detail/sidebar/components/KakaoShareButton.tsx @@ -0,0 +1,36 @@ +import { shareKakao } from '@/global/utils/shareKakao'; +import Image from 'next/image'; + +interface KakaoShareButtonProps { + albumId: string; +} + +export default function KakaoShareButton({ albumId }: KakaoShareButtonProps) { + const handleClick = (): void => { + shareKakao({ + title: '앨범에 초대해요', + description: '치이이즈: 추억은 따끈할 때 제맛', + imageUrl: `https://say-cheese.me/assets/og/og_kakao.png`, + imageWidth: 1200, + imageHeight: 630, + link: `https://say-cheese.me/album/entry/${albumId}`, + }); + }; + + return ( + + ); +} diff --git a/src/feature/album/detail/sidebar/components/MoreShareButton.tsx b/src/feature/album/detail/sidebar/components/MoreShareButton.tsx new file mode 100644 index 00000000..0e8826b2 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/MoreShareButton.tsx @@ -0,0 +1,37 @@ +import { shareViaNavigator } from '@/global/utils/shareNavigator'; +import { Ellipsis } from 'lucide-react'; + +const getAlbumEntryUrl = (albumId: string) => + `${process.env.NEXT_PUBLIC_CLIENT_URL}/album/entry/${albumId}`; + +interface MoreShareButtonProps { + albumId: string; +} + +export default function MoreShareButton({ albumId }: MoreShareButtonProps) { + const handleClick = (): void => { + const shareData = { + title: `우리 공유앨범에 초대합니다 - 치이이즈`, + text: '일주일 뒤에는 앨범이 사라져요!', + url: getAlbumEntryUrl(albumId), + }; + + shareViaNavigator({ + data: shareData, + fallbackMessage: '이 기능을 지원하지 않는 브라우저입니다.', + }); + }; + + return ( + + ); +} diff --git a/src/feature/album/detail/sidebar/components/QrcodeShareButton.tsx b/src/feature/album/detail/sidebar/components/QrcodeShareButton.tsx new file mode 100644 index 00000000..84e74cd3 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/QrcodeShareButton.tsx @@ -0,0 +1,27 @@ +import { QrCode } from 'lucide-react'; +import { useRouter } from 'next/navigation'; + +interface QrcodeShareButtonProps { + albumId: string; +} + +export default function QrcodeShareButton({ albumId }: QrcodeShareButtonProps) { + const router = useRouter(); + + const handleClick = (): void => { + router.push(`/album/qrcode/${albumId}`); + }; + + return ( + + ); +} diff --git a/src/feature/album/detail/sidebar/components/ScreenAlbumSidebar.tsx b/src/feature/album/detail/sidebar/components/ScreenAlbumSidebar.tsx new file mode 100644 index 00000000..ccedaaa3 --- /dev/null +++ b/src/feature/album/detail/sidebar/components/ScreenAlbumSidebar.tsx @@ -0,0 +1,96 @@ +'use client'; + +import { HEADER_HEIGHT } from '@/global/components/header/CustomHeader'; +import ConfirmModal from '@/global/components/modal/ConfirmModal'; +import Toast from '@/global/components/toast/Toast'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import { + formatExpirationTime, + getIsExpired, +} from '@/global/utils/time/formatExpirationTime'; +import { X } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import { useGetAlbumInfo } from '../../hooks/useGetAlbumInfo'; +import { useAlbumExitMutation } from '../hooks/useAlbumExitMutation'; +import AlbumParticipants from './AlbumParticipants'; + +interface ScreenAlbumSidebarProps { + albumId: string; +} + +export default function ScreenAlbumSidebar({ + albumId, +}: ScreenAlbumSidebarProps) { + const router = useRouter(); + const { data, isPending, isError } = useGetAlbumInfo(albumId); + const { mutateAsync } = useAlbumExitMutation(); + + if (isPending) return null; + if (isError) return null; + + const handleExit = async (): Promise => { + try { + await mutateAsync(albumId); + router.replace('/main'); + Toast.check(`${data?.title ? `${data.title} ` : ''}앨범이 삭제됐어요.`); + } catch (e) { + console.log(e); + Toast.alert(`앨범 삭제를 실패하였어요.\n다시한번 시도해주세요.`); + } + }; + const isExpired = getIsExpired(data?.expiredAt); + + return ( + <> +
+
+ +
+ {data?.themeEmoji ? convertUnicodeToEmoji(data?.themeEmoji) : '😀'} +
+

+ {data?.title} +

+

+ {data?.eventDate} +

+ {!isExpired && ( +
+ 앨범 소멸까지 {formatExpirationTime(data?.expiredAt)} +
+ )} +
+ + + +
+ + 앨범 나가기 + + } + title='앨범에서 나갈까요?' + description='나가더라도 내가 올린 사진은 앨범에 남아요.' + cancelText='다음에' + confirmText='앨범 나가기' + confirmClassName='bg-button-accent-fill text-white active:bg-button-accent-pressed active:text-basic-inverse' + onConfirm={handleExit} + /> +
+
+ + ); +} diff --git a/src/feature/album/detail/sidebar/hooks/useAlbumExitMutation.ts b/src/feature/album/detail/sidebar/hooks/useAlbumExitMutation.ts new file mode 100644 index 00000000..9e42e61e --- /dev/null +++ b/src/feature/album/detail/sidebar/hooks/useAlbumExitMutation.ts @@ -0,0 +1,18 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +async function fetchData(albumId: string) { + const res = await api.delete({ + path: EP.album.albumParticipantsMe(albumId), + }); + return res.result; +} + +export function useAlbumExitMutation() { + const mutation = useMutation({ + mutationFn: (albumId: string) => fetchData(albumId), + }); + + return mutation; +} diff --git a/src/feature/album/qrcode/components/CardAlbumQrcode.tsx b/src/feature/album/qrcode/components/CardAlbumQrcode.tsx new file mode 100644 index 00000000..3738ee5f --- /dev/null +++ b/src/feature/album/qrcode/components/CardAlbumQrcode.tsx @@ -0,0 +1,38 @@ +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import QRCode from 'react-qr-code'; +import { useGetAlbumInfo } from '../../detail/hooks/useGetAlbumInfo'; + +interface CardAlbumQrcodeProps { + albumId: string; +} + +export default function CardAlbumQrcode({ albumId }: CardAlbumQrcodeProps) { + const { data, isPending, isError } = useGetAlbumInfo(albumId); + + if (isPending) return null; + if (isError) return null; + if (!data) return null; + + const qrValue = `${process.env.NEXT_PUBLIC_CLIENT_URL}/album/entry/${albumId}`; + + return ( +
+
+ {data.themeEmoji ? convertUnicodeToEmoji(data.themeEmoji) : '😀'} +
+

+ {data.title} +

+

+ {data.eventDate} +

+
+ +
+
+ ); +} diff --git a/src/feature/album/qrcode/components/ScreenAlbumQrcode.tsx b/src/feature/album/qrcode/components/ScreenAlbumQrcode.tsx new file mode 100644 index 00000000..9c15380f --- /dev/null +++ b/src/feature/album/qrcode/components/ScreenAlbumQrcode.tsx @@ -0,0 +1,27 @@ +'use client'; +import { X } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import CardAlbumQrcode from './CardAlbumQrcode'; + +interface ScreenAlbumQrcodeProps { + albumId: string; +} + +export default function ScreenAlbumQrcode({ albumId }: ScreenAlbumQrcodeProps) { + const router = useRouter(); + + return ( +
+ +
+ +
+
+ ); +} diff --git a/src/feature/create-album/api/checkAvailableCount.server.ts b/src/feature/create-album/api/checkAvailableCount.server.ts new file mode 100644 index 00000000..264da435 --- /dev/null +++ b/src/feature/create-album/api/checkAvailableCount.server.ts @@ -0,0 +1,15 @@ +import { serverApi } from '@/global/utils/serverApi'; + +/** + * 앨범에 업로드 가능한 남은 이미지 개수 조회 (Server Component용) + * @param albumId 앨범 ID + * @returns {Promise} 남은 업로드 가능 개수 + */ +export async function checkAvailableCountServer( + albumId: string, +): Promise { + const response = await serverApi.get<{ availableCount: number }>({ + path: `/v1/album/${albumId}/available-count`, + }); + return response.result.availableCount; +} diff --git a/src/feature/create-album/api/checkAvailableCount.ts b/src/feature/create-album/api/checkAvailableCount.ts new file mode 100644 index 00000000..707939aa --- /dev/null +++ b/src/feature/create-album/api/checkAvailableCount.ts @@ -0,0 +1,13 @@ +import { api } from '@/global/utils/api'; + +/** + * 앨범에 업로드 가능한 남은 이미지 개수 조회 (Client Component용) + * @param albumId 앨범 ID + * @returns {Promise} 남은 업로드 가능 개수 + */ +export async function checkAvailableCount(albumId: string): Promise { + const response = await api.get<{ availableCount: number }>({ + path: `/v1/album/${albumId}/available-count`, + }); + return response.result.availableCount; +} diff --git a/src/feature/create-album/api/createAlbumApi.ts b/src/feature/create-album/api/createAlbumApi.ts new file mode 100644 index 00000000..af70cb4b --- /dev/null +++ b/src/feature/create-album/api/createAlbumApi.ts @@ -0,0 +1,33 @@ +import { api } from '../../../global/utils/api'; + +export interface CreateAlbumRequest { + themeEmoji: string; // U+1F9C0 형식의 유니코드 + title: string; + participant: number; + eventDate: string; // YYYY-MM-DD 형식 +} + +export interface CreateAlbumResponse { + themeEmoji: string; + title: string; + eventDate: string; + currentPhotoCnt: number; + code: string; +} + +/** + * 앨범 생성 API + * @param data 앨범 생성 요청 데이터 + * @returns API 응답 전체 (isSuccess, code, message, result 포함) + */ +export async function createAlbumApi(data: CreateAlbumRequest) { + return await api.post({ + path: '/v1/album', + body: { + themeEmoji: data.themeEmoji, + title: data.title, + participant: data.participant, + eventDate: data.eventDate, + }, + }); +} diff --git a/src/feature/create-album/components/AlbumEmojiSelector.tsx b/src/feature/create-album/components/AlbumEmojiSelector.tsx new file mode 100644 index 00000000..0ccdda7c --- /dev/null +++ b/src/feature/create-album/components/AlbumEmojiSelector.tsx @@ -0,0 +1,100 @@ +'use client'; +import { Pencil } from 'lucide-react'; +import dynamic from 'next/dynamic'; +import React, { useEffect, useRef, useState } from 'react'; + +const EmojiPicker = dynamic(() => import('emoji-picker-react'), { + ssr: false, + loading: () => ( +
+ ), +}); + +interface EmojiClickData { + emoji: string; + unified: string; +} + +interface AlbumEmojiSelectorProps { + selectedEmoji: string; + onEmojiSelect: (emoji: string) => void; +} + +const AlbumEmojiSelector = React.memo(function AlbumEmojiSelector({ + selectedEmoji, + onEmojiSelect, +}: AlbumEmojiSelectorProps) { + const [showPicker, setShowPicker] = useState(false); + const pickerRef = useRef(null); + const containerRef = useRef(null); + + const handleEmojiClick = (emojiData: EmojiClickData) => { + onEmojiSelect(emojiData.emoji); + setShowPicker(false); + }; + + useEffect(() => { + if (!showPicker) return; + const handleClick = (e: MouseEvent) => { + if ( + (pickerRef.current && pickerRef.current.contains(e.target as Node)) || + (containerRef.current && + containerRef.current.contains(e.target as Node)) + ) { + return; + } + setShowPicker(false); + }; + document.addEventListener('mousedown', handleClick); + return () => { + document.removeEventListener('mousedown', handleClick); + }; + }, [showPicker]); + + return ( +
+ {/* 이모지 표시 */} +
{ + e.stopPropagation(); + setShowPicker((v) => !v); + }} + > +
+ {selectedEmoji} +
+ +
+ + {/* 이모지 피커 (fixed로 고정) */} + {showPicker && ( +
+ +
+ )} +
+ ); +}); + +export default AlbumEmojiSelector; diff --git a/src/feature/create-album/components/CreateAlbumList.tsx b/src/feature/create-album/components/CreateAlbumList.tsx new file mode 100644 index 00000000..f106abb9 --- /dev/null +++ b/src/feature/create-album/components/CreateAlbumList.tsx @@ -0,0 +1,114 @@ +'use client'; +import LongButton from '@/global/components/LongButton'; +import dynamic from 'next/dynamic'; +const BottomSheetModal = dynamic( + () => import('@/global/components/modal/BottomSheetModal'), + { ssr: false }, +); +import { useRouter } from 'next/navigation'; +import { useState } from 'react'; +import { + useCreateAlbum, + type CreateAlbumApiResponse, + type CreateAlbumError, +} from '../hook/useCreateAlbum'; + +import Toast from '@/global/components/toast/Toast'; +import AlbumEmojiSelector from './AlbumEmojiSelector'; +import CreateInputList from './CreateInputList'; + +// 이모지를 유니코드 코드포인트 형식으로 변환 (예: 😊 → U+1F60A) +const emojiToUnicode = (emoji: string): string => { + const codePoint = emoji.codePointAt(0); + if (!codePoint) return ''; + return `U+${codePoint.toString(16).toUpperCase().padStart(4, '0')}`; +}; + +export default function CreateAlbumList() { + const router = useRouter(); + const [selectedEmoji, setSelectedEmoji] = useState('😊'); + const [eventName, setEventName] = useState(''); + const [eventDate, setEventDate] = useState(''); + const [participantCount, setParticipantCount] = useState(''); + const [hasFormError, setHasFormError] = useState(false); + + const { mutate: createAlbum } = useCreateAlbum(); + + const handleSubmit = () => { + const emojiUnicode = emojiToUnicode(selectedEmoji); + createAlbum( + { + themeEmoji: emojiUnicode, + title: eventName, + participant: parseInt(participantCount, 10), + eventDate, + }, + { + onSuccess: (result: CreateAlbumApiResponse) => { + if (result.result.code) { + router.push(`/create-album/${result.result.code}/complete`); + } + }, + onError: (err: CreateAlbumError) => { + Toast.alert(err.message || '앨범 생성에 실패했습니다.'); + console.error('앨범 생성 실패:', err); + }, + }, + ); + }; + + const participantCountNumber = parseInt(participantCount, 10); + const isFormComplete = + eventName.trim() !== '' && + eventDate.trim() !== '' && + participantCount.trim() !== '' && + participantCountNumber >= 1 && + participantCountNumber <= 64 && + !hasFormError; + + return ( +
+ + + + } + showCloseButton={false} + className='pt-10 pb-5 pl-6' + dismissible={true} + showHandle={false} + > +
+ + 치즈 앨범 메뉴얼 + +
    +
  • • 이 앨범은 7일 뒤 자동으로 사라져요.
  • +
  • • 메이커는 규칙을 어긴 참여자를 내보낼 수 있어요.
  • +
  • • 메이커는 모든 사진을 정리 • 삭제할 수 있어요
  • +
+
+ +
+
+ ); +} diff --git a/src/feature/create-album/components/CreateComplete.tsx b/src/feature/create-album/components/CreateComplete.tsx new file mode 100644 index 00000000..cb924b3c --- /dev/null +++ b/src/feature/create-album/components/CreateComplete.tsx @@ -0,0 +1,32 @@ +'use client'; +import { CountdownTimer } from '@/global/components/CountdownTimer'; +import LongButton from '@/global/components/LongButton'; +import { Check } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import { useCallback } from 'react'; + +interface CreateCompleteProps { + albumId: string; +} + +export default function CreateComplete({ albumId }: CreateCompleteProps) { + const router = useRouter(); + + const handleClick = useCallback(() => { + router.push(`/album/upload/${albumId}`); + }, [router, albumId]); + + return ( +
+
+ +
+
+ 만들기 성공! + 앨범이 열렸어요 +
+ + +
+ ); +} diff --git a/src/feature/create-album/components/CreateInputList.tsx b/src/feature/create-album/components/CreateInputList.tsx new file mode 100644 index 00000000..baab1128 --- /dev/null +++ b/src/feature/create-album/components/CreateInputList.tsx @@ -0,0 +1,114 @@ +'use client'; + +import DateXInput from '@/global/components/DateXInput'; +import XInput from '@/global/components/XInput'; +import { useState } from 'react'; + +interface CreateInputListProps { + eventName: string; + eventDate: string; + participantCount: string; + onEventNameChange: (value: string) => void; + onEventDateChange: (value: string) => void; + onParticipantCountChange: (value: string) => void; + onErrorChange?: (hasError: boolean) => void; +} + +export default function CreateInputList({ + eventName, + eventDate, + participantCount, + onEventNameChange, + onEventDateChange, + onParticipantCountChange, + onErrorChange, +}: CreateInputListProps) { + const [eventNameError, setEventNameError] = useState(''); + const [participantCountError, setParticipantCountError] = useState(''); + + // 이벤트 이름 검증: 13글자 이내의 한글, 영문, 숫자, _, .만 허용 + const handleEventNameChange = (value: string) => { + const validPattern = /^[가-힣ㄱ-ㅎㅏ-ㅣa-zA-Z0-9_. ]*$/; + let error = ''; + + if (!validPattern.test(value)) { + error = + '13글자 이내의 한글, 영문, 숫자, 특수기호(_), (.)만 쓸 수 있어요.'; + } + + setEventNameError(error); + onErrorChange?.(error !== '' || participantCountError !== ''); + onEventNameChange(value); + }; + + const handleParticipantCountChange = (value: string) => { + if (value === '') { + setParticipantCountError(''); + onErrorChange?.(eventNameError !== ''); + onParticipantCountChange(value); + return; + } + + const sanitizedValue = value.replace(/\D/g, ''); + const numberValue = + sanitizedValue === '' ? '' : String(parseInt(sanitizedValue, 10)); + + const number = parseInt(numberValue, 10); + + let error = ''; + + if (numberValue === '' || isNaN(number)) { + error = ''; + onParticipantCountChange(''); + } else if (number === 0) { + error = '최소 1명 이상 가능해요'; + onParticipantCountChange(numberValue); + } else if (number > 64) { + error = '최대 64명까지 가능해요'; + onParticipantCountChange(numberValue); + } else { + error = ''; + onParticipantCountChange(numberValue); + } + + setParticipantCountError(error); + onErrorChange?.(error !== '' || eventNameError !== ''); + }; + + // 로컬 시간대 기준으로 어제 날짜를 YYYY-MM-DD로 계산 + const yesterdayDate = new Date(); + yesterdayDate.setDate(yesterdayDate.getDate() - 1); + const yyyy = yesterdayDate.getFullYear(); + const mm = String(yesterdayDate.getMonth() + 1).padStart(2, '0'); // 0-11이므로 +1 + const dd = String(yesterdayDate.getDate()).padStart(2, '0'); + const yesterday = `${yyyy}-${mm}-${dd}`; + + return ( +
+ + + +
+ ); +} diff --git a/src/feature/create-album/components/WaitingAlbum.tsx b/src/feature/create-album/components/WaitingAlbum.tsx new file mode 100644 index 00000000..9ff65690 --- /dev/null +++ b/src/feature/create-album/components/WaitingAlbum.tsx @@ -0,0 +1,98 @@ +'use client'; + +import CheeseCartLoading from '@/../public/assets/album/CheeseCart_Loading.json'; +import Toast from '@/global/components/toast/Toast'; +import { useImageStore } from '@/store/useImageStore'; +import { motion } from 'framer-motion'; +import dynamic from 'next/dynamic'; +import { useRouter } from 'next/navigation'; +import { useEffect } from 'react'; +const Lottie = dynamic(() => import('lottie-react'), { ssr: false }); + +interface WaitingAlbumProps { + albumId: string; +} + +export default function WaitingAlbum({ albumId }: WaitingAlbumProps) { + const router = useRouter(); + const { images } = useImageStore(); + + useEffect(() => { + const processImages = async () => { + const startTime = Date.now(); + + try { + // 최소 2.5초 대기 보장 + const elapsedTime = Date.now() - startTime; + const remainingTime = Math.max(0, 2500 - elapsedTime); + await new Promise((resolve) => setTimeout(resolve, remainingTime)); + // Zustand에 저장된 이미지가 있으면 → 일부 사진에 문제 → select로 이동 + if (images.length > 0) { + router.replace(`/album/${albumId}/select`); + return; + } + + // 업로드 완료 후 detail 페이지로 이동 + router.replace(`/album/detail/${albumId}`); + } catch (err) { + console.error('Image processing error:', err); + Toast.alert('사진 처리 중 에러가 발생했습니다.'); + router.replace(`/album/detail/${albumId}`); + } + }; + + processImages(); + }, [albumId, images, router]); + + const dotVariants = { + initial: { opacity: 0 }, + animate: { + opacity: 1, + transition: { + duration: 0.5, + repeat: Infinity, + repeatType: 'reverse' as const, + }, + }, + }; + + const containerVariants = { + animate: { + transition: { + staggerChildren: 0.3, + }, + }, + }; + + return ( +
+ +
+ 잠시만 기다려주세요 + + . + . + . + +
+
+ ); +} diff --git a/src/feature/create-album/hook/useCheckImages.ts b/src/feature/create-album/hook/useCheckImages.ts new file mode 100644 index 00000000..fc38d5ef --- /dev/null +++ b/src/feature/create-album/hook/useCheckImages.ts @@ -0,0 +1,21 @@ +import { useMutation, UseMutationOptions } from '@tanstack/react-query'; +import { checkImages, type CheckImagesResult } from '../utils/checkImages'; + +export type CheckImagesVariables = { + files: File[]; + albumId: string; +}; + +export function useCheckImages( + options?: UseMutationOptions< + CheckImagesResult, + unknown, + CheckImagesVariables + >, +) { + return useMutation({ + mutationKey: ['checkImages'], + mutationFn: ({ files, albumId }) => checkImages(files, albumId), + ...options, + }); +} diff --git a/src/feature/create-album/hook/useCreateAlbum.ts b/src/feature/create-album/hook/useCreateAlbum.ts new file mode 100644 index 00000000..352ea8f1 --- /dev/null +++ b/src/feature/create-album/hook/useCreateAlbum.ts @@ -0,0 +1,37 @@ +import { useMutation, UseMutationOptions } from '@tanstack/react-query'; +import { + createAlbumApi, + type CreateAlbumRequest, + type CreateAlbumResponse, +} from '../api/createAlbumApi'; + +export type CreateAlbumApiResponse = { + result: CreateAlbumResponse; + code: number; + isSuccess: boolean; + message: string; +}; + +export type CreateAlbumError = { + code: number; + isSuccess: boolean; + message: string; +}; + +export function useCreateAlbum( + options?: UseMutationOptions< + CreateAlbumApiResponse, + CreateAlbumError, + CreateAlbumRequest + >, +) { + return useMutation< + CreateAlbumApiResponse, + CreateAlbumError, + CreateAlbumRequest + >({ + mutationKey: ['createAlbum'], + mutationFn: createAlbumApi, + ...options, + }); +} diff --git a/src/feature/create-album/utils/checkImages.test.ts b/src/feature/create-album/utils/checkImages.test.ts new file mode 100644 index 00000000..ff51c0e6 --- /dev/null +++ b/src/feature/create-album/utils/checkImages.test.ts @@ -0,0 +1,43 @@ +import { checkAvailableCount } from '@/feature/create-album/api/checkAvailableCount'; +import { validateImages } from '@/feature/create-album/utils/validateImages'; +import { beforeEach, describe, expect, it, vi } from 'vitest'; +import { checkImages } from './checkImages'; + +// Mock dependencies +vi.mock('@/feature/create-album/api/checkAvailableCount', () => ({ + checkAvailableCount: vi.fn(), +})); + +vi.mock('@/feature/create-album/utils/validateImages', () => ({ + validateImages: vi.fn(), +})); + +describe('checkImages', () => { + const mockFiles = [new File([], 'test1.jpg')]; + const albumId = 'test-album-id'; + + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should return oversized files and available count', async () => { + const mockOversizedFiles = ['oversized.jpg']; + const mockAvailableCount = 10; + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateImages as any).mockReturnValue({ + oversizedFiles: mockOversizedFiles, + }); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (checkAvailableCount as any).mockResolvedValue(mockAvailableCount); + + const result = await checkImages(mockFiles, albumId); + + expect(result).toEqual({ + oversizedFiles: mockOversizedFiles, + availableCount: mockAvailableCount, + }); + expect(validateImages).toHaveBeenCalledWith(mockFiles); + expect(checkAvailableCount).toHaveBeenCalledWith(albumId); + }); +}); diff --git a/src/feature/create-album/utils/checkImages.ts b/src/feature/create-album/utils/checkImages.ts new file mode 100644 index 00000000..e7f51fe8 --- /dev/null +++ b/src/feature/create-album/utils/checkImages.ts @@ -0,0 +1,26 @@ +import { checkAvailableCount } from '@/feature/create-album/api/checkAvailableCount'; +import { validateImages } from '@/feature/create-album/utils/validateImages'; + +export type CheckImagesResult = { + oversizedFiles: string[]; + availableCount: number; +}; + +/** + * 업로드 전 이미지 상태 확인 유틸 + * - oversizedFiles: 6MB를 초과한 파일 이름 목록 + * - availableCount: 서버에서 조회한 남은 업로드 가능 개수 + * 선택/표시가 필요한 화면(예: SelectAlbum)에서 사용하세요. + */ +export async function checkImages( + files: File[], + albumId: string, +): Promise { + const { oversizedFiles } = validateImages(files); + const availableCount = await checkAvailableCount(albumId); + + return { + oversizedFiles, + availableCount, + }; +} diff --git a/src/feature/create-album/utils/getFilesWithCaptureTime.ts b/src/feature/create-album/utils/getFilesWithCaptureTime.ts new file mode 100644 index 00000000..eec18412 --- /dev/null +++ b/src/feature/create-album/utils/getFilesWithCaptureTime.ts @@ -0,0 +1,32 @@ +import exifr from 'exifr'; + +export async function getFilesWithCaptureTime( + files: File[], +): Promise> { + return Promise.all( + files.map(async (file) => { + let date: Date = new Date(file.lastModified); + try { + const tags = await exifr.parse(file, { + pick: ['DateTimeOriginal', 'CreateDate', 'ModifyDate'], + translateValues: true, + }); + const dt: Date | undefined = + (tags?.DateTimeOriginal as Date | undefined) ?? + (tags?.CreateDate as Date | undefined) ?? + (tags?.ModifyDate as Date | undefined); + if ( + dt && + typeof dt.getTime === 'function' && + !Number.isNaN(dt.getTime()) + ) { + date = dt; + } + } catch {} + // yyyy-MM-ddTHH:mm:ss 형식으로 변환 + const pad = (n: number) => n.toString().padStart(2, '0'); + const captureTime = `${date.getFullYear()}-${pad(date.getMonth() + 1)}-${pad(date.getDate())}T${pad(date.getHours())}:${pad(date.getMinutes())}:${pad(date.getSeconds())}`; + return { file, captureTime }; + }), + ); +} diff --git a/src/feature/create-album/utils/handleFileUpload.test.ts b/src/feature/create-album/utils/handleFileUpload.test.ts new file mode 100644 index 00000000..ff5d1f0a --- /dev/null +++ b/src/feature/create-album/utils/handleFileUpload.test.ts @@ -0,0 +1,114 @@ +import { presignedAndUploadToNCP } from '@/global/api/presignedAndUploadToNCP'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { beforeEach, describe, expect, it, vi } from 'vitest'; +import { getFilesWithCaptureTime } from './getFilesWithCaptureTime'; +import { handleFileUpload } from './handleFileUpload'; +import { convertHeicFilesToJpeg } from './heicToJpeg'; +import { saveFilesToStore } from './saveFilesToStore'; +import { sortImagesByDate } from './sortImagesByDate'; +import { validateUpload } from './validateUpload'; + +// Mock dependencies +vi.mock('@/global/api/presignedAndUploadToNCP', () => ({ + presignedAndUploadToNCP: vi.fn(), +})); + +vi.mock('@/store/useUploadingStore', () => ({ + useUploadingStore: { + getState: vi.fn().mockReturnValue({ + setUploaded: vi.fn(), + setUploadedCount: vi.fn(), + }), + }, +})); + +vi.mock('./getFilesWithCaptureTime', () => ({ + getFilesWithCaptureTime: vi.fn(), +})); + +vi.mock('./heicToJpeg', () => ({ + convertHeicFilesToJpeg: vi.fn(), +})); + +vi.mock('./saveFilesToStore', () => ({ + saveFilesToStore: vi.fn(), +})); + +vi.mock('./sortImagesByDate', () => ({ + sortImagesByDate: vi.fn(), +})); + +vi.mock('./validateUpload', () => ({ + validateUpload: vi.fn(), +})); + +describe('handleFileUpload', () => { + const mockRouter = { push: vi.fn(), replace: vi.fn() }; + const albumId = 'test-album-id'; + const mockFile = new File(['content'], 'test.jpg', { type: 'image/jpeg' }); + const mockEvent = { + target: { + files: [mockFile], + value: 'some-value', + }, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } as any; + + beforeEach(() => { + vi.clearAllMocks(); + // Default mock implementations + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (sortImagesByDate as any).mockResolvedValue([mockFile]); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (convertHeicFilesToJpeg as any).mockResolvedValue([mockFile]); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateUpload as any).mockResolvedValue({ ok: true }); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (getFilesWithCaptureTime as any).mockResolvedValue([ + { file: mockFile, captureTime: '2023-01-01' }, + ]); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (presignedAndUploadToNCP as any).mockResolvedValue({ + success: 1, + failed: 0, + }); + }); + + it('should orchestrate the upload flow correctly', async () => { + await handleFileUpload(mockEvent, albumId, mockRouter); + + expect(sortImagesByDate).toHaveBeenCalled(); + expect(convertHeicFilesToJpeg).toHaveBeenCalled(); + expect(validateUpload).toHaveBeenCalled(); + expect(useUploadingStore.getState().setUploaded).toHaveBeenCalledWith(true); + expect(mockRouter.push).toHaveBeenCalledWith(`/album/${albumId}/waiting`); + expect(getFilesWithCaptureTime).toHaveBeenCalled(); + expect(presignedAndUploadToNCP).toHaveBeenCalled(); + expect(useUploadingStore.getState().setUploadedCount).toHaveBeenCalledWith( + 1, + ); + }); + + it('should handle validation failure', async () => { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateUpload as any).mockResolvedValue({ ok: false, reason: 'size' }); + + await handleFileUpload(mockEvent, albumId, mockRouter); + + expect(saveFilesToStore).toHaveBeenCalled(); + expect(mockRouter.push).toHaveBeenCalledWith(`/album/${albumId}/waiting`); + expect(presignedAndUploadToNCP).not.toHaveBeenCalled(); + }); + + it('should not redirect if stay option is true', async () => { + await handleFileUpload(mockEvent, albumId, mockRouter, { stay: true }); + + expect(mockRouter.push).not.toHaveBeenCalled(); + }); + + it('should clear input value after execution', async () => { + await handleFileUpload(mockEvent, albumId, mockRouter); + + expect(mockEvent.target.value).toBe(''); + }); +}); diff --git a/src/feature/create-album/utils/handleFileUpload.ts b/src/feature/create-album/utils/handleFileUpload.ts new file mode 100644 index 00000000..1e453380 --- /dev/null +++ b/src/feature/create-album/utils/handleFileUpload.ts @@ -0,0 +1,65 @@ +import { presignedAndUploadToNCP } from '@/global/api/presignedAndUploadToNCP'; +import { useUploadingStore } from '@/store/useUploadingStore'; +import { ChangeEvent } from 'react'; +import { getFilesWithCaptureTime } from './getFilesWithCaptureTime'; +import { convertHeicFilesToJpeg } from './heicToJpeg'; +import { saveFilesToStore } from './saveFilesToStore'; +import { sortImagesByDate } from './sortImagesByDate'; +import { validateUpload } from './validateUpload'; + +export async function handleFileUpload( + e: ChangeEvent, + albumId: string, + router?: { push: (path: string) => void; replace: (path: string) => void }, + options?: { stay?: boolean }, +): Promise<{ + success?: number; + failed?: number; + failedPhotoIds?: number[]; +}> { + const fl = e.target.files; + if (!fl) return {}; + + const startTime = Date.now(); + let uploadResult: { success: number; failed: number } | null = null; + + try { + let files = Array.from(fl).filter((f) => f.type.startsWith('image/')); + files = await sortImagesByDate(files); + files = await convertHeicFilesToJpeg(files); + + const validationResult = await validateUpload(files, albumId); + if (validationResult.ok) { + useUploadingStore.getState().setUploaded(true); + if (!options?.stay && router) { + router.push(`/album/${albumId}/waiting`); + } + const filesWithCapture = await getFilesWithCaptureTime(files); + const fileInfos = filesWithCapture.map(({ file, captureTime }) => ({ + fileName: file.name, + fileSize: file.size, + contentType: file.type, + captureTime, + })); + uploadResult = await presignedAndUploadToNCP({ + albumCode: albumId, + files, + fileInfos, + }); + + if (uploadResult.success > 0) { + useUploadingStore.getState().setUploadedCount(uploadResult.success); + } + } else { + saveFilesToStore(files); + if (router) { + router.push(`/album/${albumId}/waiting`); + } + return {}; + } + } finally { + if (e.target) e.target.value = ''; + } + + return uploadResult || {}; +} diff --git a/src/feature/create-album/utils/heicToJpeg.test.ts b/src/feature/create-album/utils/heicToJpeg.test.ts new file mode 100644 index 00000000..3ad1fa07 --- /dev/null +++ b/src/feature/create-album/utils/heicToJpeg.test.ts @@ -0,0 +1,75 @@ +import Toast from '@/global/components/toast/Toast'; +import { beforeEach, describe, expect, it, vi } from 'vitest'; +import { convertHeicFilesToJpeg } from './heicToJpeg'; + +// Mock dependencies +vi.mock('@/global/components/toast/Toast', () => ({ + default: { + alert: vi.fn(), + }, +})); + +vi.mock('heic-to', () => ({ + heicTo: vi.fn().mockImplementation(async ({ blob }) => { + // Return a dummy blob as if it were a JPEG + return new Blob(['dummy-jpeg-content'], { type: 'image/jpeg' }); + }), +})); + +describe('convertHeicFilesToJpeg', () => { + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should return original files if no HEIC files are present', async () => { + const files = [ + new File(['content'], 'test1.jpg', { type: 'image/jpeg' }), + new File(['content'], 'test2.png', { type: 'image/png' }), + ]; + + const result = await convertHeicFilesToJpeg(files); + + expect(result).toHaveLength(2); + expect(result[0].name).toBe('test1.jpg'); + expect(result[1].name).toBe('test2.png'); + expect(Toast.alert).not.toHaveBeenCalled(); + }); + + it('should convert HEIC files to JPEG', async () => { + const files = [ + new File(['heic-content'], 'image.heic', { type: 'image/heic' }), + new File(['jpg-content'], 'image.jpg', { type: 'image/jpeg' }), + ]; + + const result = await convertHeicFilesToJpeg(files); + + expect(result).toHaveLength(2); + // The first file should be converted to jpg + expect(result[0].name).toBe('image.jpg'); + expect(result[0].type).toBe('image/jpeg'); + // The second file should remain as is + expect(result[1].name).toBe('image.jpg'); + expect(Toast.alert).toHaveBeenCalledWith( + '1개의 HEIC 파일을 변환 중입니다...', + ); + }); + + it('should handle conversion failure gracefully', async () => { + const { heicTo } = await import('heic-to'); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (heicTo as any).mockRejectedValueOnce(new Error('Conversion failed')); + + const files = [ + new File(['heic-content'], 'fail.heic', { type: 'image/heic' }), + ]; + + const result = await convertHeicFilesToJpeg(files); + + expect(result).toHaveLength(1); + // Should return original file on failure + expect(result[0].name).toBe('fail.heic'); + expect(Toast.alert).toHaveBeenCalledWith( + 'fail.heic 파일의 변환에 실패했습니다.', + ); + }); +}); diff --git a/src/feature/create-album/utils/heicToJpeg.ts b/src/feature/create-album/utils/heicToJpeg.ts new file mode 100644 index 00000000..2f729977 --- /dev/null +++ b/src/feature/create-album/utils/heicToJpeg.ts @@ -0,0 +1,41 @@ +import Toast from '@/global/components/toast/Toast'; +import { heicTo } from 'heic-to'; + +export async function convertHeicFilesToJpeg(files: File[]): Promise { + const heicFiles = files.filter( + (file) => + /heic|heif/i.test(file.type) || /\.heic$|\.heif$/i.test(file.name), + ); + + if (heicFiles.length > 0) { + Toast.alert(`${heicFiles.length}개의 HEIC 파일을 변환 중입니다...`); + } + + const convertedFiles = await Promise.all( + files.map(async (file) => { + if (/heic|heif/i.test(file.type) || /\.heic$|\.heif$/i.test(file.name)) { + try { + const jpegBlob = await heicTo({ + blob: file, + type: 'image/jpeg', + quality: 0.9, + }); + return new File( + [jpegBlob], + file.name.replace(/\.(heic|heif)$/i, '.jpg'), + { + type: 'image/jpeg', + lastModified: file.lastModified, + }, + ); + } catch (e) { + Toast.alert(`${file.name} 파일의 변환에 실패했습니다.`); + return file; + } + } + return file; + }), + ); + + return convertedFiles; +} diff --git a/src/feature/create-album/utils/saveFilesToStore.ts b/src/feature/create-album/utils/saveFilesToStore.ts new file mode 100644 index 00000000..5cbe3c69 --- /dev/null +++ b/src/feature/create-album/utils/saveFilesToStore.ts @@ -0,0 +1,15 @@ +import { useImageStore } from '@/store/useImageStore'; + +/** + * 이미지 파일 배열을 zustand 이미지 스토어에 저장 + * @param files File[] + */ +export function saveFilesToStore(files: File[]) { + const setImages = useImageStore.getState().setImages; + setImages( + files.map((file) => ({ + id: `${file.name}-${crypto.randomUUID()}`, + file, + })), + ); +} diff --git a/src/feature/create-album/utils/sortImagesByDate.ts b/src/feature/create-album/utils/sortImagesByDate.ts new file mode 100644 index 00000000..91258d30 --- /dev/null +++ b/src/feature/create-album/utils/sortImagesByDate.ts @@ -0,0 +1,11 @@ +import { getFilesWithCaptureTime } from './getFilesWithCaptureTime'; + +// captureTime(yyyy-MM-ddTHH:mm:ss) 기준 최신순 내림차순 정렬 +export async function sortImagesByDate(files: File[]): Promise { + const filesWithCapture = await getFilesWithCaptureTime(files); + filesWithCapture.sort( + (a, b) => + new Date(b.captureTime).getTime() - new Date(a.captureTime).getTime(), + ); + return filesWithCapture.map((f) => f.file); +} diff --git a/src/feature/create-album/utils/validateImages.ts b/src/feature/create-album/utils/validateImages.ts new file mode 100644 index 00000000..ce6ae113 --- /dev/null +++ b/src/feature/create-album/utils/validateImages.ts @@ -0,0 +1,27 @@ +const MAX_SIZE = 6 * 1024 * 1024; // 6MB in bytes + +export function validateImage(file: File): boolean { + return file.size <= MAX_SIZE; +} + +export function validateImageCount(files: File[]): number { + return files.filter((file) => !validateImage(file)).length; +} + +export function validateImages(files: File[]): { + valid: boolean; + oversizedFiles: string[]; +} { + const oversizedFiles: string[] = []; + + files.forEach((file) => { + if (!validateImage(file)) { + oversizedFiles.push(file.name); + } + }); + + return { + valid: oversizedFiles.length === 0, + oversizedFiles, + }; +} diff --git a/src/feature/create-album/utils/validateUpload.test.ts b/src/feature/create-album/utils/validateUpload.test.ts new file mode 100644 index 00000000..b7fc9567 --- /dev/null +++ b/src/feature/create-album/utils/validateUpload.test.ts @@ -0,0 +1,74 @@ +import { checkAvailableCount } from '@/feature/create-album/api/checkAvailableCount'; +import { validateImageCount } from '@/feature/create-album/utils/validateImages'; +import { beforeEach, describe, expect, it, vi } from 'vitest'; +import { validateUpload } from './validateUpload'; + +// Mock dependencies +vi.mock('@/feature/create-album/api/checkAvailableCount', () => ({ + checkAvailableCount: vi.fn(), +})); + +vi.mock('@/feature/create-album/utils/validateImages', () => ({ + validateImageCount: vi.fn(), +})); + +describe('validateUpload', () => { + const mockFiles = [new File([], 'test1.jpg'), new File([], 'test2.jpg')]; + const albumId = 'test-album-id'; + + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should return ok: true when validation passes', async () => { + // Mock validateImageCount to return 0 (no oversized files) + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateImageCount as any).mockReturnValue(0); + // Mock checkAvailableCount to return enough space + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (checkAvailableCount as any).mockResolvedValue(5); + + const result = await validateUpload(mockFiles, albumId); + + expect(result).toEqual({ ok: true }); + }); + + it('should return ok: false with reason: size when files are oversized', async () => { + // Mock validateImageCount to return 1 (1 oversized file) + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateImageCount as any).mockReturnValue(1); + // Mock checkAvailableCount to return enough space + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (checkAvailableCount as any).mockResolvedValue(5); + + const result = await validateUpload(mockFiles, albumId); + + expect(result).toEqual({ ok: false, reason: 'size' }); + }); + + it('should return ok: false with reason: count when file count exceeds limit', async () => { + // Mock validateImageCount to return 0 + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateImageCount as any).mockReturnValue(0); + // Mock checkAvailableCount to return less than file count + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (checkAvailableCount as any).mockResolvedValue(1); + + const result = await validateUpload(mockFiles, albumId); + + expect(result).toEqual({ ok: false, reason: 'count' }); + }); + + it('should return ok: false with reason: both when both checks fail', async () => { + // Mock validateImageCount to return 1 + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (validateImageCount as any).mockReturnValue(1); + // Mock checkAvailableCount to return less than file count + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (checkAvailableCount as any).mockResolvedValue(1); + + const result = await validateUpload(mockFiles, albumId); + + expect(result).toEqual({ ok: false, reason: 'both' }); + }); +}); diff --git a/src/feature/create-album/utils/validateUpload.ts b/src/feature/create-album/utils/validateUpload.ts new file mode 100644 index 00000000..b3d52040 --- /dev/null +++ b/src/feature/create-album/utils/validateUpload.ts @@ -0,0 +1,36 @@ +import { checkAvailableCount } from '@/feature/create-album/api/checkAvailableCount'; +import { validateImageCount } from '@/feature/create-album/utils/validateImages'; + +export type ValidateUploadResult = + | { ok: true } + | { ok: false; reason: 'size' | 'count' | 'both' }; + +/** + * 이미지 업로드 전 검증 (용량, 업로드 가능 개수) + * WaitingAlbum에서 사용 - 성공/실패 reason만 반환 + * 상세 정보가 필요하면 SelectAlbumBody에서 validateImages 직접 사용 + */ +export async function validateUpload( + files: File[], + albumId: string, +): Promise { + const oversizedCount = validateImageCount(files); + const sizeValid = oversizedCount === 0; + + const availableCount = await checkAvailableCount(albumId); + const countValid = files.length <= availableCount; + + if (!sizeValid && !countValid) { + return { ok: false, reason: 'both' }; + } + + if (!sizeValid) { + return { ok: false, reason: 'size' }; + } + + if (!countValid) { + return { ok: false, reason: 'count' }; + } + + return { ok: true }; +} diff --git a/src/feature/login/components/KakaoSignupButton.tsx b/src/feature/login/components/KakaoSignupButton.tsx new file mode 100644 index 00000000..71337fdc --- /dev/null +++ b/src/feature/login/components/KakaoSignupButton.tsx @@ -0,0 +1,37 @@ +'use client'; +import { buildQuery } from '@/global/utils/buildQuery'; +import Image from 'next/image'; +import { useSearchParams } from 'next/navigation'; + +const KAKAO_AUTH_URL = `https://dev.say-cheese.me/oauth2/authorization/kakao`; + +export default function KakaoSignupButton() { + const searchParams = useSearchParams(); + const redirect = searchParams.get('redirect'); + + const handleKakaoLogin = async () => { + try { + const kakaoUrl = redirect + ? `${KAKAO_AUTH_URL}${buildQuery({ redirect })}` + : KAKAO_AUTH_URL; + + window.location.href = kakaoUrl; + } catch (err) { + console.error('카카오 인증 GET 요청 실패:', err); + } + }; + return ( +
+ 카카오 로고 + 카카오 로그인 +
+ ); +} diff --git a/src/feature/main/closed-album/components/ClosedAlbumSectionList.tsx b/src/feature/main/closed-album/components/ClosedAlbumSectionList.tsx new file mode 100644 index 00000000..a794d397 --- /dev/null +++ b/src/feature/main/closed-album/components/ClosedAlbumSectionList.tsx @@ -0,0 +1,30 @@ +import CloseAlbum from '@/feature/main/components/close-album/CloseAlbum'; +import { type ClosedAlbumSection } from '../utils/buildClosedAlbumSections'; + +interface ClosedAlbumSectionListProps { + sections: ClosedAlbumSection[]; +} + +export default function ClosedAlbumSectionList({ + sections, +}: ClosedAlbumSectionListProps) { + if (sections.length === 0) return null; + + return sections.map(({ year, albums }) => ( +
+

{year}

+
+ {albums.map((album) => ( + + ))} +
+
+ )); +} diff --git a/src/feature/main/closed-album/components/ScreenMainClosedAlbum.tsx b/src/feature/main/closed-album/components/ScreenMainClosedAlbum.tsx new file mode 100644 index 00000000..e6255888 --- /dev/null +++ b/src/feature/main/closed-album/components/ScreenMainClosedAlbum.tsx @@ -0,0 +1,69 @@ +'use client'; +import EmptyAlbum from '@/feature/main/components/EmptyAlbum'; +import { useAlbumClosedInfiniteQuery } from '@/feature/main/hooks/useAlbumClosedInfiniteQuery'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import { useEffect, useMemo, useRef } from 'react'; +import { buildClosedAlbumSections } from '../utils/buildClosedAlbumSections'; +import ClosedAlbumSectionList from './ClosedAlbumSectionList'; + +const LOADING_TEXT = '불러오는 중...'; + +interface ScreenMainClosedAlbumProps {} + +export default function ScreenMainClosedAlbum({}: ScreenMainClosedAlbumProps) { + const { items, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading } = + useAlbumClosedInfiniteQuery(); + const loadMoreRef = useRef(null); + + const sections = useMemo(() => buildClosedAlbumSections(items), [items]); + const showLoadingState = isLoading && items.length === 0; + const showEmptyState = !isLoading && items.length === 0; + + useEffect(() => { + if (!hasNextPage) return; + const target = loadMoreRef.current; + if (!target) return; + + const observer = new IntersectionObserver( + (entries) => { + const entry = entries[0]; + if (entry.isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } + }, + { rootMargin: '200px 0px' }, + ); + + observer.observe(target); + + return () => { + observer.disconnect(); + }; + }, [fetchNextPage, hasNextPage, isFetchingNextPage]); + + return ( +
+ +
+ {showLoadingState && ( +
+ {LOADING_TEXT} +
+ )} + + {!showLoadingState && showEmptyState && ( + + )} + + {!showLoadingState && } + + {isFetchingNextPage && ( +
+ {LOADING_TEXT} +
+ )} +
+
+
+ ); +} diff --git a/src/feature/main/closed-album/utils/buildClosedAlbumSections.ts b/src/feature/main/closed-album/utils/buildClosedAlbumSections.ts new file mode 100644 index 00000000..ee62553e --- /dev/null +++ b/src/feature/main/closed-album/utils/buildClosedAlbumSections.ts @@ -0,0 +1,78 @@ +import { type AlbumClosedItem } from '@/feature/main/hooks/useAlbumClosedInfiniteQuery'; +import { formatEventDate } from '@/global/utils/formatEventDate'; + +const MAX_IMAGES = 4; + +export interface ClosedAlbumSectionItem { + code: string; + title: string; + date: string; + author: string; + images: string[]; + rawDate: string; +} + +export interface ClosedAlbumSection { + year: string; + albums: ClosedAlbumSectionItem[]; +} + +export function buildClosedAlbumSections( + items: AlbumClosedItem[], +): ClosedAlbumSection[] { + const grouped = items.reduce>( + (acc, item) => { + const year = extractYear(item.eventDate); + if (!acc[year]) { + acc[year] = []; + } + + acc[year].push({ + code: item.code, + title: item.title ?? '', + date: formatEventDate(item.eventDate), + author: item.makerName ?? '', + images: + item.thumbnails + ?.filter((thumbnail): thumbnail is string => Boolean(thumbnail)) + ?.slice(0, MAX_IMAGES) ?? [], + rawDate: item.eventDate ?? '', + }); + + return acc; + }, + {}, + ); + + return Object.entries(grouped) + .sort(([a], [b]) => toComparableYear(b) - toComparableYear(a)) + .map(([year, albums]) => ({ + year, + albums: albums.sort((a, b) => compareByDateDesc(a.rawDate, b.rawDate)), + })); +} + +function toComparableYear(year: string) { + const yearNumber = Number(year); + return Number.isNaN(yearNumber) ? 0 : yearNumber; +} + +function compareByDateDesc(a: string, b: string) { + const aTime = Date.parse(a); + const bTime = Date.parse(b); + if (Number.isNaN(aTime) || Number.isNaN(bTime)) { + return 0; + } + + return bTime - aTime; +} + +function extractYear(date?: string) { + if (!date) return '기타'; + const parsed = new Date(date); + if (Number.isNaN(parsed.getTime())) { + return date.slice(0, 4) || '기타'; + } + + return String(parsed.getFullYear()); +} diff --git a/src/feature/main/components/EmptyAlbum.tsx b/src/feature/main/components/EmptyAlbum.tsx new file mode 100644 index 00000000..10ef93fc --- /dev/null +++ b/src/feature/main/components/EmptyAlbum.tsx @@ -0,0 +1,16 @@ +import EmptySvg from './svg/EmptySvg'; + +interface EmptyAlbumProps { + title: string; +} + +export default function EmptyAlbum({ title }: EmptyAlbumProps) { + return ( +
+ +

+ {title} +

+
+ ); +} diff --git a/src/feature/main/components/ScreenMain.tsx b/src/feature/main/components/ScreenMain.tsx new file mode 100644 index 00000000..645ba45f --- /dev/null +++ b/src/feature/main/components/ScreenMain.tsx @@ -0,0 +1,28 @@ +'use client'; +import LogoHeader from '@/global/components/header/LogoHeader'; +import LongButton from '@/global/components/LongButton'; +import { useRouter } from 'next/navigation'; +import CloseAlbumContainer from './close-album/CloseAlbumContainer'; +import OpenAlbumContainer from './open-album/OpenAlbumContainer'; +import ProfileMypage from './profile/ProfileMypage'; + +interface ScreenMainProps {} + +export default function ScreenMain({}: ScreenMainProps) { + const router = useRouter(); + + return ( +
+ + + + +
+ router.push('/create-album')} + /> +
+
+ ); +} diff --git a/src/feature/main/components/close-album/CloseAlbum.tsx b/src/feature/main/components/close-album/CloseAlbum.tsx new file mode 100644 index 00000000..4d486415 --- /dev/null +++ b/src/feature/main/components/close-album/CloseAlbum.tsx @@ -0,0 +1,53 @@ +import Link from 'next/link'; +import SvgCloseAlbumEmptyPhoto from '../svg/SvgCloseAlbumEmptyPhoto'; + +interface CloseAlbumProps { + code: string; + title: string; + date: string; + author: string; + images: string[]; +} + +export default function CloseAlbum({ + code, + title, + date, + author, + images, +}: CloseAlbumProps) { + return ( + +
+ {images.length < 4 ? ( +
+ +
+ ) : ( +
+ {images.slice(0, 4).map((src, i) => ( + {`이미지${i + ))} +
+ )} +
+ +
+

+ {title} +

+
+ {date} · {author} +
+
+ + ); +} diff --git a/src/feature/main/components/close-album/CloseAlbumContainer.tsx b/src/feature/main/components/close-album/CloseAlbumContainer.tsx new file mode 100644 index 00000000..f6c256d8 --- /dev/null +++ b/src/feature/main/components/close-album/CloseAlbumContainer.tsx @@ -0,0 +1,93 @@ +'use client'; +import { ChevronRight } from 'lucide-react'; +import Link from 'next/link'; +import { useEffect, useMemo, useRef } from 'react'; +import { useAlbumClosedInfiniteQuery } from '../../hooks/useAlbumClosedInfiniteQuery'; +import { mapClosedAlbumItems } from '../../utils/mapClosedAlbumItems'; +import EmptyAlbum from '../EmptyAlbum'; +import CloseAlbum from './CloseAlbum'; + +const LOADING_TEXT = '불러오는 중...'; + +export default function CloseAlbumContainer() { + const { items, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading } = + useAlbumClosedInfiniteQuery(); + const loadMoreRef = useRef(null); + const albums = useMemo(() => mapClosedAlbumItems(items), [items]); + const showLoadingState = isLoading && albums.length === 0; + const showEmptyState = !isLoading && albums.length === 0; + + useEffect(() => { + if (!hasNextPage) return; + const target = loadMoreRef.current; + if (!target) return; + + const observer = new IntersectionObserver( + (entries) => { + const entry = entries[0]; + if (entry.isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } + }, + { + rootMargin: '200px 0px', + }, + ); + + observer.observe(target); + + return () => { + observer.disconnect(); + }; + }, [fetchNextPage, hasNextPage, isFetchingNextPage]); + + return ( +
+ {albums.length > 0 ? ( +
+ +

+ 닫힌 앨범 {albums.length} + +

+ +
+ ) : ( +

+ 닫힌 앨범 0 +

+ )} +
+ {showLoadingState && ( +
+ {LOADING_TEXT} +
+ )} + {!showLoadingState && + albums.map((album) => ( + + ))} + {!showLoadingState && showEmptyState && ( + + )} +
+ {isFetchingNextPage && ( +
+ {LOADING_TEXT} +
+ )} +
+
+ ); +} diff --git a/src/feature/main/components/open-album/ButtonMore.tsx b/src/feature/main/components/open-album/ButtonMore.tsx new file mode 100644 index 00000000..592a88d3 --- /dev/null +++ b/src/feature/main/components/open-album/ButtonMore.tsx @@ -0,0 +1,25 @@ +import { ChevronDown } from 'lucide-react'; + +interface ButtonMoreProps { + onClick?: () => void; + moreCount: number; +} + +export default function ButtonMore({ onClick, moreCount }: ButtonMoreProps) { + const handleClick = () => { + onClick?.(); + }; + + return ( + + ); +} diff --git a/src/feature/main/components/open-album/OpenAlbum.tsx b/src/feature/main/components/open-album/OpenAlbum.tsx new file mode 100644 index 00000000..64f09599 --- /dev/null +++ b/src/feature/main/components/open-album/OpenAlbum.tsx @@ -0,0 +1,146 @@ +import PersonSvg from '@/global/svg/PersonSvg'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import Image from 'next/image'; +import Link from 'next/link'; +import { getOpenAlbumGridConfig } from '../../utils/getOpenAlbumGridConfig'; + +const defaultThumbnail = '/assets/album/bg-album-default.png'; + +interface OpenAlbumProps { + code: string; + expirationTime: string; + title: string; + date: string; + author: string; + totalMembers: number; + joinedMembers: number; + thumbnails?: string[]; + emoji: string; +} + +export default function OpenAlbum({ + code, + expirationTime, + title, + date, + author, + totalMembers, + joinedMembers, + thumbnails = [], + emoji, +}: OpenAlbumProps) { + const [main, side1, side2] = thumbnails; + const count = thumbnails.length; + const gridConfig = getOpenAlbumGridConfig(count); + + return ( + +
+
+
+
+ {/* 메인 썸네일 */} +
+ {main ? ( + 메인 사진 + ) : ( + 메인 사진 + )} +
+ + {/* 썸네일 2개 이상일 때 side1 */} + {count >= 2 && ( +
+ {side1 && ( + 주요 사진 + )} +
+ )} + + {/* 썸네일 3개일 때만 side2 */} + {count >= 3 && ( +
+ {side2 && ( + 주요 사진 + )} +
+ )} +
+
+ + {/* 소멸까지 */} +
+ + 소멸까지 {expirationTime} + +
+
+ + {/* 아래 정보 */} +
+
+ {emoji ? convertUnicodeToEmoji(emoji) : '😀'} +
+
+

+ {title} +

+

+ {`${date} · ${author}`} +

+
+ +
+
+ + + {joinedMembers} / {totalMembers} 명 + +
+
+
+
+ + ); +} diff --git a/src/feature/main/components/open-album/OpenAlbumContainer.tsx b/src/feature/main/components/open-album/OpenAlbumContainer.tsx new file mode 100644 index 00000000..655aaafc --- /dev/null +++ b/src/feature/main/components/open-album/OpenAlbumContainer.tsx @@ -0,0 +1,121 @@ +'use client'; +import { useEffect, useMemo, useRef, useState } from 'react'; +import { + useAlbumOpenInfiniteQuery, + type AlbumOpenType, +} from '../../hooks/useAlbumOpenInfiniteQuery'; +import { mapOpenAlbumItems } from '../../utils/mapOpenAlbumItems'; +import EmptyAlbum from '../EmptyAlbum'; +import ButtonMore from './ButtonMore'; +import OpenAlbum from './OpenAlbum'; +import SkeletonOpenAlbum from './SkeletonOpenAlbum'; +import ToggleAlbumType from './ToggleAlbumType'; + +const MIN_VISIBLE_COUNT = 2; + +interface OpenAlbumContainerProps {} + +export default function OpenAlbumContainer({}: OpenAlbumContainerProps) { + const [albumType, setAlbumType] = useState('all'); + const [isMoreOpened, setIsMoreOpened] = useState(false); + const [hasOpenedMine, setHasOpenedMine] = useState(false); + const loadMoreRef = useRef(null); + + useEffect(() => { + if (albumType === 'mine' && !hasOpenedMine) { + setHasOpenedMine(true); + } + }, [albumType, hasOpenedMine]); + + useEffect(() => { + setIsMoreOpened(false); + }, [albumType]); + + const allQuery = useAlbumOpenInfiniteQuery({ type: 'all' }); + const mineQuery = useAlbumOpenInfiniteQuery({ + type: 'mine', + enabled: albumType === 'mine' || hasOpenedMine, + }); + + const activeQuery = albumType === 'all' ? allQuery : mineQuery; + const { + items: activeItems, + fetchNextPage, + hasNextPage, + isFetchingNextPage, + isLoading, + } = activeQuery; + + const albums = useMemo(() => mapOpenAlbumItems(activeItems), [activeItems]); + const showing = isMoreOpened ? albums : albums.slice(0, MIN_VISIBLE_COUNT); + const moreCount = Math.max(albums.length - MIN_VISIBLE_COUNT, 0); + const showMoreButton = !isMoreOpened && moreCount > 0; + const showLoadingState = isLoading && albums.length === 0; + const showEmptyState = !isLoading && albums.length === 0; + + useEffect(() => { + if (!isMoreOpened) return; + if (!hasNextPage) return; + + const target = loadMoreRef.current; + if (!target) return; + + const observer = new IntersectionObserver( + (entries) => { + const entry = entries[0]; + if (entry.isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } + }, + { + rootMargin: '200px 0px', + }, + ); + + observer.observe(target); + + return () => { + observer.disconnect(); + }; + }, [fetchNextPage, hasNextPage, isFetchingNextPage, isMoreOpened]); + + return ( +
+

+ 열린 앨범 {albums.length} +

+ setAlbumType(next)} + /> +
+ {!showLoadingState && + showing.map((album) => )} + {showLoadingState && ( + <> + + + + )} + {!showLoadingState && showEmptyState && ( + + )} +
+ {showMoreButton && ( + setIsMoreOpened(true)} + /> + )} + +
+
+ ); +} diff --git a/src/feature/main/components/open-album/SkeletonOpenAlbum.tsx b/src/feature/main/components/open-album/SkeletonOpenAlbum.tsx new file mode 100644 index 00000000..5463f0cd --- /dev/null +++ b/src/feature/main/components/open-album/SkeletonOpenAlbum.tsx @@ -0,0 +1,7 @@ +interface SkeletonOpenAlbumProps {} + +export default function SkeletonOpenAlbum({}: SkeletonOpenAlbumProps) { + return ( +
+ ); +} diff --git a/src/feature/main/components/open-album/ToggleAlbumType.tsx b/src/feature/main/components/open-album/ToggleAlbumType.tsx new file mode 100644 index 00000000..d4f74ea4 --- /dev/null +++ b/src/feature/main/components/open-album/ToggleAlbumType.tsx @@ -0,0 +1,40 @@ +'use client'; + +interface ToggleAlbumTypeProps { + value: T; + onChange: (next: T) => void; + labels: Record; +} + +export default function ToggleAlbumType({ + value, + onChange, + labels, +}: ToggleAlbumTypeProps) { + const keys = Object.keys(labels) as T[]; + + return ( +
+ {keys.map((key) => { + const isActive = value === key; + const label = labels[key]; + + return ( + + ); + })} +
+ ); +} diff --git a/src/feature/main/components/profile/ProfileMypage.tsx b/src/feature/main/components/profile/ProfileMypage.tsx new file mode 100644 index 00000000..43db4294 --- /dev/null +++ b/src/feature/main/components/profile/ProfileMypage.tsx @@ -0,0 +1,72 @@ +'use client'; +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; +import { Settings } from 'lucide-react'; +import Link from 'next/link'; +import { useRouter } from 'next/navigation'; +import { useGetUserMe } from '../../hooks/useGetUserMe'; + +interface ProfileMypageProps {} + +export default function ProfileMypage({}: ProfileMypageProps) { + const router = useRouter(); + const { data } = useGetUserMe(); + + return ( +
+
+
+ 프로필사진 +
+ +
+
+ + {data?.name ?? '사용자'} + + + + + +
+ +
+
+ + {data?.albumCount ?? 0} + + + 앨범 수 + +
+
+ + {data?.photoCount ?? 0} + + + 올린 사진 + +
+
+ + {data?.likesCount ?? 0} + + + 받은 띱 + +
+
+
+
+
+ ); +} diff --git a/src/feature/main/components/svg/EmptySvg.tsx b/src/feature/main/components/svg/EmptySvg.tsx new file mode 100644 index 00000000..83bb749f --- /dev/null +++ b/src/feature/main/components/svg/EmptySvg.tsx @@ -0,0 +1,35 @@ +interface EmptySvgProps {} + +export default function EmptySvg({}: EmptySvgProps) { + return ( + + + + + + ); +} diff --git a/src/feature/main/components/svg/SvgCloseAlbumEmptyPhoto.tsx b/src/feature/main/components/svg/SvgCloseAlbumEmptyPhoto.tsx new file mode 100644 index 00000000..3b0afc35 --- /dev/null +++ b/src/feature/main/components/svg/SvgCloseAlbumEmptyPhoto.tsx @@ -0,0 +1,26 @@ +interface SvgCloseAlbumEmptyPhotoProps {} + +export default function SvgCloseAlbumEmptyPhoto({}: SvgCloseAlbumEmptyPhotoProps) { + return ( + + + + + + + + + + + + ); +} diff --git a/src/feature/main/hooks/useAlbumClosedInfiniteQuery.ts b/src/feature/main/hooks/useAlbumClosedInfiniteQuery.ts new file mode 100644 index 00000000..2876f63c --- /dev/null +++ b/src/feature/main/hooks/useAlbumClosedInfiniteQuery.ts @@ -0,0 +1,65 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useInfiniteQuery } from '@tanstack/react-query'; + +type AlbumClosedPage = NonNullable; +export type AlbumClosedItem = AlbumClosedPage['responses'][number]; + +interface FetchPageParams { + pageParam: number; + /** 페이지 사이즈 (기본 20) */ + size: number; +} + +const fetchData = async ({ + pageParam, + size, +}: FetchPageParams): Promise => { + const res = await api.get({ + path: EP.album.albumClosed(), + params: { page: pageParam, size }, + }); + + const result: AlbumClosedPage = res.result ?? createEmptyPage({ pageParam }); + + return { ...result, page: pageParam }; +}; + +const createEmptyPage = ({ + pageParam, +}: { + pageParam: number; +}): AlbumClosedPage => ({ + responses: [], + listSize: 0, + isFirst: pageParam === 0, + isLast: true, + hasNext: false, +}); + +interface UseAlbumClosedInfiniteQueryProps { + size?: number; + enabled?: boolean; +} + +export function useAlbumClosedInfiniteQuery({ + size = 20, + enabled = true, +}: UseAlbumClosedInfiniteQueryProps = {}) { + const query = useInfiniteQuery({ + queryKey: [EP.album.albumClosed(), size], + initialPageParam: 0, + enabled: enabled, + queryFn: ({ pageParam }) => fetchData({ pageParam, size }), + getNextPageParam: (lastPage) => + lastPage.hasNext ? lastPage.page + 1 : undefined, + }); + + const items: AlbumClosedItem[] = + query.data?.pages.flatMap((p) => p.responses ?? []) ?? []; + + return { + ...query, + items, + }; +} diff --git a/src/feature/main/hooks/useAlbumOpenInfiniteQuery.ts b/src/feature/main/hooks/useAlbumOpenInfiniteQuery.ts new file mode 100644 index 00000000..99e14b17 --- /dev/null +++ b/src/feature/main/hooks/useAlbumOpenInfiniteQuery.ts @@ -0,0 +1,74 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useInfiniteQuery } from '@tanstack/react-query'; + +type AlbumOpenType = 'all' | 'mine'; +type AlbumOpenPage = NonNullable; +type AlbumOpenItem = AlbumOpenPage['responses'][number]; + +interface FetchPageParams { + pageParam: number; + /** 페이지 사이즈 (기본 20) */ + size: number; + path: string; +} + +const fetchData = async ({ + pageParam, + size, + path, +}: FetchPageParams): Promise => { + const res = await api.get({ + path, + params: { page: pageParam, size }, + }); + + const result: AlbumOpenPage = res.result ?? createEmptyPage({ pageParam }); + + return { ...result, page: pageParam }; +}; + +const createEmptyPage = ({ + pageParam, +}: { + pageParam: number; +}): AlbumOpenPage => ({ + responses: [], + listSize: 0, + isFirst: pageParam === 0, + isLast: true, + hasNext: false, +}); + +interface UseAlbumOpenInfiniteQueryProps { + size?: number; + enabled?: boolean; + type?: AlbumOpenType; +} + +export function useAlbumOpenInfiniteQuery({ + size = 10, + enabled = true, + type = 'all', +}: UseAlbumOpenInfiniteQueryProps = {}) { + const path = type === 'mine' ? EP.album.albumOpenMe() : EP.album.albumOpen(); + const query = useInfiniteQuery({ + queryKey: [path, size], + initialPageParam: 0, + enabled: enabled, + queryFn: ({ pageParam }) => fetchData({ pageParam, size, path }), + getNextPageParam: (lastPage) => + lastPage.hasNext ? lastPage.page + 1 : undefined, + }); + + const items: AlbumOpenItem[] = + query.data?.pages.flatMap((p) => p.responses ?? []) ?? []; + + return { + type, + ...query, + items, + }; +} + +export type { AlbumOpenItem, AlbumOpenType }; diff --git a/src/feature/main/hooks/useGetUserMe.ts b/src/feature/main/hooks/useGetUserMe.ts new file mode 100644 index 00000000..84e1d0ef --- /dev/null +++ b/src/feature/main/hooks/useGetUserMe.ts @@ -0,0 +1,26 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery, UseQueryOptions } from '@tanstack/react-query'; + +const fetchData = async () => { + const response = await api.get({ + path: EP.user.userMe(), + }); + + return response.result; +}; + +export function useGetUserMe( + options?: Omit< + UseQueryOptions, + 'queryKey' | 'queryFn' + >, +) { + const query = useQuery({ + queryKey: [EP.user.userMe()], + queryFn: () => fetchData(), + ...options, + }); + + return query; +} diff --git a/src/feature/main/utils/getOpenAlbumGridConfig.ts b/src/feature/main/utils/getOpenAlbumGridConfig.ts new file mode 100644 index 00000000..1178c4b7 --- /dev/null +++ b/src/feature/main/utils/getOpenAlbumGridConfig.ts @@ -0,0 +1,31 @@ +interface GetOpenAlbumGridConfig { + columns: string; + rows: string; + areas: string; + side1Row?: string; +} + +export function getOpenAlbumGridConfig(count: number): GetOpenAlbumGridConfig { + if (count <= 1) { + return { + columns: '1fr', + rows: '1fr', + areas: '"main"', + }; + } + + if (count === 2) { + return { + columns: '2fr 1fr', + rows: '1fr', + areas: '"main side1"', + side1Row: '1 / span 2', + }; + } + + return { + columns: '2fr 1fr', + rows: '1fr 1fr', + areas: '"main side1"\n "main side2"', + }; +} diff --git a/src/feature/main/utils/mapClosedAlbumItems.ts b/src/feature/main/utils/mapClosedAlbumItems.ts new file mode 100644 index 00000000..d4d1bffa --- /dev/null +++ b/src/feature/main/utils/mapClosedAlbumItems.ts @@ -0,0 +1,27 @@ +import { formatEventDate } from '@/global/utils/formatEventDate'; +import { AlbumClosedItem } from '../hooks/useAlbumClosedInfiniteQuery'; + +const MAX_IMAGES = 4; + +interface CloseAlbumListItem { + code: string; + title: string; + date: string; + author: string; + images: string[]; +} + +export function mapClosedAlbumItems( + items: AlbumClosedItem[], +): CloseAlbumListItem[] { + return items.map((item) => ({ + code: item.code, + title: item.title ?? '', + date: formatEventDate(item.eventDate), + author: item.makerName ?? '', + images: + item.thumbnails + ?.filter((thumbnail): thumbnail is string => Boolean(thumbnail)) + ?.slice(0, MAX_IMAGES) ?? [], + })); +} diff --git a/src/feature/main/utils/mapOpenAlbumItems.ts b/src/feature/main/utils/mapOpenAlbumItems.ts new file mode 100644 index 00000000..28d0cd14 --- /dev/null +++ b/src/feature/main/utils/mapOpenAlbumItems.ts @@ -0,0 +1,32 @@ +import { formatEventDate } from '@/global/utils/formatEventDate'; +import { formatExpirationTime } from '@/global/utils/time/formatExpirationTime'; +import { AlbumOpenItem } from '../hooks/useAlbumOpenInfiniteQuery'; + +interface OpenAlbumListItem { + code: string; + author: string; + date: string; + expirationTime: string; + joinedMembers: number; + totalMembers: number; + title: string; + thumbnails: string[]; + emoji: string; +} + +export function mapOpenAlbumItems(items: AlbumOpenItem[]): OpenAlbumListItem[] { + return items.map((item) => ({ + code: item.code, + author: item.makerName ?? '', + date: formatEventDate(item.eventDate), + expirationTime: formatExpirationTime(item.expiredAt), + joinedMembers: item.currentParticipant ?? 0, + totalMembers: item.participant ?? 0, + title: item.title ?? '', + thumbnails: + item.recentPhotoThumbnails + ?.slice(0, 3) + .filter((thumbnail): thumbnail is string => Boolean(thumbnail)) ?? [], + emoji: item.themeEmoji, + })); +} diff --git a/src/feature/mypage/components/ButtonDeleteAccount.tsx b/src/feature/mypage/components/ButtonDeleteAccount.tsx new file mode 100644 index 00000000..65ad66de --- /dev/null +++ b/src/feature/mypage/components/ButtonDeleteAccount.tsx @@ -0,0 +1,32 @@ +import ConfirmModal from '@/global/components/modal/ConfirmModal'; + +interface ButtonDeleteAccountProps { + onConfirm?: () => Promise | void; +} + +export default function ButtonDeleteAccount({ + onConfirm, +}: ButtonDeleteAccountProps) { + const handleConfirm = async () => { + try { + if (onConfirm) await onConfirm(); + } catch (err) { + console.error(err); + } + }; + + return ( + + 탈퇴하기 + + } + title='정말 탈퇴하시겠어요?' + description='계정은 삭제되며, 복구되지 않아요.' + cancelText='다음에' + confirmText='탈퇴하기' + onConfirm={handleConfirm} + /> + ); +} diff --git a/src/feature/mypage/components/ButtonLogout.tsx b/src/feature/mypage/components/ButtonLogout.tsx new file mode 100644 index 00000000..75f0f27a --- /dev/null +++ b/src/feature/mypage/components/ButtonLogout.tsx @@ -0,0 +1,33 @@ +import ConfirmModal from '@/global/components/modal/ConfirmModal'; +import { useRouter } from 'next/navigation'; +import { useLogoutMutation } from '../hooks/useLogoutMutation'; + +interface ButtonLogoutProps {} + +export default function ButtonLogout({}: ButtonLogoutProps) { + const router = useRouter(); + const { mutateAsync } = useLogoutMutation(); + + const handleConfirm = async () => { + try { + await mutateAsync(); + router.push('/'); + } catch (err) { + console.error(err); + } + }; + + return ( + + 로그아웃 + + } + title='로그아웃 하시겠어요?' + cancelText='다음에' + confirmText='로그아웃' + onConfirm={handleConfirm} + /> + ); +} diff --git a/src/feature/mypage/components/ModalLoginExpired.tsx b/src/feature/mypage/components/ModalLoginExpired.tsx new file mode 100644 index 00000000..e69de29b diff --git a/src/feature/mypage/components/ProfileSetting.tsx b/src/feature/mypage/components/ProfileSetting.tsx new file mode 100644 index 00000000..21e1c94b --- /dev/null +++ b/src/feature/mypage/components/ProfileSetting.tsx @@ -0,0 +1,36 @@ +'use client'; +import { useGetUserMe } from '@/feature/main/hooks/useGetUserMe'; +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; + +interface ProfileSettingProps {} + +export default function ProfileSetting({}: ProfileSettingProps) { + const { data } = useGetUserMe(); + + return ( +
+
+
+
+ 프로필 이미지 +
+ +
+

+ {data?.name} +

+

+ {data?.email} +

+
+
+
+
+ ); +} diff --git a/src/feature/mypage/components/ScreenMypage.tsx b/src/feature/mypage/components/ScreenMypage.tsx new file mode 100644 index 00000000..d63bb529 --- /dev/null +++ b/src/feature/mypage/components/ScreenMypage.tsx @@ -0,0 +1,15 @@ +import CustomHeader from '@/global/components/header/CustomHeader'; +import ProfileSetting from './ProfileSetting'; +import SettingButtons from './SettingButtons'; + +interface ScreenMypageProps {} + +export default function ScreenMypage({}: ScreenMypageProps) { + return ( + <> + + + + + ); +} diff --git a/src/feature/mypage/components/SettingButtons.tsx b/src/feature/mypage/components/SettingButtons.tsx new file mode 100644 index 00000000..be92f591 --- /dev/null +++ b/src/feature/mypage/components/SettingButtons.tsx @@ -0,0 +1,22 @@ +'use client'; +import Link from 'next/link'; +import ButtonLogout from './ButtonLogout'; + +interface SettingButtonsProps {} + +export default function SettingButtons({}: SettingButtonsProps) { + return ( +
+ + 서비스 이용약관 + + + 개인정보 처리방침 + + + + {/* TODO : 기능 만들기 전까지 주석처리 */} + {/* */} +
+ ); +} diff --git a/src/feature/mypage/hooks/useLogoutMutation.ts b/src/feature/mypage/hooks/useLogoutMutation.ts new file mode 100644 index 00000000..abc98dc7 --- /dev/null +++ b/src/feature/mypage/hooks/useLogoutMutation.ts @@ -0,0 +1,14 @@ +import { EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +const fetchData = async () => { + const res = await api.post({ path: EP.auth.logout() }); + return res.result; +}; + +export function useLogoutMutation() { + const mutation = useMutation({ mutationFn: () => fetchData() }); + + return mutation; +} diff --git a/src/feature/onboarding/components/ProfileAgree.tsx b/src/feature/onboarding/components/ProfileAgree.tsx new file mode 100644 index 00000000..ef793e1b --- /dev/null +++ b/src/feature/onboarding/components/ProfileAgree.tsx @@ -0,0 +1,107 @@ +'use client'; +import { Check, ChevronRight } from 'lucide-react'; +import Link from 'next/link'; + +interface AgreementItem { + id: string; + label: string; + required: boolean; +} + +const agreementItems: AgreementItem[] = [ + { id: 'terms', label: '[필수] 이용약관 동의', required: true }, + { id: 'privacy', label: '[필수] 개인정보 수집 및 이용 동의', required: true }, + { + id: 'thirdParty', + label: '[필수] 개인정보의 제3자 제공 동의', + required: true, + }, + { id: 'marketing', label: '[선택] 마케팅 정보 수신 동의', required: false }, +]; + +interface ProfileAgreeProps { + agreements: Record; + onAgreementsChange: (agreements: Record) => void; +} + +export function ProfileAgree({ + agreements, + onAgreementsChange, +}: ProfileAgreeProps) { + const allAgreed = agreementItems.every((item) => agreements[item.id]); + + function handleAllAgree() { + const newValue = !allAgreed; + const newAgreements = { + terms: newValue, + privacy: newValue, + thirdParty: newValue, + marketing: newValue, + }; + onAgreementsChange(newAgreements); + } + + function handleIndividualAgree(id: string) { + const newAgreements = { + ...agreements, + [id]: !agreements[id], + }; + onAgreementsChange(newAgreements); + } + + return ( +
+
+ +
+ 전체 동의하기 +
+
+ +
+ {agreementItems.map((item) => ( +
+
+ + + {item.label} + +
+ + + +
+ ))} +
+
+ ); +} diff --git a/src/feature/onboarding/components/ProfileImage.tsx b/src/feature/onboarding/components/ProfileImage.tsx new file mode 100644 index 00000000..d09ac44d --- /dev/null +++ b/src/feature/onboarding/components/ProfileImage.tsx @@ -0,0 +1,141 @@ +'use client'; +import { DrawerClose } from '@/components/ui/drawer'; +const BottomSheetModal = dynamic( + () => import('@/global/components/modal/BottomSheetModal'), + { ssr: false }, +); +import { Pencil } from 'lucide-react'; +import Image from 'next/image'; +import { memo, useState } from 'react'; +import { useGetAllProfiles } from '../hooks/useGetAllProfile'; +import dynamic from 'next/dynamic'; + +interface ProfileImageProps { + selectedImage: string | null; + onImageSelect: (image: string) => void; +} + +function ProfileImage({ selectedImage, onImageSelect }: ProfileImageProps) { + const [shouldFetchProfiles, setShouldFetchProfiles] = useState(false); + + const { data, isLoading, isError } = useGetAllProfiles(shouldFetchProfiles); + + const PROFILE_IMAGES: Record = { + P1: 'https://say-cheese-profile.edge.naverncp.com/profile/sign_up_profile_1.jpg', + }; + + const imageList = + data?.opts?.filter((img) => img.imageCode && img.profileImageUrl) ?? []; + + const getCurrentImageUrl = (): string => { + if (imageList.length > 0 && selectedImage) { + const foundImage = imageList.find( + (img) => img.imageCode === selectedImage, + ); + if (foundImage?.profileImageUrl) { + return foundImage.profileImageUrl; + } + } + + if (selectedImage && PROFILE_IMAGES[selectedImage]) { + return PROFILE_IMAGES[selectedImage]; + } + + return PROFILE_IMAGES['P1']; + }; + + const currentImage = getCurrentImageUrl(); + + return ( +
+
+ + 프로필 이미지 +
+ +
+ + } + showCloseButton={false} + className='h-90 px-5' + showHandle={true} + dismissible={true} + onOpenChange={(isOpen) => { + // 모달이 열리면 API fetch 활성화 (한번만) + if (isOpen && !shouldFetchProfiles) { + setShouldFetchProfiles(true); + } + }} + > + {isLoading ? null : isError ? ( +
+ 이미지 목록을 불러오지 못했습니다. +
+ ) : ( +
+ {imageList.map((img, index) => { + const url = img.profileImageUrl; + const key = img.imageCode || url; + return ( + + + + ); + })} +
+ )} +
+
+ + {/* CSS 애니메이션 정의 */} + +
+ ); +} + +export default memo(ProfileImage); diff --git a/src/feature/onboarding/components/ScreenOnBoarding.tsx b/src/feature/onboarding/components/ScreenOnBoarding.tsx new file mode 100644 index 00000000..56256719 --- /dev/null +++ b/src/feature/onboarding/components/ScreenOnBoarding.tsx @@ -0,0 +1,109 @@ +'use client'; +import { ProfileAgree } from '@/feature/onboarding/components/ProfileAgree'; +import ProfileImage from '@/feature/onboarding/components/ProfileImage'; +import { useOnBoardingMutation } from '@/feature/onboarding/hooks/useOnBoardingMutation'; +import LogoHeader from '@/global/components/header/LogoHeader'; +import LongButton from '@/global/components/LongButton'; +import Toast from '@/global/components/toast/Toast'; +import XInput from '@/global/components/XInput'; +import { useRouter, useSearchParams } from 'next/navigation'; +import { useState } from 'react'; + +export default function ScreenOnBoarding() { + const searchParams = useSearchParams(); + const router = useRouter(); + + const [selectedImage, setSelectedImage] = useState('P1'); + + const nameFromQuery = searchParams.get('name') || ''; + const [nickname, setNickname] = useState( + decodeURIComponent(nameFromQuery), + ); + + const [nicknameError, setNicknameError] = useState(''); + + const handleNicknameChange = (value: string) => { + // 한글(완성형+자음+모음), 영문, 숫자, 공백만 허용하는 정규식 + const validPattern = /^[가-힣ㄱ-ㅎㅏ-ㅣa-zA-Z0-9 ]*$/; + + if (!validPattern.test(value)) { + setNicknameError('10글자 이내의 한글, 영문, 숫자만 쓸 수 있어요'); + } else { + setNicknameError(''); + } + + setNickname(value); + }; + + const [agreements, setAgreements] = useState>({ + terms: false, + privacy: false, + thirdParty: false, + marketing: false, + }); + + const requiredAgreements = ['terms', 'privacy', 'thirdParty']; + const isRequiredAgreed = requiredAgreements.every((key) => agreements[key]); + + const isFormComplete = + selectedImage && + nickname.trim() !== '' && + nicknameError === '' && + isRequiredAgreed; + + const { mutate, status } = useOnBoardingMutation(); + const isLoading = status === 'pending'; + + const handleSubmit = () => { + if (!isFormComplete || isLoading) return; + mutate( + { + name: nickname, + imageCode: selectedImage, + isServiceAgreement: agreements.terms, + isUserInfoAgreement: agreements.privacy, + isMarketingAgreement: agreements.marketing, + isThirdPartyAgreement: agreements.thirdParty, + }, + { + onSuccess: () => { + router.push('/onboarding/complete'); + }, + onError: (error) => { + Toast.alert('가입에 실패했어요. 잠시 후 다시 시도해주세요.'); + console.error('온보딩 실패:', error); + }, + }, + ); + }; + + return ( +
+ + + + + + +
+ ); +} diff --git a/src/feature/onboarding/components/ScreenOnBoardingComplete.tsx b/src/feature/onboarding/components/ScreenOnBoardingComplete.tsx new file mode 100644 index 00000000..ac05e08d --- /dev/null +++ b/src/feature/onboarding/components/ScreenOnBoardingComplete.tsx @@ -0,0 +1,44 @@ +'use client'; +import LongButton from '@/global/components/LongButton'; +import { clearEntryCookie, getEntryCookie } from '@/global/utils/cookies'; +import Image from 'next/image'; +import { useRouter } from 'next/navigation'; +import { useCallback } from 'react'; + +export default function ScreenOnboardingComplete() { + const router = useRouter(); + + const handleStartClick = useCallback(() => { + const entry = getEntryCookie(); + + if (entry === 'create-album') { + router.push('/create-album'); + } else { + router.push('/main'); + } + + clearEntryCookie(); + }, [router]); + return ( +
+ 환영합니다 + + 치이이즈에 +
+ 오신 걸 환영해요 +
+ +
+ ); +} diff --git a/src/feature/onboarding/components/TermContent.tsx b/src/feature/onboarding/components/TermContent.tsx new file mode 100644 index 00000000..139a4d1a --- /dev/null +++ b/src/feature/onboarding/components/TermContent.tsx @@ -0,0 +1,214 @@ +import { FC } from 'react'; // React 19+에서 타입 추론이 강력해졌지만, FC로 명시해서 타입 안전성 강화 + +const TermsComponent: FC = () => ( +
+
+
+
+ 이용약관 동의 +
+
+ 본 약관은 치이이즈(이하 "회사")가 제공하는 웹 기반 사진 공유 + 서비스(이하 "서비스")의 이용과 관련하여 회사와 이용자 간의 + 권리, 의무 및 책임 사항, 기타 필요한 사항을 규정함을 목적으로 합니다. +
+
+
+
+ 제1조 (서비스의 내용) +
+
+ 이 서비스는 사용자가 QR코드 또는 URL을 통해 웹페이지에 접속하여 사진을 + 업로드하고, 다른 사용자와 함께 공유하며, 7일이 경과한 후 자동으로 + 삭제되는 단기형 공유 앨범을 제공합니다. 본 서비스는 별도의 + 애플리케이션 설치 없이 웹 환경에서 이용할 수 있습니다. +
+
+
+
+ 제2조 (이용자의 의무) +
+
+ 이용자는 서비스를 이용함에 있어 다음 각 호의 행위를 하여서는 아니 + 됩니다. +

+   ·  타인의 개인정보를 무단으로 수집하거나 + 부정하게 사용하는 행위 +

+

+   ·  불법적이거나 사회질서에 반하는 내용의 사진 + 또는 글을 게시하는 행위 +

+

+   ·  회사 및 제3자의 저작권, 초상권 등 타인의 + 권리를 침해하는 행위 +

+

+   ·  본 서비스를 영리 목적 또는 부정한 목적으로 + 이용하는 행위 +

+
+ 이용자는 본 서비스의 취지에 맞게 타인에게 불쾌감을 주거나 불법적인 + 콘텐츠를 업로드하지 않아야 하며, 위반 시 이용이 제한될 수 있습니다. +
+
+
+
+ 제3조 (서비스의 변경 및 중단) +
+
+ 회사는 서비스의 개선을 위하여 제공 중인 서비스의 전부 또는 일부를 + 변경하거나 중단할 수 있습니다. +
+ 회사는 서비스 중단으로 발생하는 데이터 손실에 대하여 별도의 보상을 + 하지 않으며, 이용자는 7일의 유효기간 내에 필요한 데이터를 직접 + 저장해야 합니다. +
+
+
+
+ 제4조 (면책) +
+
+ 회사는 천재지변, 시스템 장애, 통신 두절 등 불가항력적인 사유로 인한 + 서비스 중단 및 데이터 손실에 대하여 책임을 지지 않습니다. 이용자가 + 업로드한 모든 콘텐츠의 저작권 및 책임은 전적으로 해당 이용자에게 + 있으며, 회사는 그 내용에 대해 일체의 책임을 지지 않습니다. +
+
+
+
+); + +const PrivacyComponent: FC = () => ( +
+
+
+
+ 개인정보 수집 및 이용 동의 +
+
+ 회사는 서비스 제공을 위하여 다음과 같은 개인정보를 수집하고 + 이용합니다. +

① 수집 항목

+

+   ·  필수항목: 닉네임, 로그인 계정(카카오 등 + 간편 로그인 정보), +
+      업로드된 사진 및 메타데이터, 접속 + 로그, 쿠키 등 +

+

+   ·  선택항목: 이메일 주소(이벤트 및 알림 + 수신용), 프로필 이미지 +

+
+

② 수집 및 이용 목적

+

1. 서비스 이용자 식별 및 참여 관리

+

2. 사진 업로드, 다운로드 등 주요 기능 제공

+

3. 불법·비정상 이용행위 방지 및 보안 관리

+

4. 서비스 개선, 이용 통계 및 분석

+
+

③ 보유 및 이용 기간

+

+   ·  업로드된 사진 및 관련 데이터는 +
+      앨범 생성 시점으로부터 7일간 보관 후 + 자동 삭제됩니다. +

+

+   ·  로그인 계정 정보 등 식별정보는
+      회원 탈퇴 시 또는 이용 목적이 달성된 + 후 즉시 파기됩니다. +

+
④ 이용자는 개인정보의 열람, 정정, 삭제를 요청할 수 있으며, + 회사는 관련 법령에 따라 이에 성실히 응합니다. +
+
+
+
+); + +const ThirdPartyComponent: FC = () => ( +
+
+
+
+ 개인정보의 제3자 제공 동의 +
+
+ 회사는 서비스의 안정적 운영과 기능 제공을 위하여, 이용자의 개인정보를 + 아래와 같이 제3자에게 제공합니다. 회사는 정보통신망 이용촉진 및 + 정보보호 등에 관한 법률 및 개인정보보호법 등 관련 법령을 준수하며, + 이용자의 개인정보를 목적 외로 제공하지 않습니다. +
+
+ 회사는 이용자가 업로드한 사진과 관련 데이터를 안전하게 저장하고 + 관리하기 위하여, 클라우드 서비스 제공업체인 네이버클라우드플랫폼(Naver + Cloud Platform)에 일부 정보를 제공합니다. 네이버클라우드플랫폼은 서버 + 저장, 백업, 데이터 관리 등의 목적으로만 해당 정보를 이용하며, 회사의 + 지침에 따라 엄격하게 관리합니다. 제공되는 항목은 업로드된 사진, 앨범 + 식별 코드, 업로드 시각 등 서비스 운영에 필요한 최소한의 데이터이며, + 해당 정보는 앨범 생성 시점으로부터 7일간 보관된 후 자동으로 + 삭제됩니다. +
+
+ 또한, 간편 로그인 기능 제공을 위해 카카오(주)에 로그인 계정 + 정보(이메일, 닉네임 등)가 제공됩니다. 제공된 정보는 간편 로그인 및 + 사용자 인증을 위한 용도로만 이용되며, 이용자가 탈퇴하거나 계정 연동을 + 해제할 경우 즉시 삭제됩니다. +
+
+ 서비스 품질 개선과 이용 통계 분석을 위해 Google Analytics에 접속 로그 + 및 이용 패턴 등 익명화된 정보가 제공될 수 있습니다. 이러한 정보는 + 개인을 식별할 수 없는 형태로 수집되며, 통계적 분석 목적으로만 사용된 + 뒤 익명화 후 6개월간 보관됩니다. +
+
+ 회사는 위의 목적 외에는 이용자의 개인정보를 제3자에게 제공하지 않으며, + 새로운 제휴 또는 제공이 필요한 경우 이용자에게 사전 고지 후 별도의 + 동의를 받습니다. +
+
+
+
+); + +const MarketingComponent: FC = () => ( +
+
+
+
+ 마케팅 정보 수신 동의 +
+
+ 회사는 이벤트, 신규 서비스, 프로모션 등 유용한 정보를 이메일, + 문자메시지, 카카오 알림톡 등의 방법으로 발송할 수 있습니다. +
이용자는 마케팅 정보 수신에 동의하지 않아도 기본 서비스 이용에 + 제한이 없습니다. +
마케팅 정보의 수신 동의는 언제든 설정 메뉴 또는 수신 거부 + 링크를 통해 철회할 수 있습니다. +
+
+
+
+); + +export const TermContent = { + terms: { + title: '이용약관 동의', + content: TermsComponent, + }, + privacy: { + title: '개인정보 수집 및 이용 동의', + content: PrivacyComponent, + }, + thirdParty: { + title: '개인정보의 제3자 제공 동의', + content: ThirdPartyComponent, + }, + marketing: { + title: '마케팅 정보 수신 동의', + content: MarketingComponent, + }, +}; diff --git a/src/feature/onboarding/hooks/useGetAllProfile.ts b/src/feature/onboarding/hooks/useGetAllProfile.ts new file mode 100644 index 00000000..ff857470 --- /dev/null +++ b/src/feature/onboarding/hooks/useGetAllProfile.ts @@ -0,0 +1,18 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery } from '@tanstack/react-query'; +const getAllProfile = async () => { + const data = await api.get({ + path: EP.user.userProfileImages(), + }); + return data.result; +}; + +export function useGetAllProfiles(enabled: boolean = true) { + return useQuery({ + queryKey: [EP.user.userProfileImages()], + queryFn: getAllProfile, + staleTime: 36000, + enabled, + }); +} diff --git a/src/feature/onboarding/hooks/useOnBoardingMutation.test.tsx b/src/feature/onboarding/hooks/useOnBoardingMutation.test.tsx new file mode 100644 index 00000000..8ce1a8a3 --- /dev/null +++ b/src/feature/onboarding/hooks/useOnBoardingMutation.test.tsx @@ -0,0 +1,67 @@ +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; +import { beforeEach, describe, expect, it, vi } from 'vitest'; +import { useOnBoardingMutation } from './useOnBoardingMutation'; + +// Mock api +vi.mock('@/global/utils/api', () => ({ + api: { + post: vi.fn(), + }, +})); + +// Mock useMutation +vi.mock('@tanstack/react-query', () => ({ + useMutation: vi.fn(), +})); + +describe('useOnBoardingMutation', () => { + beforeEach(() => { + vi.clearAllMocks(); + }); + + it('should call useMutation with correct mutationFn', () => { + useOnBoardingMutation(); + + expect(useMutation).toHaveBeenCalledWith( + expect.objectContaining({ + mutationFn: expect.any(Function), + }), + ); + }); + + it('should call api.post when mutationFn is executed', async () => { + // Capture the mutationFn passed to useMutation + // eslint-disable-next-line @typescript-eslint/no-explicit-any + let capturedMutationFn: any; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (useMutation as any).mockImplementation(({ mutationFn }: any) => { + capturedMutationFn = mutationFn; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + return {} as any; + }); + + useOnBoardingMutation(); + + const mockPayload = { + name: 'Test User', + imageCode: 'img-123', + isServiceAgreement: true, + isUserInfoAgreement: true, + isMarketingAgreement: false, + isThirdPartyAgreement: true, + }; + + const mockResponse = { result: { success: true } }; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + (api.post as any).mockResolvedValue(mockResponse); + + // Execute the captured mutationFn + await capturedMutationFn(mockPayload); + + expect(api.post).toHaveBeenCalledWith({ + path: expect.stringContaining('/user/onboarding'), + body: mockPayload, + }); + }); +}); diff --git a/src/feature/onboarding/hooks/useOnBoardingMutation.ts b/src/feature/onboarding/hooks/useOnBoardingMutation.ts new file mode 100644 index 00000000..14af9ae9 --- /dev/null +++ b/src/feature/onboarding/hooks/useOnBoardingMutation.ts @@ -0,0 +1,26 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +type OnboardingPayload = { + name: string; + imageCode: string; + isServiceAgreement: boolean; + isUserInfoAgreement: boolean; + isMarketingAgreement: boolean; + isThirdPartyAgreement: boolean; +}; + +async function postUserOnboarding(payload: OnboardingPayload) { + const res = await api.post({ + path: EP.user.userOnboarding(), + body: payload, + }); + return res.result; +} + +export function useOnBoardingMutation() { + return useMutation({ + mutationFn: (payload: OnboardingPayload) => postUserOnboarding(payload), + }); +} diff --git a/src/feature/photo-detail/components/FooterPhotoDetail.tsx b/src/feature/photo-detail/components/FooterPhotoDetail.tsx new file mode 100644 index 00000000..b838e6ba --- /dev/null +++ b/src/feature/photo-detail/components/FooterPhotoDetail.tsx @@ -0,0 +1,170 @@ +'use client'; +import { EP } from '@/global/api/ep'; +import BottomSheetModal from '@/global/components/modal/BottomSheetModal'; +import Toast from '@/global/components/toast/Toast'; +import { downloadFile } from '@/global/utils/downloadFile'; +import { getDeviceType } from '@/global/utils/getDeviceType'; +import { shareImage } from '@/global/utils/image/shareImage'; +import { useQueryClient } from '@tanstack/react-query'; +import { Download, Heart, Info } from 'lucide-react'; +import { useState } from 'react'; +import { usePhotoDownloadMutation } from '../hooks/usePhotoDownloadMutation'; +import { usePhotoLikedMutation } from '../hooks/usePhotoLikedMutation'; +import { usePhotoUnlikedMutation } from '../hooks/usePhotoUnlikedMutation'; +import { updateCacheAlbumPhotosLike } from '../modules/updateCacheAlbumPhotosLike'; +import ListPhotoLikers from './ListPhotoLikers'; +import SectionPhotoData from './SectionPhotoData'; + +interface FooterPhotoDetailProps { + albumId: string; + photoId: number; + isLiked: boolean; + likeCnt: number; + isRecentlyDownloaded: boolean; + imageUrl: string | undefined; +} + +export default function FooterPhotoDetail({ + albumId, + photoId, + isLiked, + likeCnt, + isRecentlyDownloaded, + imageUrl, +}: FooterPhotoDetailProps) { + const queryClient = useQueryClient(); + const [isDownloading, setIsDownloading] = useState(false); + const [isPhotoInfoOpen, setIsPhotoInfoOpen] = useState(false); + const { mutateAsync: mutateAsyncLike, isPending: isLiking } = + usePhotoLikedMutation(); + const { mutateAsync: mutateAsyncUnlike, isPending: isUnliking } = + usePhotoUnlikedMutation(); + const { mutateAsync: mutateAsyncDownload } = usePhotoDownloadMutation(); + + const handleDeepToggle = async (): Promise => { + try { + if (isLiked) { + if (!isUnliking) await mutateAsyncUnlike(photoId); + } else { + if (!isLiking) await mutateAsyncLike(photoId); + } + + updateCacheAlbumPhotosLike({ + albumId, + isCurrentlyLiked: isLiked, + photoId, + queryClient, + }); + queryClient.invalidateQueries({ + queryKey: [EP.album.albumPhotosLikers(albumId, photoId)], + }); + queryClient.invalidateQueries({ + queryKey: [EP.album.likedPhotos(albumId)], + }); + } catch (e) { + console.error(e); + Toast.alert(`좋에요에 실패하였습니다.`); + } + }; + + const handleDownload = async (): Promise => { + if (!imageUrl) return; + if (isRecentlyDownloaded) { + Toast.alert(`금방 다운받은 사진이에요.\n1시간 뒤에 다시 시도하세요.`); + return; + } + if (isDownloading) return; + + try { + setIsDownloading(true); + const deviceType = getDeviceType(); + const fileName = `IMG_${photoId}`; + + if (deviceType === 'ios') { + shareImage({ + imageUrls: imageUrl, + imageTitle: fileName, + onSuccess: () => { + mutateAsyncDownload({ albumId, photoIds: [photoId] }); + }, + onError: () => { + downloadFile(imageUrl, fileName); + }, + }); + } else { + await downloadFile(imageUrl, fileName); + mutateAsyncDownload({ albumId, photoIds: [photoId] }); + } + } catch (e) { + console.log(e); + Toast.alert('사진을 준비하는 중 오류가 발생했습니다.'); + } finally { + setIsDownloading(false); + } + }; + + return ( +
+ + + + } + > + setIsPhotoInfoOpen(false)} + /> + + + + +
+ + + + + {likeCnt} + + + } + > + + +
+
+ ); +} diff --git a/src/feature/photo-detail/components/HeaderPhotoDetail.tsx b/src/feature/photo-detail/components/HeaderPhotoDetail.tsx new file mode 100644 index 00000000..99a0639b --- /dev/null +++ b/src/feature/photo-detail/components/HeaderPhotoDetail.tsx @@ -0,0 +1,40 @@ +'use client'; +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; +import { X } from 'lucide-react'; +import { useRouter } from 'next/navigation'; + +interface HeaderPhotoDetailProps { + profileImageUrl?: string; + name?: string; +} + +export default function HeaderPhotoDetail({ + profileImageUrl, + name, +}: HeaderPhotoDetailProps) { + const router = useRouter(); + + const handleClose = (): void => { + router.back(); + }; + + return ( +
+
+ 프로필 사진 +
+ + {name} + + +
+ ); +} diff --git a/src/feature/photo-detail/components/ItemMemberData.tsx b/src/feature/photo-detail/components/ItemMemberData.tsx new file mode 100644 index 00000000..291dd52c --- /dev/null +++ b/src/feature/photo-detail/components/ItemMemberData.tsx @@ -0,0 +1,49 @@ +import { DEFAULT_PROFILE_IMAGE } from '@/global/constants/images'; + +interface ItemMemberDataProps { + profileImageUrl: string | undefined; + nickname: string; + isMe: boolean; + isMaker: boolean; +} + +export default function ItemMemberData({ + profileImageUrl, + nickname, + isMe, + isMaker, +}: ItemMemberDataProps) { + return ( +
+
+ {`${nickname}의 +
+ +
+ + {nickname} + + +
+ {isMe && ( + + 나 + + )} + + {isMaker && ( + + 메이커 + + )} +
+
+
+ ); +} diff --git a/src/feature/photo-detail/components/ListPhotoLikers.tsx b/src/feature/photo-detail/components/ListPhotoLikers.tsx new file mode 100644 index 00000000..9d062ae2 --- /dev/null +++ b/src/feature/photo-detail/components/ListPhotoLikers.tsx @@ -0,0 +1,35 @@ +import { usePhotoLikersQuery } from '../hooks/usePhotoLikersQuery'; +import ItemMemberData from './ItemMemberData'; + +interface ListPhotoLikersProps { + albumId: string; + photoId: number; +} + +export default function ListPhotoLikers({ + albumId, + photoId, +}: ListPhotoLikersProps) { + const { data, isPending, isError } = usePhotoLikersQuery({ + albumId, + photoId, + }); + + if (isPending) return null; + if (isError) return null; + if (!data || !data.photoLikers) return null; + + return ( +
+ {data.photoLikers.map(({ name, profileImageUrl, role, isMe }, index) => ( + + ))} +
+ ); +} diff --git a/src/feature/photo-detail/components/MainPhotoDetail.tsx b/src/feature/photo-detail/components/MainPhotoDetail.tsx new file mode 100644 index 00000000..fe2dc209 --- /dev/null +++ b/src/feature/photo-detail/components/MainPhotoDetail.tsx @@ -0,0 +1,68 @@ +import { PhotoListResponseSchema } from '@/global/api/ep'; +import Spinner from '@/global/components/Spinner'; +import dynamic from 'next/dynamic'; +import { useState } from 'react'; +import FooterPhotoDetail from './FooterPhotoDetail'; +import HeaderPhotoDetail from './HeaderPhotoDetail'; +const SwiperPhotoList = dynamic(() => import('./SwiperPhotoList'), { + ssr: false, + loading: () => ( +
+ +
+ ), +}); + +interface MainPhotoDetailProps { + images: PhotoListResponseSchema[]; + albumId: string; + photoId: number | null; +} + +export default function MainPhotoDetail({ + albumId, + images, + photoId, +}: MainPhotoDetailProps) { + const [activeIndex, setActiveIndex] = useState( + findImageIndexByPhotoId(images, photoId), + ); + + const changeActiveIndex = (newIndex: number): void => { + setActiveIndex(newIndex); + }; + + const activeImage = images[activeIndex]; + if (!activeImage) return null; + + return ( + <> + + + + + ); +} + +function findImageIndexByPhotoId( + images: PhotoListResponseSchema[], + targetPhotoId: number | null, +): number { + if (!targetPhotoId) return -1; + + return images.findIndex((image) => image.photoId === targetPhotoId); +} diff --git a/src/feature/photo-detail/components/ScreenPhotoDetail.tsx b/src/feature/photo-detail/components/ScreenPhotoDetail.tsx new file mode 100644 index 00000000..42392372 --- /dev/null +++ b/src/feature/photo-detail/components/ScreenPhotoDetail.tsx @@ -0,0 +1,35 @@ +'use client'; + +import { PhotoSorting } from '@/global/api/ep'; +import { useSearchParams } from 'next/navigation'; +import { useAlbumPhotosInfiniteQuery } from '../hooks/useAlbumPhotosInfiniteQuery'; +import MainPhotoDetail from './MainPhotoDetail'; + +interface ScreenPhotoDetailProps { + albumId: string; +} + +export default function ScreenPhotoDetail({ albumId }: ScreenPhotoDetailProps) { + const searchParams = useSearchParams(); + + const sort: PhotoSorting = + (searchParams.get('sort') as PhotoSorting) || 'CREATED_AT'; + const photoIdParam = searchParams.get('photoId'); + + const { items: images } = useAlbumPhotosInfiniteQuery({ + code: albumId, + size: 2000, + sorting: sort, + }); + + if (images.length === 0) return null; + + const photoId = + photoIdParam === null ? images[0].photoId : Number(photoIdParam); + + return ( +
+ +
+ ); +} diff --git a/src/feature/photo-detail/components/SectionPhotoData.tsx b/src/feature/photo-detail/components/SectionPhotoData.tsx new file mode 100644 index 00000000..c3b598ad --- /dev/null +++ b/src/feature/photo-detail/components/SectionPhotoData.tsx @@ -0,0 +1,115 @@ +import { EP } from '@/global/api/ep'; +import ConfirmModal from '@/global/components/modal/ConfirmModal'; +import { useQueryClient } from '@tanstack/react-query'; +import { useDeleteAlbumPhotoMutation } from '../hooks/useDeleteAlbumPhotoMutation'; +import { usePhotoDetailQuery } from '../hooks/usePhotoDetailQuery'; + +interface SectionPhotoDataProps { + albumId: string; + photoId: number; + onAfterDelete?: () => void; +} + +// 촬영 시각: 사진 EXIF 시간 그대로 표시 (타임존 변환 안 함) +const formatCaptureTime = (isoString?: string): string => { + if (!isoString) return ''; + + // ISO 문자열에서 직접 파싱 (타임존 변환 없이) + const match = isoString.match( + /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2})/, + ); + if (!match) return '정보 없음'; + + const [, year, month, day, hour, minute] = match; + + return `${year}년 ${month}월 ${day}일 ${hour}시 ${minute}분`; +}; + +// 업로드 시각: UTC를 로컬 시간(KST)으로 변환 +const formatKoreanDateTime = (isoString?: string): string => { + if (!isoString) return ''; + + // If the string doesn't end with Z and doesn't have a timezone offset, assume it's UTC + const targetDate = + !isoString.endsWith('Z') && !/[+-]\d{2}:\d{2}$/.test(isoString) + ? `${isoString}Z` + : isoString; + + const date = new Date(targetDate); + if (Number.isNaN(date.getTime())) return '정보 없음'; + + const year = date.getFullYear(); + const month = date.getMonth() + 1; + const day = date.getDate(); + const hour = date.getHours(); + const minute = date.getMinutes().toString().padStart(2, '0'); + + return `${year}년 ${month}월 ${day}일 ${hour}시 ${minute}분`; +}; + +export default function SectionPhotoData({ + albumId, + photoId, + onAfterDelete, +}: SectionPhotoDataProps) { + const queryClient = useQueryClient(); + const { data, isPending, isError } = usePhotoDetailQuery({ + albumId, + photoId, + }); + const { mutateAsync } = useDeleteAlbumPhotoMutation(); + + if (isPending) return null; + if (isError) return null; + + const handleDeleteClick = async () => { + try { + await mutateAsync({ albumId, photoId }); + queryClient.invalidateQueries({ queryKey: [EP.album.photos(albumId)] }); + } finally { + onAfterDelete?.(); + } + }; + + return ( +
+
+
+
업로드한 사람
+
{data?.name}
+
+
+
촬영 시각
+
+ {formatCaptureTime(data?.captureTime)} +
+
+
+
업로드 시각
+
+ {formatKoreanDateTime(data?.createdAt)} +
+
+
+ + {data?.canDelete && ( + + 사진 삭제하기 + + } + /> + )} +
+ ); +} diff --git a/src/feature/photo-detail/components/SwiperPhotoList.tsx b/src/feature/photo-detail/components/SwiperPhotoList.tsx new file mode 100644 index 00000000..6789c7e3 --- /dev/null +++ b/src/feature/photo-detail/components/SwiperPhotoList.tsx @@ -0,0 +1,166 @@ +'use client'; + +import { PhotoListResponseSchema } from '@/global/api/ep'; +import { useCallback, useEffect, useState } from 'react'; +import type { Swiper as SwiperType } from 'swiper'; +import 'swiper/css'; +import { Swiper, SwiperSlide } from 'swiper/react'; +import { calcThumbSwiperCenterOffset } from '../util/calcThumbSwiperCenterOffset'; + +interface SwiperPhotoListProps { + images?: PhotoListResponseSchema[]; + activeIndex: number; + changeActiveIndex: (newIndex: number) => void; +} + +export default function SwiperPhotoList({ + activeIndex, + changeActiveIndex, + images = [], +}: SwiperPhotoListProps) { + const [mainSwiper, setMainSwiper] = useState(null); + const [thumbSwiper, setThumbSwiper] = useState(null); + const [thumbOffset, setThumbOffset] = useState(0); + + const updateThumbOffset = useCallback(() => { + const vw = window.innerWidth; + setThumbOffset( + calcThumbSwiperCenterOffset({ + viewportWidth: vw, + activeMargin: 12, + activeWidth: 30, + inactiveWidth: 15, + inactiveMargin: 2, + index: activeIndex, + }), + ); + }, [activeIndex]); + + useEffect(() => { + updateThumbOffset(); + window.addEventListener('resize', updateThumbOffset); + return () => window.removeEventListener('resize', updateThumbOffset); + }, [updateThumbOffset]); + + useEffect(() => { + if (!thumbSwiper || thumbSwiper.destroyed) return; + + thumbSwiper.slideTo(activeIndex); + requestAnimationFrame(() => { + thumbSwiper.setTranslate(thumbOffset); + }); + }, [activeIndex, thumbOffset, thumbSwiper]); + + return ( + <> +
+ {/* 위: 메인 이미지 Swiper */} +
+ { + const idx = sw.activeIndex; + changeActiveIndex(idx); + if (thumbSwiper && !thumbSwiper.destroyed) { + thumbSwiper.slideTo(sw.activeIndex); + } + }} + onTap={(swiper, event) => { + if (!swiper || swiper.destroyed) return; + // 마우스/터치 겸용으로 clientX 추출 + const e = event as MouseEvent | TouchEvent; + let clientX: number | null = null; + + if ('clientX' in e) { + clientX = e.clientX; + } else if ('changedTouches' in e && e.changedTouches.length > 0) { + clientX = e.changedTouches[0].clientX; + } + + if (clientX == null) return; + + const { left, width } = swiper.el.getBoundingClientRect(); + const clickPosition = clientX - left; + + if (clickPosition < width / 2) { + swiper.slidePrev(); + } else { + swiper.slideNext(); + } + }} + > + {images.map(({ thumbnailUrl, photoId }, i) => { + const isActive = activeIndex === i; + return ( + +
+ {`photo-${i}`} +
+
+ ); + })} +
+
+ + {/* 아래: 썸네일 컨트롤러 */} +
+ + {images.map(({ thumbnailUrl, photoId }, i) => { + const isActive = activeIndex === i; + return ( + { + changeActiveIndex(i); + mainSwiper?.slideTo(i); + thumbSwiper?.slideTo(i); + }} + > + {/* TODO : 이미지 아직 불러오는 중일때 스켈레톤 띄우기 */} + {`thumb-${i}`} + + ); + })} + +
+
+ + ); +} diff --git a/src/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery.ts b/src/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery.ts new file mode 100644 index 00000000..1138fe6c --- /dev/null +++ b/src/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery.ts @@ -0,0 +1,62 @@ +import { ApiReturns, EP, PhotoSorting } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useInfiniteQuery } from '@tanstack/react-query'; + +interface FetchPageParams { + code: string; + pageParam: number; + /** 페이지 사이즈 (기본 20) */ + size: number; + /** API가 받는 정렬 형식: 예) 'CREATED_AT' */ + sorting: PhotoSorting; +} + +const fetchAlbumPhotosPage = async ({ + code, + pageParam, + size, + sorting, +}: FetchPageParams) => { + const res = await api.get({ + path: EP.album.photos(code), + params: { page: pageParam, size, sorting }, + }); + + if (!res.result) throw Error('result가 없습니다.'); + + return { ...res.result, page: pageParam }; +}; + +interface UseAlbumPhotosInfiniteQueryProps { + code: string; + size?: number; + sorting?: PhotoSorting; + enabled?: boolean; + refetchOnMount?: boolean | 'always'; +} + +export function useAlbumPhotosInfiniteQuery({ + code, + size = 20, + sorting = 'CREATED_AT', + enabled = true, + refetchOnMount = true, +}: UseAlbumPhotosInfiniteQueryProps) { + const query = useInfiniteQuery({ + queryKey: [EP.album.photos(code), size, sorting], + initialPageParam: 0, + enabled: enabled && !!code, + queryFn: ({ pageParam }) => + fetchAlbumPhotosPage({ code, pageParam, size, sorting }), + getNextPageParam: (lastPage) => + lastPage.hasNext ? lastPage.page + 1 : undefined, + refetchOnMount, + }); + + const items = query.data?.pages.flatMap((p) => p.responses ?? []) ?? []; + + return { + ...query, + items, + }; +} diff --git a/src/feature/photo-detail/hooks/useAlbumPhotosLikedInfiniteQuery.ts b/src/feature/photo-detail/hooks/useAlbumPhotosLikedInfiniteQuery.ts new file mode 100644 index 00000000..b1654f1e --- /dev/null +++ b/src/feature/photo-detail/hooks/useAlbumPhotosLikedInfiniteQuery.ts @@ -0,0 +1,75 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useInfiniteQuery } from '@tanstack/react-query'; + +type AlbumPhotosLikedPage = NonNullable; +type AlbumPhotosLikedItem = AlbumPhotosLikedPage['responses'][number]; + +interface FetchPageParams { + code: string; + pageParam: number; + /** 페이지 사이즈 (기본 20) */ + size: number; +} + +const fetchAlbumPhotosPage = async ({ + code, + pageParam, + size, +}: FetchPageParams): Promise => { + const res = await api.get({ + path: EP.album.likedPhotos(code), + params: { page: pageParam, size }, + }); + + const result: AlbumPhotosLikedPage = + res.result ?? createEmptyPage({ pageParam }); + + return { ...result, page: pageParam }; +}; + +const createEmptyPage = ({ + pageParam, +}: { + pageParam: number; +}): AlbumPhotosLikedPage => ({ + responses: [], + listSize: 0, + isFirst: pageParam === 0, + isLast: true, + hasNext: false, +}); + +interface UseAlbumPhotosLikedInfiniteQueryProps { + code: string; + size?: number; + enabled?: boolean; + refetchOnMount?: boolean | 'always'; +} + +export function useAlbumPhotosLikedInfiniteQuery({ + code, + size = 20, + enabled = true, + refetchOnMount, +}: UseAlbumPhotosLikedInfiniteQueryProps) { + const query = useInfiniteQuery({ + queryKey: [EP.album.likedPhotos(code), size], + initialPageParam: 0, + enabled: enabled && !!code, + queryFn: ({ pageParam }) => fetchAlbumPhotosPage({ code, pageParam, size }), + getNextPageParam: (lastPage) => + lastPage.hasNext ? lastPage.page + 1 : undefined, + refetchOnMount, + }); + + const items: AlbumPhotosLikedItem[] = + query.data?.pages.flatMap((p) => p.responses ?? []) ?? []; + + return { + ...query, + items, + }; +} + +export type { AlbumPhotosLikedItem }; diff --git a/src/feature/photo-detail/hooks/useDeleteAlbumPhotoMutation.ts b/src/feature/photo-detail/hooks/useDeleteAlbumPhotoMutation.ts new file mode 100644 index 00000000..d8db6977 --- /dev/null +++ b/src/feature/photo-detail/hooks/useDeleteAlbumPhotoMutation.ts @@ -0,0 +1,24 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +const fetchData = async (albumId: string, photoId: number) => { + const res = await api.delete({ + path: EP.album.albumPhoto(albumId, photoId), + }); + return res.result; +}; + +interface DeleteAlbumPhotoMutateProps { + albumId: string; + photoId: number; +} + +export function useDeleteAlbumPhotoMutation() { + const mutation = useMutation({ + mutationFn: ({ albumId, photoId }: DeleteAlbumPhotoMutateProps) => + fetchData(albumId, photoId), + }); + + return mutation; +} diff --git a/src/feature/photo-detail/hooks/usePhotoDetailQuery.ts b/src/feature/photo-detail/hooks/usePhotoDetailQuery.ts new file mode 100644 index 00000000..bfb9b822 --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoDetailQuery.ts @@ -0,0 +1,44 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { + keepPreviousData, + useQuery, + UseQueryOptions, +} from '@tanstack/react-query'; + +interface FetchDataProps { + albumId: string; + photoId: number; +} + +const fetchData = async ({ albumId, photoId }: FetchDataProps) => { + const response = await api.get({ + path: EP.album.photoDetail(albumId, photoId), + }); + + return response.result; +}; + +interface UsePhotoDetailQueryProps { + albumId: string; + photoId: number; + options?: Omit< + UseQueryOptions, + 'queryKey' | 'queryFn' + >; +} + +export function usePhotoDetailQuery({ + albumId, + photoId, + options, +}: UsePhotoDetailQueryProps) { + const query = useQuery({ + queryKey: [EP.album.photoDetail(albumId, photoId)], + queryFn: () => fetchData({ albumId, photoId }), + placeholderData: keepPreviousData, + ...options, + }); + + return query; +} diff --git a/src/feature/photo-detail/hooks/usePhotoDownloadMutation.ts b/src/feature/photo-detail/hooks/usePhotoDownloadMutation.ts new file mode 100644 index 00000000..0ee17296 --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoDownloadMutation.ts @@ -0,0 +1,37 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation, useQueryClient } from '@tanstack/react-query'; +import { updateCacheAlbumPhotosDownload } from '../modules/updateCacheAlbumPhotosDownload'; + +const fetchData = async ({ + albumId, + photoIds, +}: UsePhotoDownloadMutationProps) => { + const response = await api.post({ + path: EP.photo.presignedDownload(), + body: { code: albumId, photoIds: photoIds }, + }); + + return response.result; +}; + +interface UsePhotoDownloadMutationProps { + albumId: string; + photoIds: number[]; +} + +export function usePhotoDownloadMutation() { + const queryClient = useQueryClient(); + const mutation = useMutation({ + mutationFn: ({ albumId, photoIds }: UsePhotoDownloadMutationProps) => + fetchData({ albumId, photoIds }), + onSuccess: (_data, variables) => { + const { albumId, photoIds } = variables; + photoIds.forEach((photoId) => + updateCacheAlbumPhotosDownload({ albumId, photoId, queryClient }), + ); + }, + }); + + return mutation; +} diff --git a/src/feature/photo-detail/hooks/usePhotoExifQuery.ts b/src/feature/photo-detail/hooks/usePhotoExifQuery.ts new file mode 100644 index 00000000..e02552bf --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoExifQuery.ts @@ -0,0 +1,14 @@ +import { getExifFromUrl, PhotoExifInfo } from '@/global/utils/getExifFromUrl'; +import { useQuery } from '@tanstack/react-query'; + +export function usePhotoExifQuery(imageUrl?: string) { + return useQuery({ + queryKey: ['photoExif', imageUrl], + queryFn: () => { + if (!imageUrl) throw new Error('imageUrl이 없습니다.'); + // TODO : exif 가져오는 로직 이슈 있음. 사진 fetch 시 CORS발생 + return getExifFromUrl(imageUrl); + }, + enabled: !!imageUrl, + }); +} diff --git a/src/feature/photo-detail/hooks/usePhotoLikedMutation.ts b/src/feature/photo-detail/hooks/usePhotoLikedMutation.ts new file mode 100644 index 00000000..b5f4dc32 --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoLikedMutation.ts @@ -0,0 +1,18 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +async function fetchData(photoId: number) { + const res = await api.post({ + path: EP.photo.like(photoId), + }); + return res.result; +} + +export function usePhotoLikedMutation() { + const mutation = useMutation({ + mutationFn: (photoId: number) => fetchData(photoId), + }); + + return mutation; +} diff --git a/src/feature/photo-detail/hooks/usePhotoLikersQuery.ts b/src/feature/photo-detail/hooks/usePhotoLikersQuery.ts new file mode 100644 index 00000000..e54986f5 --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoLikersQuery.ts @@ -0,0 +1,33 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery } from '@tanstack/react-query'; + +interface FetchDataProps { + albumId: string; + photoId: number; +} + +const fetchData = async ({ albumId, photoId }: FetchDataProps) => { + const response = await api.get({ + path: EP.album.albumPhotosLikers(albumId, photoId), + }); + + return response.result; +}; + +interface UsePhotoLikersQueryProps { + albumId: string; + photoId: number; +} + +export function usePhotoLikersQuery({ + albumId, + photoId, +}: UsePhotoLikersQueryProps) { + const query = useQuery({ + queryKey: [EP.album.albumPhotosLikers(albumId, photoId)], + queryFn: () => fetchData({ albumId, photoId }), + }); + + return query; +} diff --git a/src/feature/photo-detail/hooks/usePhotoUnlikedMutation.ts b/src/feature/photo-detail/hooks/usePhotoUnlikedMutation.ts new file mode 100644 index 00000000..c15365c7 --- /dev/null +++ b/src/feature/photo-detail/hooks/usePhotoUnlikedMutation.ts @@ -0,0 +1,18 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useMutation } from '@tanstack/react-query'; + +async function fetchData(photoId: number) { + const res = await api.delete({ + path: EP.photo.unlike(photoId), + }); + return res.result; +} + +export function usePhotoUnlikedMutation() { + const mutation = useMutation({ + mutationFn: (photoId: number) => fetchData(photoId), + }); + + return mutation; +} diff --git a/src/feature/photo-detail/modules/updateCacheAlbumPhotosDownload.ts b/src/feature/photo-detail/modules/updateCacheAlbumPhotosDownload.ts new file mode 100644 index 00000000..930c0dce --- /dev/null +++ b/src/feature/photo-detail/modules/updateCacheAlbumPhotosDownload.ts @@ -0,0 +1,41 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { InfiniteData, QueryClient } from '@tanstack/react-query'; + +interface UpdateCacheAlbumPhotosDownloadParams { + queryClient: QueryClient; + albumId: string; + photoId: number; +} + +export function updateCacheAlbumPhotosDownload({ + queryClient, + albumId, + photoId, +}: UpdateCacheAlbumPhotosDownloadParams) { + queryClient.setQueriesData>( + { queryKey: [EP.album.photos(albumId)] }, + (old) => { + if (!old) return old; + + return { + ...old, + pages: old.pages.map((page) => { + if (!page || !page.responses) return page; + + return { + ...page, + responses: page.responses.map((res) => + res.photoId === photoId + ? { + ...res, + isRecentlyDownloaded: true, + isDownloaded: true, + } + : res, + ), + }; + }), + }; + }, + ); +} diff --git a/src/feature/photo-detail/modules/updateCacheAlbumPhotosLike.ts b/src/feature/photo-detail/modules/updateCacheAlbumPhotosLike.ts new file mode 100644 index 00000000..d5abb5bb --- /dev/null +++ b/src/feature/photo-detail/modules/updateCacheAlbumPhotosLike.ts @@ -0,0 +1,52 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { InfiniteData, QueryClient } from '@tanstack/react-query'; + +interface ToggleAlbumPhotoLikeInCacheParams { + queryClient: QueryClient; + albumId: string; + photoId: number; + isCurrentlyLiked: boolean; +} + +export function updateCacheAlbumPhotosLike({ + queryClient, + albumId, + photoId, + isCurrentlyLiked, +}: ToggleAlbumPhotoLikeInCacheParams) { + queryClient.setQueriesData>( + { queryKey: [EP.album.photos(albumId)] }, + (old) => { + if (!old) return old; + + return { + ...old, + pages: old.pages.map((page) => { + if (!page) return page; + if (!page.responses) return page; + + return { + ...page, + responses: page.responses.map((res) => { + if (res.photoId !== photoId) return res; + + const updated = { ...res }; + + if (res.isLiked !== undefined) { + updated.isLiked = !res.isLiked; + } + + if (res.likeCnt !== undefined) { + updated.likeCnt = isCurrentlyLiked + ? res.likeCnt - 1 + : res.likeCnt + 1; + } + + return updated; + }), + }; + }), + }; + }, + ); +} diff --git a/src/feature/photo-detail/util/calcThumbSwiperCenterOffset.ts b/src/feature/photo-detail/util/calcThumbSwiperCenterOffset.ts new file mode 100644 index 00000000..028b41d4 --- /dev/null +++ b/src/feature/photo-detail/util/calcThumbSwiperCenterOffset.ts @@ -0,0 +1,51 @@ +interface CalcThumbSwiperCenterOffsetProps { + /** + * 현재 뷰포트(또는 Swiper 컨테이너)의 전체 너비 (px 단위) + * - 기본적으로 window.innerWidth를 사용 + * - 반응형 환경에서는 부모 컨테이너 width를 넘겨도 됨 + */ + viewportWidth: number; + + /** + * 활성(선택된) 썸네일의 실제 렌더링 폭 (px 단위) + */ + activeWidth: number; + + /** + * 활성 썸네일의 좌우 margin 값 (px 단위) + */ + activeMargin: number; + + /** + * 비활성 썸네일의 폭 (px 단위) + */ + inactiveWidth: number; + + /** + * 비활성 썸네일의 좌우 margin 값 (px 단위) + */ + inactiveMargin: number; + + /** + * 현재 활성화된 슬라이드의 인덱스 + */ + index: number; +} + +/** preview swiper의 active요소를 화면 중앙에 정렬시키기 위한 translate(오른쪽으로부터 얼마나 떨어져있는지) 계산 */ +export const calcThumbSwiperCenterOffset = ({ + viewportWidth, + activeWidth, + activeMargin, + inactiveWidth, + inactiveMargin, + index, +}: CalcThumbSwiperCenterOffsetProps) => { + // 화면 중앙에서 활성 썸네일의 중심까지 거리 + const baseOffset = viewportWidth / 2 - (activeMargin + activeWidth / 2); + + // 인덱스만큼 비활성 썸네일 폭만큼 왼쪽으로 이동 + const indexOffset = (inactiveWidth + inactiveMargin) * index; + + return baseOffset - indexOffset; +}; diff --git a/src/feature/photo-detail/util/downloadImageFromUrl.ts b/src/feature/photo-detail/util/downloadImageFromUrl.ts new file mode 100644 index 00000000..1faebc3b --- /dev/null +++ b/src/feature/photo-detail/util/downloadImageFromUrl.ts @@ -0,0 +1,23 @@ +export async function downloadImageFromUrl( + imageUrl: string, + fallbackFileName?: string, +) { + // TODO : 사진 fetch 시 CORS발생 수정 필요 + const res = await fetch(imageUrl, { + mode: 'cors', + credentials: 'include', + headers: { Accept: 'image/*' }, + }); + if (!res.ok) throw new Error('이미지 요청 실패'); + + const blob = await res.blob(); + const blobUrl = URL.createObjectURL(blob); + + const a = document.createElement('a'); + a.href = blobUrl; + a.download = fallbackFileName || 'download.jpg'; + document.body.appendChild(a); + a.click(); + document.body.removeChild(a); + URL.revokeObjectURL(blobUrl); +} diff --git a/src/feature/photo-entry/components/AlbumPreviewCard.tsx b/src/feature/photo-entry/components/AlbumPreviewCard.tsx new file mode 100644 index 00000000..392d8c79 --- /dev/null +++ b/src/feature/photo-entry/components/AlbumPreviewCard.tsx @@ -0,0 +1,34 @@ +interface AlbumPreviewCardProps { + imageUrl: string; + nickname: string; + profileUrl: string; +} + +export default function AlbumPreviewCard({ + imageUrl, + nickname, + profileUrl, +}: AlbumPreviewCardProps) { + return ( +
+ 미리보기 이미지 +
+ {`${nickname}님의 + {nickname} +
+
+ ); +} diff --git a/src/feature/photo-entry/components/AlbumSharePreviewSection.tsx b/src/feature/photo-entry/components/AlbumSharePreviewSection.tsx new file mode 100644 index 00000000..7c11bbbd --- /dev/null +++ b/src/feature/photo-entry/components/AlbumSharePreviewSection.tsx @@ -0,0 +1,65 @@ +'use client'; + +import ThreeTags_Fill_Album from '@/../public/assets/album/3Tags_Fill_Album.json'; +import { useGetAlbumInvitation } from '@/feature/album/detail/hooks/useGetAlbumInvitation'; +import { useAlbumPhotosInfiniteQuery } from '@/feature/photo-detail/hooks/useAlbumPhotosInfiniteQuery'; +import MarqueeCarousel from '@/global/components/carousel/MarqueeCarousel'; +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; +import dynamic from 'next/dynamic'; +import AlbumPreviewCard from './AlbumPreviewCard'; +const Lottie = dynamic(() => import('lottie-react'), { ssr: false }); + +interface AlbumSharePreviewSectionProps { + albumId: string; +} + +export default function AlbumSharePreviewSection({ + albumId, +}: AlbumSharePreviewSectionProps) { + const { data, isPending, isError } = useGetAlbumInvitation(albumId); + const { items, isLoading: isItemsLoading } = useAlbumPhotosInfiniteQuery({ + code: albumId, + }); + + if (isPending || isItemsLoading) return null; + if (isError) return null; + if (!data) return null; + + const isEmpty = items.length === 0; + + return ( + <> +
+ + {convertUnicodeToEmoji(data.themeEmoji)} + +
+ +
+

+ {data.title} +

+

{data.eventDate}

+
+ +
+ {isEmpty ? ( + + ) : ( + ( +
+ +
+ ))} + itemWidth={180} + /> + )} +
+ + ); +} diff --git a/src/feature/photo-entry/components/ScreenPhotoShareEntry.tsx b/src/feature/photo-entry/components/ScreenPhotoShareEntry.tsx new file mode 100644 index 00000000..28310874 --- /dev/null +++ b/src/feature/photo-entry/components/ScreenPhotoShareEntry.tsx @@ -0,0 +1,92 @@ +'use client'; +import { useAlbumEnterMutation } from '@/feature/album-entry/hooks/useAlbumEnterMutation'; +import { useGetAlbumAvailableCount } from '@/feature/album/detail/hooks/useGetAlbumAvailableCount'; +import { handleFileUpload } from '@/feature/create-album/utils/handleFileUpload'; +import CheckNoImgModal from '@/feature/upload/components/CheckNoImgModal'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import LongButton from '@/global/components/LongButton'; +import Toast from '@/global/components/toast/Toast'; +import BubbleTooltip from '@/global/components/tooltip/BubbleTooltip'; +import { useRouter, useSearchParams } from 'next/navigation'; +import { useEffect, useRef } from 'react'; +import AlbumSharePreviewSection from './AlbumSharePreviewSection'; + +interface ScreenPhotoShareEntryProps { + albumId: string; +} + +export default function ScreenPhotoShareEntry({ + albumId, +}: ScreenPhotoShareEntryProps) { + const router = useRouter(); + const searchParams = useSearchParams(); + const isInvite = searchParams.get('isInvite'); + const fileInputRef = useRef(null); + + const { mutateAsync } = useAlbumEnterMutation(); + const { data } = useGetAlbumAvailableCount(albumId); + + useEffect(() => { + if (!isInvite) return; + + mutateAsync({ albumId }); + }, []); + + const handleUpload = () => { + fileInputRef.current?.click(); + }; + + const onFileChange = async (e: React.ChangeEvent) => { + const { success } = await handleFileUpload(e, albumId, router); + + if (success) { + setTimeout(() => Toast.check(`총 ${success}장을 앨범에 채웠어요.`), 2000); + } + }; + + return ( + <> + + +
+ + +
+
+ {data?.availableCount && ( + + )} + + +
+ + + 올릴 사진이 없어요 + + } + /> +
+
+ + ); +} diff --git a/src/feature/root/components/RendingFooter.tsx b/src/feature/root/components/RendingFooter.tsx new file mode 100644 index 00000000..4dc8f805 --- /dev/null +++ b/src/feature/root/components/RendingFooter.tsx @@ -0,0 +1,26 @@ +import Link from 'next/link'; + +export default function RendingFooter() { + return ( +
+ 치이이즈 + + @치이이즈. ALL RIGHTS RESERVED + + + 버그 및 불편사항 제보 + + + 개인정보처리방침 | 서비스 이용약관 + +
+ ); +} diff --git a/src/feature/root/components/ScreenRoot.tsx b/src/feature/root/components/ScreenRoot.tsx new file mode 100644 index 00000000..645133c4 --- /dev/null +++ b/src/feature/root/components/ScreenRoot.tsx @@ -0,0 +1,139 @@ +'use client'; +import LogoHeader from '@/global/components/header/LogoHeader'; +import LongButton from '@/global/components/LongButton'; +import { useCheckAuth } from '@/global/hooks/useCheckAuth'; +import dynamic from 'next/dynamic'; +import Image from 'next/image'; +import { useRouter } from 'next/navigation'; +import { useCallback, useState } from 'react'; +import SelectMenu from './SelectMenu'; + +const RendingFooter = dynamic(() => import('./RendingFooter'), { ssr: false }); +const SelectedList = dynamic(() => import('./SelectedList'), { ssr: false }); +const SwipeList = dynamic(() => import('./SwipeList'), { ssr: false }); + +export default function ScreenRoot() { + const router = useRouter(); + const [selectedMenu, setSelectedMenu] = useState< + 'first' | 'second' | 'third' + >('first'); + const handleCreateAlbumClick = useCallback(() => { + if (typeof document !== 'undefined') { + document.cookie = 'entry=create-album; path=/;'; + } + router.push('/login'); + }, [router]); + + useCheckAuth({ onAuthed: () => router.push('/main') }); + + return ( +
+ + + 딱 7일만 열리는 특별한
+ 공유 앨범 서비스 +
+ +
+ 블러 배경 + 핸드폰 일러스트 +
+ + + 92개 모임에서 치이이즈를 사용했어요 + + + + 삼각형 + 소개 텍스트 + 치즈 아이콘 + 치즈 아이콘 + + 일정의 마무리, +
+ 사진 주고받기도 가볍고 귀엽게 +
+
+ 추억을 더 소중하게 만드는 + 치이이즈의 세 가지 방법 +
+
+ +
+
+ +
+
+ 서비스에 그대로 담긴 + 사진 찍고 나눌 때의 감정 +
+ 박스 일러스트 +
+ 추억할 일이 있는 모든 곳에서 + 모두 함께 치이이즈 +
+ + +
+ ); +} diff --git a/src/feature/root/components/SelectMenu.tsx b/src/feature/root/components/SelectMenu.tsx new file mode 100644 index 00000000..2913de44 --- /dev/null +++ b/src/feature/root/components/SelectMenu.tsx @@ -0,0 +1,64 @@ +'use client'; + +interface MenuItemProps { + title: string; + selected?: boolean; + onClick?: () => void; +} + +function MenuItem({ title, selected, onClick }: MenuItemProps) { + return ( +
+ {title} +
+ ); +} + +const MENU_TITLES = ['7일', '띱', '치즈네컷']; + +export interface SelectMenuProps { + selectedMenu?: 'first' | 'second' | 'third'; + setSelectedMenu?: (menu: 'first' | 'second' | 'third') => void; +} + +export default function SelectMenu({ + selectedMenu = 'first', // 기본 first (1번) + setSelectedMenu, +}: SelectMenuProps) { + const selectedIdx = + selectedMenu === 'first' + ? 0 + : selectedMenu === 'second' + ? 1 + : selectedMenu === 'third' + ? 2 + : 0; // 기본 0 + + const handleClick = (idx: number) => { + if (!setSelectedMenu) return; + if (idx === 0) setSelectedMenu('first'); + else if (idx === 1) setSelectedMenu('second'); + else if (idx === 2) setSelectedMenu('third'); + }; + + return ( +
+ {MENU_TITLES.map((title, idx) => ( + handleClick(idx)} + /> + ))} +
+ ); +} diff --git a/src/feature/root/components/SelectedList.tsx b/src/feature/root/components/SelectedList.tsx new file mode 100644 index 00000000..b0fb19d2 --- /dev/null +++ b/src/feature/root/components/SelectedList.tsx @@ -0,0 +1,147 @@ +'use client'; + +function debounce( + func: (...args: T) => void, + wait: number, +) { + let timeout: ReturnType; + return (...args: T) => { + clearTimeout(timeout); + timeout = setTimeout(() => func(...args), wait); + }; +} + +import Image from 'next/image'; +import { useEffect, useRef } from 'react'; + +interface SelectedListProps { + selectedMenu: 'first' | 'second' | 'third'; + setSelectedMenu: (menu: 'first' | 'second' | 'third') => void; +} + +const MENU_IMAGES = [ + { key: 'first', src: '/assets/rending/first.svg', alt: '7일 설명' }, + { key: 'second', src: '/assets/rending/second.svg', alt: '띱 설명' }, + { key: 'third', src: '/assets/rending/third.svg', alt: '치즈네컷 설명' }, +]; + +export default function SelectedList({ + selectedMenu, + setSelectedMenu, +}: SelectedListProps) { + const selectedIdx = + selectedMenu === 'first' + ? 0 + : selectedMenu === 'second' + ? 1 + : selectedMenu === 'third' + ? 2 + : 0; + + const containerRef = useRef(null); + const isScrollingProgrammatically = useRef(false); + const ITEM_WIDTH = '100%'; // 각 아이템의 너비 (부모 컨테이너에서 좌우 패딩 및 간격 제외한 값) + + // 스크롤 시 중앙에 가장 가까운 이미지의 index 계산 및 상태 변경 (debounce 적용) + + const handleScrollCore = () => { + if (!containerRef.current) return; + if (isScrollingProgrammatically.current) return; + const containerRect = containerRef.current.getBoundingClientRect(); + const containerCenter = containerRect.left + containerRect.width / 2; + const children = containerRef.current.children; + + let closestIndex = 0; + let minDistance = Infinity; + + for (let i = 0; i < children.length; i++) { + const childRect = children[i].getBoundingClientRect(); + const childCenter = childRect.left + childRect.width / 2; + const distance = Math.abs(childCenter - containerCenter); + if (distance < minDistance) { + minDistance = distance; + closestIndex = i; + } + } + + const newMenu = MENU_IMAGES[closestIndex].key as + | 'first' + | 'second' + | 'third'; + if (newMenu !== selectedMenu) { + setSelectedMenu(newMenu); + } + }; + + const handleScroll = debounce(handleScrollCore, 50); + + useEffect(() => { + if (!containerRef.current) return; + + const container = containerRef.current; + const children = container.children; + + if (children[selectedIdx]) { + const targetChild = children[selectedIdx] as HTMLElement; + const containerRect = container.getBoundingClientRect(); + const targetRect = targetChild.getBoundingClientRect(); + + const scrollLeft = + container.scrollLeft + + (targetRect.left - containerRect.left) - + (containerRect.width / 2 - targetRect.width / 2); + + isScrollingProgrammatically.current = true; + container.scrollTo({ left: scrollLeft, behavior: 'smooth' }); + + setTimeout(() => { + isScrollingProgrammatically.current = false; + }, 500); + } + }, [selectedIdx]); + + return ( +
+
+ {MENU_IMAGES.map((image) => ( +
+ {image.alt} +
+ ))} +
+
+ {MENU_IMAGES.map((image, idx) => ( +
+ setSelectedMenu(image.key as 'first' | 'second' | 'third') + } + /> + ))} +
+
+ ); +} diff --git a/src/feature/root/components/SwipeList.tsx b/src/feature/root/components/SwipeList.tsx new file mode 100644 index 00000000..a79d2d09 --- /dev/null +++ b/src/feature/root/components/SwipeList.tsx @@ -0,0 +1,30 @@ +import MarqueeCarousel from '@/global/components/carousel/MarqueeCarousel'; +import Image from 'next/image'; + +const SWIPE_IMAGES = [ + '/assets/rending/swipe/1.png', + '/assets/rending/swipe/2.png', + '/assets/rending/swipe/3.png', + '/assets/rending/swipe/4.png', +]; + +export default function SwipeList() { + return ( + ( + {`스와이프 + ))} + itemWidth={163} + gap={0} + speed={60} + className='py-2' + /> + ); +} diff --git a/src/feature/term/ScreenTerm.tsx b/src/feature/term/ScreenTerm.tsx new file mode 100644 index 00000000..411de989 --- /dev/null +++ b/src/feature/term/ScreenTerm.tsx @@ -0,0 +1,30 @@ +'use client'; +import { TermContent } from '@/feature/onboarding/components/TermContent'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import { useSearchParams } from 'next/navigation'; + +export default function ScreenTerm() { + const searchParams = useSearchParams(); + const termType = searchParams.get('type'); + const currentTerm = + termType && Object.prototype.hasOwnProperty.call(TermContent, termType) + ? TermContent[termType as keyof typeof TermContent] + : null; + + if (!currentTerm) { + return ( +
+ 존재하지 않는 약관입니다. +
+ ); + } + + return ( +
+ +
+ +
+
+ ); +} diff --git a/src/feature/upload/api/getAlbumInvitation.server.ts b/src/feature/upload/api/getAlbumInvitation.server.ts new file mode 100644 index 00000000..e3a635ee --- /dev/null +++ b/src/feature/upload/api/getAlbumInvitation.server.ts @@ -0,0 +1,16 @@ +import { serverApi } from '@/global/utils/serverApi'; +import type { AlbumInvitationResponse } from './getAlbumInvitation'; + +/** + * 앨범 초대 정보 조회 (Server Component용) + * @param code 앨범 코드 + * @returns 앨범 초대 정보 + */ +export async function getAlbumInvitationServer( + code: string, +): Promise { + const response = await serverApi.get({ + path: `/v1/album/${code}/invitation`, + }); + return response.result; +} diff --git a/src/feature/upload/api/getAlbumInvitation.ts b/src/feature/upload/api/getAlbumInvitation.ts new file mode 100644 index 00000000..23b3e1f6 --- /dev/null +++ b/src/feature/upload/api/getAlbumInvitation.ts @@ -0,0 +1,25 @@ +import { api } from '@/global/utils/api'; + +export type AlbumInvitationResponse = { + title: string; + themeEmoji: string; + eventDate: string; + expiredAt: string; + makerName: string; + makerProfileImage: string; + isExpired: boolean; +}; + +/** + * 앨범 초대 정보 조회 (Client Component용) + * @param code 앨범 코드 + * @returns 앨범 초대 정보 + */ +export async function getAlbumInvitation( + code: string, +): Promise { + const response = await api.get({ + path: `/v1/album/${code}/invitation`, + }); + return response.result; +} diff --git a/src/feature/upload/components/AlbumInfoHeader.tsx b/src/feature/upload/components/AlbumInfoHeader.tsx new file mode 100644 index 00000000..f9fff0a4 --- /dev/null +++ b/src/feature/upload/components/AlbumInfoHeader.tsx @@ -0,0 +1,61 @@ +import { convertUnicodeToEmoji } from '@/global/utils/convertEmoji'; + +export interface AlbumParticipant { + name?: string; + profileImage?: string; + role?: string; // "MAKER" | "GUEST" 등으로 좁힐 수 있음 + isMe?: boolean; +} + +export interface AlbumInvitation { + currentParticipantCount?: number; + eventDate?: string; + expiredAt?: string; + isExpired?: boolean; + maxParticipantCount?: number; + myRole?: string; + participants?: AlbumParticipant[]; + themeEmoji?: string; + title?: string; +} + +type AlbumInfoHeaderProps = { + photoCount: number; + albumData: AlbumInvitation; +}; + +export default function AlbumInfoHeader({ + photoCount, + albumData, +}: AlbumInfoHeaderProps) { + const emoji = convertUnicodeToEmoji(albumData.themeEmoji || ''); + + return ( +
+
+ {emoji} +
+ +
+

+ {albumData.title || '제목 없음'} +

+ {photoCount === 0 ? ( +

+ 앨범을 채워주세요 +

+ ) : ( +

+ {albumData.eventDate || '날짜 없음'} +

+ )} +
+
+ ); +} diff --git a/src/feature/upload/components/AvailableCountBubble.tsx b/src/feature/upload/components/AvailableCountBubble.tsx new file mode 100644 index 00000000..636ee8d5 --- /dev/null +++ b/src/feature/upload/components/AvailableCountBubble.tsx @@ -0,0 +1,23 @@ +interface AvailableCountBubbleProps { + availableCount: number; +} + +export default function AvailableCountBubble({ + availableCount, +}: AvailableCountBubbleProps) { + return ( +
+
+
+ + 📸 + + 지금 {availableCount}장 더 올릴 수 있어요 +
+
+
+
+
+
+ ); +} diff --git a/src/feature/upload/components/CheckNoImgModal.tsx b/src/feature/upload/components/CheckNoImgModal.tsx new file mode 100644 index 00000000..571f02b6 --- /dev/null +++ b/src/feature/upload/components/CheckNoImgModal.tsx @@ -0,0 +1,67 @@ +'use client'; + +import { handleFileUpload } from '@/feature/create-album/utils/handleFileUpload'; +import ConfirmModal from '@/global/components/modal/ConfirmModal'; +import Toast from '@/global/components/toast/Toast'; +import { useRouter } from 'next/navigation'; +import { ReactNode, useRef } from 'react'; + +interface CheckNoImgModalProps { + trigger: ReactNode; + albumId: string; + onConfirm?: () => void; +} + +export default function CheckNoImgModal({ + trigger, + albumId, + onConfirm, +}: CheckNoImgModalProps) { + const router = useRouter(); + const fileInputRef = useRef(null); + + const handleCancel = () => { + fileInputRef.current?.click(); + }; + + const handleConfirm = () => { + if (onConfirm) { + onConfirm(); + return; + } + // 기본 동작: 현재 앨범 경로로 이동 (WaitingAlbum에서 분기 처리) + router.push(`/album/detail/${albumId}`); + }; + + const handleFileChange = async (e: React.ChangeEvent) => { + try { + await handleFileUpload(e, albumId, router); + } catch (e) { + console.error(e); + Toast.alert('사진 업로드 중 오류가 발생했습니다.'); + } + }; + + return ( + <> + + + + ); +} diff --git a/src/feature/upload/components/UploadAlbumPage.tsx b/src/feature/upload/components/UploadAlbumPage.tsx new file mode 100644 index 00000000..67004824 --- /dev/null +++ b/src/feature/upload/components/UploadAlbumPage.tsx @@ -0,0 +1,71 @@ +'use client'; +import LotateAnimation from '@/../public/assets/upload/3Tags_Fill Album.json'; +import CheckNoImgModal from '@/feature/upload/components/CheckNoImgModal'; +import CustomHeader from '@/global/components/header/CustomHeader'; +import dynamic from 'next/dynamic'; +import { useGetAlbumInform } from '../hooks/useGetAlbumInform'; +import AlbumInfoHeader from './AlbumInfoHeader'; +import AvailableCountBubble from './AvailableCountBubble'; +import UploadButton from './UploadButton'; + +const Lottie = dynamic(() => import('lottie-react'), { ssr: false }); + +interface AlbumCard { + imageUrl: string; + nickname: string; + profileUrl: string; +} + +// ✅ API 연동 전까지는 상수로 유지 +const MOCK_CARDS: AlbumCard[] = []; + +interface UploadAlbumPageProps { + albumId: string; +} + +export default function UploadAlbumPage({ albumId }: UploadAlbumPageProps) { + const { data } = useGetAlbumInform({ code: albumId }); + const cards = MOCK_CARDS; // ✅ 나중에 API 결과로 교체 예정 + const availableCount = + (data?.maxParticipantCount ?? 0) - (data?.currentParticipantCount ?? 0); + return ( +
+ +
+
+ {data && ( + + )} + +
+ +
+ {data?.myRole !== 'MAKER' ? ( + + ) : ( + + Tip. 첫 업로드가 참여도를 두 배 넘게 끌려올려요 + + )} + + + 올릴 사진이 없어요 + + } + /> +
+
+
+ ); +} diff --git a/src/feature/upload/components/UploadButton.tsx b/src/feature/upload/components/UploadButton.tsx new file mode 100644 index 00000000..18426639 --- /dev/null +++ b/src/feature/upload/components/UploadButton.tsx @@ -0,0 +1,43 @@ +'use client'; + +import { handleFileUpload } from '@/feature/create-album/utils/handleFileUpload'; +import LongButton from '@/global/components/LongButton'; +import { useRouter } from 'next/navigation'; +import { useRef } from 'react'; + +type UploadButtonProps = { + albumId: string; +}; + +export default function UploadButton({ albumId }: UploadButtonProps) { + const router = useRouter(); + const fileInputRef = useRef(null); + + function handleButtonClick() { + fileInputRef.current?.click(); + } + + async function onFileChange(e: React.ChangeEvent) { + await handleFileUpload(e, albumId, router); + } + + return ( + <> + +
+ +
+ + ); +} diff --git a/src/feature/upload/hooks/useGetAlbumInform.server.ts b/src/feature/upload/hooks/useGetAlbumInform.server.ts new file mode 100644 index 00000000..082784ba --- /dev/null +++ b/src/feature/upload/hooks/useGetAlbumInform.server.ts @@ -0,0 +1,9 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { serverApi } from '@/global/utils/serverApi'; + +export async function getAlbumDataWithRoleServer(code: string) { + const data = await serverApi.get({ + path: EP.album.participants(code), + }); + return data.result; +} diff --git a/src/feature/upload/hooks/useGetAlbumInform.ts b/src/feature/upload/hooks/useGetAlbumInform.ts new file mode 100644 index 00000000..9ceca3b8 --- /dev/null +++ b/src/feature/upload/hooks/useGetAlbumInform.ts @@ -0,0 +1,22 @@ +import { ApiReturns, EP } from '@/global/api/ep'; +import { api } from '@/global/utils/api'; +import { useQuery } from '@tanstack/react-query'; + +const getAlbumDataWithRole = async (code: string) => { + const data = await api.get({ + path: EP.album.participants(code), + }); + return data.result; +}; + +interface UseGetAlbumInformProps { + code: string; +} + +export function useGetAlbumInform({ code }: UseGetAlbumInformProps) { + return useQuery({ + queryKey: [EP.album.participants(code)], + queryFn: () => getAlbumDataWithRole(code), + enabled: !!code, + }); +} diff --git a/src/global/api/ep.ts b/src/global/api/ep.ts new file mode 100644 index 00000000..cb924fdb --- /dev/null +++ b/src/global/api/ep.ts @@ -0,0 +1,204 @@ +/* AUTO-GENERATED FILE. DO NOT EDIT. + * Generated by scripts/generate-ep.ts + */ + +export const EP = { + global: { + "health": () => `/v1/global/health-check`, + }, + auth: { + "exchange": () => `/v1/auth/exchange`, + "logout": () => `/v1/auth/logout`, + "reissue": () => `/v1/auth/reissue`, + }, + user: { + "userMe": () => `/v1/user/me`, + "userMeName": () => `/v1/user/me/name`, + "userMeProfileImage": () => `/v1/user/me/profile-image`, + "userOnboarding": () => `/v1/user/onboarding`, + "userProfileImages": () => `/v1/user/profile-images`, + }, + album: { + "create": () => `/v1/album`, + "availableCount": (code: string) => `/v1/album/${code}/available-count`, + "albumBest-4cut": (code: string) => `/v1/album/${code}/best-4cut`, + "enter": (code: string) => `/v1/album/${code}/enter`, + "albumInfo": (code: string) => `/v1/album/${code}/info`, + "invitation": (code: string) => `/v1/album/${code}/invitation`, + "participants": (code: string) => `/v1/album/${code}/participants`, + "albumParticipantsMe": (code: string) => `/v1/album/${code}/participants/me`, + "albumPhoto": (code: string, photoId: number) => `/v1/album/${code}/photo/${photoId}`, + "photos": (code: string) => `/v1/album/${code}/photos`, + "photoDetail": (code: string, photoId: number) => `/v1/album/${code}/photos/${photoId}`, + "albumPhotosLikers": (code: string, photoId: number) => `/v1/album/${code}/photos/${photoId}/likers`, + "likedPhotos": (code: string) => `/v1/album/${code}/photos/liked`, + "albumClosed": () => `/v1/album/closed`, + "albumOpen": () => `/v1/album/open`, + "albumOpenMe": () => `/v1/album/open/me`, + }, + photo: { + "like": (photoId: number) => `/v1/photo/${photoId}/liked`, + "unlike": (photoId: number) => `/v1/photo/${photoId}/unliked`, + "presignedDownload": () => `/v1/photo/download-url`, + "presignedUpload": () => `/v1/photo/presigned-url`, + "reportUploadResult": () => `/v1/photo/report`, + }, + cheese4cut: { + "finalize": (code: string) => `/v1/cheese4cut/${code}/fixed`, + "preview": (code: string) => `/v1/cheese4cut/${code}/preview`, + }, + internal: { + "thumbnailComplete": () => `/internal/thumbnail/complete`, + } +} as const; + +// 선택지 Enum(정렬) +export type PhotoSorting = 'POPULAR' | 'CAPTURED_AT' | 'CREATED_AT'; + +/* ======================= + * Generated Types + * ======================= */ + +// --- Components +export interface UserOnboardingRequestSchema { "name": string; "imageCode": string; "isServiceAgreement": boolean; "isUserInfoAgreement": boolean; "isMarketingAgreement"?: boolean; "isThirdPartyAgreement": boolean; } +export interface CommonResponseVoidSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: unknown; } +export interface PhotoUploadReportRequestSchema { "failurePhotoIds": number[]; } +export interface FileInfoSchema { "fileName": string; "captureTime": string; "fileSize"?: number; "contentType": string; } +export interface PhotoPresignedUrlRequestSchema { "albumCode": string; "fileInfos": FileInfoSchema[]; } +export interface CommonResponsePhotoPresignedUrlResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoPresignedUrlResponseSchema; } +export interface PhotoPresignedUrlResponseSchema { "presignedUrlInfos": PresignedUrlInfoSchema[]; } +export interface PresignedUrlInfoSchema { "photoId": number; "uploadUrl": string; } +export interface PhotoDownloadRequestSchema { "code": string; "photoIds": number[]; } +export interface CommonResponsePhotoDownloadResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoDownloadResponseSchema; } +export interface DownloadFileInfoSchema { "photoId": number; "downloadUrl": string; "fileName": string; "captureTime": string; "createdAt": string; } +export interface PhotoDownloadResponseSchema { "downloadFiles": DownloadFileInfoSchema[]; } +export interface Cheese4cutFixedRequestSchema { "photoIds": number[]; } +export interface AuthReissueRequestSchema { "refreshToken": string; } +export interface AuthReissueResponseSchema { "accessToken": string; "refreshToken": string; } +export interface CommonResponseAuthReissueResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AuthReissueResponseSchema; } +export interface AlbumCreationRequestSchema { "themeEmoji": string; "title": string; "participant": number; "eventDate": string; } +export interface AlbumCreationResponseSchema { "themeEmoji": string; "title": string; "eventDate": string; "currentPhotoCnt": number; "code": string; } +export interface CommonResponseAlbumCreationResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AlbumCreationResponseSchema; } +export type AlbumEnterResponseSchema = unknown; +export interface AlbumMakerInfoSchema { "makerName": string; "makerProfileImage": string; } +export interface CommonResponseAlbumEnterResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: ExistingEnterResponseSchema | NewEnterResponseSchema; } +export type ExistingEnterResponseSchema = AlbumEnterResponseSchema & { "joinStatus"?: "NEW" | "EXISTING" | "REJOINED"; "title"?: string; "themeEmoji"?: string; "eventDate"?: string; "expiredAt"?: string; "makerInfo"?: AlbumMakerInfoSchema; }; +export type NewEnterResponseSchema = AlbumEnterResponseSchema & { "joinStatus"?: "NEW" | "EXISTING" | "REJOINED"; "title"?: string; "themeEmoji"?: string; "eventDate"?: string; "expiredAt"?: string; "makerInfo"?: AlbumMakerInfoSchema; "remainingUploadSlots"?: number; "recentPhotos"?: RecentPhotoResponseSchema; }; +export interface RecentPhotoResponseSchema { "thumbnailUrl": string; "uploaderName": string; "uploaderProfileImage": string; } +export interface PhotoCompleteRequestSchema { "photoId": number; "thumbnailUrl": string; } +export interface UserProfileImageRequestSchema { "imageCode"?: string; } +export interface UserProfileRequestSchema { "name"?: string; } +export interface CommonResponseUserProfileImageResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: UserProfileImageResponseSchema; } +export interface ProfileImageOptSchema { "imageCode"?: string; "profileImageUrl"?: string; } +export interface UserProfileImageResponseSchema { "opts"?: ProfileImageOptSchema[]; } +export interface CommonResponseUserInfoResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: UserInfoResponseSchema; } +export interface UserInfoResponseSchema { "profileImage": string; "email": string; "name": string; "albumCount": number; "photoCount": number; "likesCount": number; } +export interface CommonResponseStringSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: string; } +export type Cheese4cutFinalResponseSchema = Cheese4cutResponseSchema & { "isFinalized"?: boolean; "photos"?: FinalPhotoInfoSchema[]; }; +export type Cheese4cutPreviewResponseSchema = Cheese4cutResponseSchema & { "isFinalized"?: boolean; "previewPhotos"?: PreviewPhotoInfoSchema[]; "uniqueLikesCount"?: number; "participant"?: number; "myRole"?: "MAKER" | "GUEST" | "BLACK"; }; +export interface Cheese4cutResponseSchema { "finalized"?: boolean; } +export interface CommonResponseCheese4cutResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: Cheese4cutFinalResponseSchema | Cheese4cutPreviewResponseSchema; } +export interface FinalPhotoInfoSchema { "photoId": number; "imageUrl": string; "photoRank": number; } +export interface PreviewPhotoInfoSchema { "photoId": number; "imageUrl": string; "photoRank": number; } +export interface AuthExchangeResponseSchema { "accessToken": string; "refreshToken": string; "isOnboarded": boolean; "userId": number; "name": string; "email": string; } +export interface CommonResponseAuthExchangeResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AuthExchangeResponseSchema; } +export interface CommonResponsePhotoPageResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoPageResponseSchema; } +export interface PhotoListResponseSchema { "name"?: string; "photoId": number; "profileImage": string; "imageUrl"?: string; "thumbnailUrl": string; "likeCnt": number; "isLiked": boolean; "isDownloaded": boolean; "isRecentlyDownloaded": boolean; } +export interface PhotoPageResponseSchema { "responses": PhotoListResponseSchema[]; "listSize": number; "isFirst": boolean; "isLast": boolean; "hasNext": boolean; } +export interface CommonResponsePhotoDetailResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoDetailResponseSchema; } +export interface PhotoDetailResponseSchema { "name": string; "profileImage": string; "photoId": number; "imageUrl": string; "thumbnailUrl": string; "likesCnt": number; "isLiked": boolean; "isDownloaded": boolean; "isRecentlyDownloaded": boolean; "canDelete"?: boolean; "captureTime"?: string; "createdAt"?: string; } +export interface CommonResponsePhotoLikedUserResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoLikedUserResponseSchema; } +export interface PhotoLikedUserResponseSchema { "likeCnt"?: number; "photoLikers"?: PhotoLikerSchema[]; } +export interface PhotoLikerSchema { "name"?: string; "profileImageUrl"?: string; "isMe"?: boolean; "role"?: "MAKER" | "GUEST" | "BLACK"; } +export interface CommonResponsePhotoLikedPageResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: PhotoLikedPageResponseSchema; } +export interface PhotoLikedPageResponseSchema { "responses": PhotoLikedResponseSchema[]; "listSize": number; "isFirst": boolean; "isLast": boolean; "hasNext": boolean; } +export interface PhotoLikedResponseSchema { "name"?: string; "photoId": number; "imageUrl"?: string; "thumbnailUrl": string; "likeCnt"?: number; "isLiked"?: boolean; "isDownloaded": boolean; "isRecentlyDownloaded": boolean; } +export interface AlbumParticipantResponseSchema { "isExpired": boolean; "title": string; "themeEmoji": string; "eventDate": string; "expiredAt": string; "maxParticipantCount": number; "currentParticipantCount": number; "participants": ParticipantInfoSchema[]; "myRole"?: "MAKER" | "GUEST" | "BLACK"; } +export interface CommonResponseAlbumParticipantResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AlbumParticipantResponseSchema; } +export interface ParticipantInfoSchema { "name": string; "profileImage": string; "role": "MAKER" | "GUEST" | "BLACK"; "isMe": boolean; } +export interface AlbumInvitationResponseSchema { "title": string; "themeEmoji": string; "eventDate": string; "expiredAt": string; "makerName": string; "makerProfileImage": string; "isExpired": boolean; } +export interface CommonResponseAlbumInvitationResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AlbumInvitationResponseSchema; } +export interface AlbumInfoResponseSchema { "makerId"?: number; "name"?: string; "title"?: string; "themeEmoji"?: string; "participant"?: number; "currentParticipant"?: number; "eventDate"?: string; "currentPhotoCnt"?: number; "expiredAt"?: string; } +export interface CommonResponseAlbumInfoResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AlbumInfoResponseSchema; } +export interface AlbumBest4CutResponseSchema { "thumbnailUrl"?: string; "likeCnt"?: number; "isLiked"?: boolean; } +export interface CommonResponseListAlbumBest4CutResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: AlbumBest4CutResponseSchema[]; } +export interface CommonResponseUploadAvailableCountResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: UploadAvailableCountResponseSchema; } +export interface UploadAvailableCountResponseSchema { "availableCount": number; "maxPhotoCount": number; "currentPhotoCount": number; } +export interface CommonResponseOpenAlbumPageResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: OpenAlbumPageResponseSchema; } +export interface OpenAlbumPageResponseSchema { "responses": OpenAlbumSummaryResponseSchema[]; "listSize": number; "isFirst": boolean; "isLast": boolean; "hasNext": boolean; } +export interface OpenAlbumSummaryResponseSchema { "code": string; "themeEmoji": string; "title": string; "eventDate": string; "makerName": string; "currentParticipant": number; "participant": number; "expiredAt": string; "recentPhotoThumbnails"?: string[]; } +export interface ClosedAlbumPageResponseSchema { "responses": ClosedAlbumSummaryResponseSchema[]; "listSize": number; "isFirst": boolean; "isLast": boolean; "hasNext": boolean; } +export interface ClosedAlbumSummaryResponseSchema { "code": string; "title": string; "makerName": string; "eventDate": string; "thumbnails"?: string[]; } +export interface CommonResponseClosedAlbumPageResponseSchema { "isSuccess"?: boolean; "code"?: number; "message"?: string; "result"?: ClosedAlbumPageResponseSchema; } + +// --- Operation Response Types +export type GlobalHealthResponse = CommonResponseStringSchema["result"]; +export type AuthExchangeResponse = CommonResponseAuthExchangeResponseSchema["result"]; +export type AuthLogoutResponse = CommonResponseVoidSchema["result"]; +export type AuthReissueResponse = CommonResponseAuthReissueResponseSchema["result"]; +export type UserUserMeResponse = CommonResponseUserInfoResponseSchema["result"]; +export type UserUserMeNameResponse = CommonResponseVoidSchema["result"]; +export type UserUserMeProfileImageResponse = CommonResponseVoidSchema["result"]; +export type UserUserOnboardingResponse = CommonResponseVoidSchema["result"]; +export type UserUserProfileImagesResponse = CommonResponseUserProfileImageResponseSchema["result"]; +export type AlbumCreateResponse = CommonResponseAlbumCreationResponseSchema["result"]; +export type AlbumAvailableCountResponse = CommonResponseUploadAvailableCountResponseSchema["result"]; +export type AlbumAlbumBest4cutResponse = CommonResponseListAlbumBest4CutResponseSchema["result"]; +export type AlbumEnterResponse = CommonResponseAlbumEnterResponseSchema["result"]; +export type AlbumAlbumInfoResponse = CommonResponseAlbumInfoResponseSchema["result"]; +export type AlbumInvitationResponse = CommonResponseAlbumInvitationResponseSchema["result"]; +export type AlbumParticipantsResponse = CommonResponseAlbumParticipantResponseSchema["result"]; +export type AlbumAlbumParticipantsMeResponse = CommonResponseVoidSchema["result"]; +export type AlbumAlbumPhotoResponse = CommonResponseVoidSchema["result"]; +export type AlbumPhotosResponse = CommonResponsePhotoPageResponseSchema["result"]; +export type AlbumPhotoDetailResponse = CommonResponsePhotoDetailResponseSchema["result"]; +export type AlbumAlbumPhotosLikersResponse = CommonResponsePhotoLikedUserResponseSchema["result"]; +export type AlbumLikedPhotosResponse = CommonResponsePhotoLikedPageResponseSchema["result"]; +export type AlbumAlbumClosedResponse = CommonResponseClosedAlbumPageResponseSchema["result"]; +export type AlbumAlbumOpenResponse = CommonResponseOpenAlbumPageResponseSchema["result"]; +export type AlbumAlbumOpenMeResponse = CommonResponseOpenAlbumPageResponseSchema["result"]; +export type PhotoLikeResponse = CommonResponseVoidSchema["result"]; +export type PhotoUnlikeResponse = CommonResponseVoidSchema["result"]; +export type PhotoPresignedDownloadResponse = CommonResponsePhotoDownloadResponseSchema["result"]; +export type PhotoPresignedUploadResponse = CommonResponsePhotoPresignedUrlResponseSchema["result"]; +export type PhotoReportUploadResultResponse = CommonResponseVoidSchema["result"]; +export type Cheese4cutFinalizeResponse = CommonResponseVoidSchema["result"]; +export type Cheese4cutPreviewResponse = CommonResponseCheese4cutResponseSchema["result"]; +export type InternalThumbnailCompleteResponse = CommonResponseVoidSchema["result"]; + +// --- Mapping: 'group.fn' -> Response Type +export interface ApiReturns { + "global.health": GlobalHealthResponse; // GET /v1/global/health-check + "auth.exchange": AuthExchangeResponse; // GET /v1/auth/exchange + "auth.logout": AuthLogoutResponse; // POST /v1/auth/logout + "auth.reissue": AuthReissueResponse; // POST /v1/auth/reissue + "user.userMe": UserUserMeResponse; // GET /v1/user/me + "user.userMeName": UserUserMeNameResponse; // PATCH /v1/user/me/name + "user.userMeProfileImage": UserUserMeProfileImageResponse; // PATCH /v1/user/me/profile-image + "user.userOnboarding": UserUserOnboardingResponse; // POST /v1/user/onboarding + "user.userProfileImages": UserUserProfileImagesResponse; // GET /v1/user/profile-images + "album.create": AlbumCreateResponse; // POST /v1/album + "album.availableCount": AlbumAvailableCountResponse; // GET /v1/album/{code}/available-count + "album.albumBest-4cut": AlbumAlbumBest4cutResponse; // GET /v1/album/{code}/best-4cut + "album.enter": AlbumEnterResponse; // POST /v1/album/{code}/enter + "album.albumInfo": AlbumAlbumInfoResponse; // GET /v1/album/{code}/info + "album.invitation": AlbumInvitationResponse; // GET /v1/album/{code}/invitation + "album.participants": AlbumParticipantsResponse; // GET /v1/album/{code}/participants + "album.albumParticipantsMe": AlbumAlbumParticipantsMeResponse; // DELETE /v1/album/{code}/participants/me + "album.albumPhoto": AlbumAlbumPhotoResponse; // DELETE /v1/album/{code}/photo/{photoId} + "album.photos": AlbumPhotosResponse; // GET /v1/album/{code}/photos + "album.photoDetail": AlbumPhotoDetailResponse; // GET /v1/album/{code}/photos/{photoId} + "album.albumPhotosLikers": AlbumAlbumPhotosLikersResponse; // GET /v1/album/{code}/photos/{photoId}/likers + "album.likedPhotos": AlbumLikedPhotosResponse; // GET /v1/album/{code}/photos/liked + "album.albumClosed": AlbumAlbumClosedResponse; // GET /v1/album/closed + "album.albumOpen": AlbumAlbumOpenResponse; // GET /v1/album/open + "album.albumOpenMe": AlbumAlbumOpenMeResponse; // GET /v1/album/open/me + "photo.like": PhotoLikeResponse; // POST /v1/photo/{photoId}/liked + "photo.unlike": PhotoUnlikeResponse; // DELETE /v1/photo/{photoId}/unliked + "photo.presignedDownload": PhotoPresignedDownloadResponse; // POST /v1/photo/download-url + "photo.presignedUpload": PhotoPresignedUploadResponse; // POST /v1/photo/presigned-url + "photo.reportUploadResult": PhotoReportUploadResultResponse; // POST /v1/photo/report + "cheese4cut.finalize": Cheese4cutFinalizeResponse; // POST /v1/cheese4cut/{code}/fixed + "cheese4cut.preview": Cheese4cutPreviewResponse; // GET /v1/cheese4cut/{code}/preview + "internal.thumbnailComplete": InternalThumbnailCompleteResponse; // POST /internal/thumbnail/complete +} diff --git a/src/global/api/getPresignedUrl.ts b/src/global/api/getPresignedUrl.ts new file mode 100644 index 00000000..64bdc698 --- /dev/null +++ b/src/global/api/getPresignedUrl.ts @@ -0,0 +1,48 @@ +import { api } from '@/global/utils/api'; +import Toast from '../components/toast/Toast'; + +export type PresignedUrlRequest = { + albumCode: string; + fileInfos: { + fileName: string; + fileSize: number; + contentType: string; + captureTime: string; // yyyy-MM-ddTHH:mm:ss + }[]; +}; + +export type PresignedUrlInfo = { + photoId: number; + uploadUrl: string; +}; + +type ApiResponse = { + isSuccess: boolean; + code: number; + message: string; + presignedUrlInfos: PresignedUrlInfo[]; +}; + +export async function getPresignedUrl( + params: PresignedUrlRequest, +): Promise { + try { + const response = await api.post({ + path: '/v1/photo/presigned-url', + body: { albumCode: params.albumCode, fileInfos: params.fileInfos }, + }); + return response.result.presignedUrlInfos; + } catch (error: unknown) { + console.error('Presigned URL 조회 실패:', error); + + if (error && typeof error === 'object' && 'message' in error) { + Toast.alert(error.message as string); + } else if (error instanceof Error) { + Toast.alert(error.message); + } else { + Toast.alert('오류가 발생했습니다.'); + } + + throw error; + } +} diff --git a/src/global/api/presignedAndUploadToNCP.ts b/src/global/api/presignedAndUploadToNCP.ts new file mode 100644 index 00000000..a5d7edac --- /dev/null +++ b/src/global/api/presignedAndUploadToNCP.ts @@ -0,0 +1,29 @@ +import { + getPresignedUrl, + PresignedUrlRequest, +} from '@/global/api/getPresignedUrl'; +import { uploadFilesToNCP } from '@/global/api/uploadToNCP'; +import { reportFailedPhotoIds } from '@/global/hooks/useReportFailed'; + +/** + * Presigned URL을 발급받고, 해당 URL로 파일을 NCP에 업로드까지 처리하는 통합 API + * @param params PresignedUrlRequest & { files: File[] } + * @returns 업로드 결과 (성공 개수, 실패 개수, 실패 photoId) + */ +export async function presignedAndUploadToNCP( + params: PresignedUrlRequest & { files: File[] }, +): Promise<{ success: number; failed: number; failedPhotoIds: number[] }> { + const presignedUrlInfos = await getPresignedUrl(params); + + const uploadResult = await uploadFilesToNCP(params.files, presignedUrlInfos); + + if (uploadResult.failedPhotoIds.length > 0) { + try { + await reportFailedPhotoIds(uploadResult.failedPhotoIds); + } catch (e) { + console.error('실패 보고 API 호출 실패:', e); + } + } + + return uploadResult; +} diff --git a/src/global/api/uploadToNCP.ts b/src/global/api/uploadToNCP.ts new file mode 100644 index 00000000..a1e5e098 --- /dev/null +++ b/src/global/api/uploadToNCP.ts @@ -0,0 +1,72 @@ +import { PresignedUrlInfo } from '@/global/api/getPresignedUrl'; + +/** + * Presigned URL을 사용하여 NCP에 파일 업로드 + * @param file 업로드할 파일 + * @param uploadUrl NCP presigned URL + * @param photoId 사진 ID + * @returns 업로드 성공 여부 + */ +export async function uploadFileToNCP( + file: File, + uploadUrl: string, + photoId: number, +): Promise { + try { + const response = await fetch(uploadUrl, { + method: 'PUT', + body: file, + }); + + if (!response.ok) { + console.error( + `파일 업로드 실패 (photoId: ${photoId}):`, + response.statusText, + ); + return false; + } + + return true; + } catch (error) { + console.error(`파일 업로드 중 오류 발생 (photoId: ${photoId}):`, error); + return false; + } +} + +/** + * 여러 파일을 NCP에 병렬 업로드 + * @param files 업로드할 파일 배열 + * @param presignedUrlInfos presigned URL 정보 배열 + * @returns 업로드 결과 (성공 개수, 실패 개수) + */ +export async function uploadFilesToNCP( + files: File[], + presignedUrlInfos: PresignedUrlInfo[], +): Promise<{ success: number; failed: number; failedPhotoIds: number[] }> { + if (files.length !== presignedUrlInfos.length) { + console.error('파일 개수와 presigned URL 개수가 일치하지 않습니다.'); + return { + success: 0, + failed: files.length, + failedPhotoIds: presignedUrlInfos.map((i) => i.photoId), + }; + } + + const uploadResults = await Promise.all( + files.map((file, index) => { + const { uploadUrl, photoId } = presignedUrlInfos[index]; + return uploadFileToNCP(file, uploadUrl, photoId).then((ok) => ({ + ok, + photoId, + })); + }), + ); + + const success = uploadResults.filter((r) => r.ok).length; + const failed = uploadResults.length - success; + const failedPhotoIds = uploadResults + .filter((r) => !r.ok) + .map((r) => r.photoId); + + return { success, failed, failedPhotoIds }; +} diff --git a/src/global/components/CountdownTimer.tsx b/src/global/components/CountdownTimer.tsx new file mode 100644 index 00000000..cf71214e --- /dev/null +++ b/src/global/components/CountdownTimer.tsx @@ -0,0 +1,165 @@ +'use client'; +import { AnimatePresence, LazyMotion, domAnimation, m } from 'framer-motion'; +import { useEffect, useState } from 'react'; + +interface TimeLeft { + days: number; + hours: number; + minutes: number; + seconds: number; +} + +interface AnimatedNumberProps { + number: number; + label: string; +} + +interface CountdownTimerProps { + albumId: string; +} + +function calculateTimeLeft(targetDate: Date): TimeLeft { + const difference = +new Date(targetDate) - +new Date(); + let timeLeft: TimeLeft = { + days: 0, + hours: 0, + minutes: 0, + seconds: 0, + }; + + if (difference > 0) { + timeLeft = { + days: Math.floor(difference / (1000 * 60 * 60 * 24)), + hours: Math.floor((difference / (1000 * 60 * 60)) % 24), + minutes: Math.floor((difference / 1000 / 60) % 60), + seconds: Math.floor((difference / 1000) % 60), + }; + } + + return timeLeft; +} + +function AnimatedNumber({ number, label }: AnimatedNumberProps) { + const formattedNumber = String(number).padStart(2, '0'); + + return ( +
+
+
+ + + {formattedNumber} + + +
+
+ + {label} + +
+ ); +} + +export function CountdownTimer({ albumId }: CountdownTimerProps) { + const [targetDate, setTargetDate] = useState(null); + const [timeLeft, setTimeLeft] = useState({ + days: 0, + hours: 0, + minutes: 0, + seconds: 0, + }); + + useEffect(() => { + const storageKey = `album_${albumId}_created_date`; + + const storedDate = localStorage.getItem(storageKey); + + let calculatedDate: Date; + + if (storedDate) { + calculatedDate = new Date(storedDate); + } else { + calculatedDate = new Date(Date.now() + 7 * 24 * 60 * 60 * 1000); + localStorage.setItem(storageKey, calculatedDate.toISOString()); + } + + setTargetDate(calculatedDate); + setTimeLeft(calculateTimeLeft(calculatedDate)); + }, [albumId]); + + useEffect(() => { + if (!targetDate) return; + + const timer = setInterval(() => { + const newTimeLeft = calculateTimeLeft(targetDate); + setTimeLeft(newTimeLeft); + + if ( + newTimeLeft.days === 0 && + newTimeLeft.hours === 0 && + newTimeLeft.minutes === 0 && + newTimeLeft.seconds === 0 + ) { + clearInterval(timer); + } + }, 1000); + + return () => clearInterval(timer); + }, [targetDate]); + + // 레이아웃 시프트 방지를 위해 초기 렌더링 시에도 공간 확보 + if (!targetDate) { + return ( +
+
+
+ + Days + +
+ : +
+
+ + HOURS + +
+ : +
+
+ + Mins + +
+ : +
+
+ + Secs + +
+
+ ); + } + + return ( + +
+ + : + + : + + : + +
+
+ ); +} diff --git a/src/global/components/DateXInput.tsx b/src/global/components/DateXInput.tsx new file mode 100644 index 00000000..8cd0982c --- /dev/null +++ b/src/global/components/DateXInput.tsx @@ -0,0 +1,104 @@ +'use client'; +import { Calendar } from '@/components/ui/calendar'; +import { + Popover, + PopoverContent, + PopoverTrigger, +} from '@/components/ui/popover'; +import { cn } from '@/lib/utils'; +import { format } from 'date-fns'; +import { CalendarIcon } from 'lucide-react'; +import * as React from 'react'; + +interface DateXInputProps { + label?: string; + value: string; + onChange: (value: string) => void; + error?: string; + helperText?: string; + min?: string; + max?: string; + placeholder?: string; + disabled?: boolean; + className?: string; +} + +export default function DateXInput({ + label, + value, + onChange, + error, + helperText, + min, + max, + placeholder = 'YYYY-MM-DD', + disabled, + className, +}: DateXInputProps) { + const [open, setOpen] = React.useState(false); + const parsedDate = + value && value !== '' && value !== placeholder + ? new Date(value) + : undefined; + + const minDate = min ? new Date(min) : undefined; + const maxDate = max ? new Date(max) : undefined; + + return ( +
+
+ {label && ( +
+ {label} +
+ )} + + + + + + { + if (date) { + onChange(format(date, 'yyyy-MM-dd')); + setOpen(false); + } + }} + fromDate={minDate} + toDate={maxDate} + /> + + + {(error || helperText) && ( +
+ {error || helperText} +
+ )} +
+
+ ); +} diff --git a/src/global/components/LongButton.tsx b/src/global/components/LongButton.tsx new file mode 100644 index 00000000..ebe8cc41 --- /dev/null +++ b/src/global/components/LongButton.tsx @@ -0,0 +1,53 @@ +interface LongButtonProps { + text: string; + disabled?: boolean; + onClick?: () => void; + sideGap?: number; // 좌우 여백(px) + bottomGap?: number; // 하단 여백(px) + noFixed?: boolean; // true면 fixed 해제하고 가로 100% 채움 + height?: number; // 버튼 세로(px) + safeArea?: boolean; // false면 safe-area-inset-bottom 미적용 +} + +export default function LongButton({ + text, + disabled = false, + onClick, + sideGap = 16, + bottomGap = 20, + noFixed = false, + height, + safeArea = true, +}: LongButtonProps) { + // noFixed가 true면 스타일 속성 없음, false면 left/right/bottom 값 적용 + const buttonStyle = { + ...(noFixed + ? {} + : { + left: `${sideGap}px`, + right: `${sideGap}px`, + bottom: safeArea + ? `calc(${bottomGap}px + env(safe-area-inset-bottom))` + : `${bottomGap}px`, + }), + ...(height ? { height: `${height}px` } : {}), + }; + + return ( + + ); +} diff --git a/src/global/components/Spinner.tsx b/src/global/components/Spinner.tsx new file mode 100644 index 00000000..1c6cd964 --- /dev/null +++ b/src/global/components/Spinner.tsx @@ -0,0 +1,18 @@ +export default function Spinner() { + return ( +
+ + + +
+ ); +} diff --git a/src/global/components/XInput.tsx b/src/global/components/XInput.tsx new file mode 100644 index 00000000..0608988b --- /dev/null +++ b/src/global/components/XInput.tsx @@ -0,0 +1,129 @@ +'use client'; +import { X } from 'lucide-react'; +import React, { InputHTMLAttributes, useRef, useState } from 'react'; + +interface InputProps + extends Omit, 'onChange'> { + /** 인풋 상단 라벨 텍스트 */ + label?: string; + /** 인풋 값 (항상 string) */ + value: string; + /** 값 변경 시 호출되는 콜백 (string) */ + onChange: (value: string) => void; + /** 에러 메시지 (있으면 하단에 표시) */ + error?: string; + /** 하단 서브 텍스트(설명, 안내 등) */ + helperText?: string; + /** X 버튼(입력 내용 지우기) 노출 여부, false면 아예 안보임 (기본값 true) */ + showClear?: boolean; + /** true면 입력값 없어도 X 버튼 항상 노출, false면 입력값 있을 때만 노출 (기본값 false) */ + showClearAlways?: boolean; +} + +export default function XInput({ + label, + value, + onChange, + error, + helperText, + showClear = true, + showClearAlways = false, + className, + disabled, + maxLength, + type, + ...restProps +}: InputProps) { + const [isFocused, setIsFocused] = useState(false); + const inputRef = useRef(null); + + const handleInputChange = (e: React.ChangeEvent) => { + onChange(e.target.value); + }; + + const handleClear = () => { + onChange(''); + inputRef.current?.focus(); + }; + + const handleInputClick = () => { + if (type === 'date' && inputRef.current) { + inputRef.current.showPicker?.(); + } + }; + + const shouldShowClear = + showClear && isFocused && !disabled && (showClearAlways || value); + + return ( +
+
+ {label && ( +
+ {label} +
+ )} + +
+ setIsFocused(true)} + onBlur={() => setIsFocused(false)} + disabled={disabled} + maxLength={maxLength} + className={`bg-element-gray-lighter typo-body-lg-medium text-text-basic placeholder:text-text-subtler w-full rounded-[8px] p-4 disabled:cursor-not-allowed disabled:opacity-50 ${ + error + ? 'outline-text-error outline-1' + : 'focus:outline-border-primary focus:outline-1' + } ${shouldShowClear ? 'pr-12' : ''} ${ + type === 'date' ? 'cursor-pointer' : '' + } ${ + type === 'number' + ? '[appearance:textfield] [&::-webkit-inner-spin-button]:appearance-none [&::-webkit-outer-spin-button]:appearance-none' + : '' + }`} + style={ + type === 'date' && !value + ? { + color: '#8E9398', + } + : type === 'date' + ? { + color: '#18191B', + } + : undefined + } + {...restProps} + /> + + {shouldShowClear && ( + + )} +
+ + {(error || helperText) && ( +
+ {error || helperText} +
+ )} +
+
+ ); +} diff --git a/src/global/components/carousel/MarqueeCarousel.tsx b/src/global/components/carousel/MarqueeCarousel.tsx new file mode 100644 index 00000000..10e5fab0 --- /dev/null +++ b/src/global/components/carousel/MarqueeCarousel.tsx @@ -0,0 +1,40 @@ +'use client'; + +type MarqueeCarouselProps = { + items: React.ReactNode[]; + itemWidth: number; + gap?: number; + speed?: number; // px/s + className?: string; +}; + +export default function MarqueeCarousel({ + items, + itemWidth, + gap = 16, + speed = 80, + className = '', +}: MarqueeCarouselProps) { + const trackWidth = (itemWidth + gap) * items.length; + + const duration = trackWidth / speed; + + return ( +
+
+ {[...items, ...items].map((item, i) => ( +
+ {item} +
+ ))} +
+
+ ); +} diff --git a/src/global/components/header/CloseButton.tsx b/src/global/components/header/CloseButton.tsx new file mode 100644 index 00000000..f17daa17 --- /dev/null +++ b/src/global/components/header/CloseButton.tsx @@ -0,0 +1,19 @@ +'use client'; +import { X } from 'lucide-react'; +import { useRouter } from 'next/navigation'; + +interface CloseButtonProps {} + +export default function CloseButton({}: CloseButtonProps) { + const router = useRouter(); + + const handleClick = () => { + router.back(); + }; + + return ( + + ); +} diff --git a/src/global/components/header/CustomHeader.tsx b/src/global/components/header/CustomHeader.tsx new file mode 100644 index 00000000..96743756 --- /dev/null +++ b/src/global/components/header/CustomHeader.tsx @@ -0,0 +1,130 @@ +/** + * CustomHeader Component + * + * 재사용 가능한 커스텀 헤더 컴포넌트 + * - 왼쪽: 뒤로가기 버튼 + 타이틀 + * - 오른쪽: 커스텀 컴포넌트 (단일 또는 배열) + * + * @example + * // 기본 사용 (타이틀만) + * + * + * @example + * // 오른쪽에 버튼 1개 + * 저장 + * } + * /> + * + * @example + * // 오른쪽에 여러 버튼 (배열) + * 취소, + * , + * ]} + * /> + * + * @example + * // Fragment 방식 + * + * + * + * + * } + * /> + * + * @example + * // 커스텀 뒤로가기 핸들러 + * router.push('/home')} + * rightContent={} + * /> + */ + +'use client'; +import { ChevronLeft } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import { ReactNode } from 'react'; + +export const HEADER_HEIGHT = 72; + +interface CustomHeaderProps { + /** 헤더 중앙에 표시될 타이틀 */ + title: string; + /** 헤더 오른쪽에 표시될 컴포넌트들 (16px gap으로 배치) - 배열 또는 단일 컴포넌트 */ + rightContent?: ReactNode | ReactNode[]; + /** 뒤로가기 버튼 클릭 핸들러 (기본: router.back()) */ + onBackClick?: () => void; + isShowBack?: boolean; + border?: boolean; + /** 헤더를 완전히 숨길지 여부 */ + isHidden?: boolean; +} + +export default function CustomHeader({ + title, + rightContent, + onBackClick, + isShowBack, + border = true, + isHidden = false, +}: CustomHeaderProps) { + const router = useRouter(); + + function handleBackClick() { + if (onBackClick) { + onBackClick(); + } else { + router.back(); + } + } + + // 숨김 처리 + if (isHidden) return null; + + return ( + <> +
+ {/* 왼쪽: 뒤로가기 + 타이틀 */} +
+ {isShowBack && ( + + )} + + {title} +
+ + {/* 오른쪽: 커스텀 컴포넌트 */} + {rightContent && ( +
{rightContent}
+ )} +
+ {/* 헤더로 인해 가려지는 영역이 없도록 아래 요소 추가 */} +
+ + ); +} diff --git a/src/global/components/header/LogoHeader.tsx b/src/global/components/header/LogoHeader.tsx new file mode 100644 index 00000000..b2b8601d --- /dev/null +++ b/src/global/components/header/LogoHeader.tsx @@ -0,0 +1,51 @@ +'use client'; +import { useCheckAuth } from '@/global/hooks/useCheckAuth'; +import Link from 'next/link'; +import { useCallback } from 'react'; +import SvgLogo from './svg/SvgLogo'; + +interface LogoHeaderProps { + showLogin?: boolean; + bgColor?: string; + border?: boolean; +} + +export default function LogoHeader({ + showLogin = true, + bgColor = 'white', + border = false, +}: LogoHeaderProps) { + const { isAuthed } = useCheckAuth(); + + const handleLoginClick = useCallback(() => { + if (typeof document !== 'undefined') { + document.cookie = 'entry=main; path=/;'; + } + }, []); + const shouldShowLogin = showLogin && !isAuthed; + + return ( + <> +
+
+ + {shouldShowLogin && ( + +
+ + 로그인 + +
+ + )} +
+
+ + {/* 헤더로인해 가려지는 영역 방지 */} +
+ + ); +} diff --git a/src/global/components/header/svg/SvgLogo.tsx b/src/global/components/header/svg/SvgLogo.tsx new file mode 100644 index 00000000..c23a990f --- /dev/null +++ b/src/global/components/header/svg/SvgLogo.tsx @@ -0,0 +1,38 @@ +interface SvgLogoProps {} + +export default function SvgLogo({}: SvgLogoProps) { + return ( + + + + + + + ); +} diff --git a/src/global/components/modal/BottomSheetModal.tsx b/src/global/components/modal/BottomSheetModal.tsx new file mode 100644 index 00000000..63d659f1 --- /dev/null +++ b/src/global/components/modal/BottomSheetModal.tsx @@ -0,0 +1,84 @@ +'use client'; + +import { + Drawer, + DrawerClose, + DrawerContent, + DrawerTitle, + DrawerTrigger, +} from '@/components/ui/drawer'; +import { cn } from '@/lib/utils'; +import React from 'react'; + +type BottomSheetModalProps = { + /** 트리거 버튼/노드 (예: ) */ + trigger: React.ReactNode; + /** 모달 className (높이, 너비, 스타일 등) */ + className?: string; + /** 모달 내용 */ + children: React.ReactNode; + /** 닫기 버튼 표시 여부 */ + showCloseButton?: boolean; + /** 모달 제목 (화면에 표시, 없으면 스크린리더용만 사용) */ + title?: string; + /** 드래그/스와이프 등으로 닫기 허용 여부 (기본값: true) */ + dismissible?: boolean; + /** 상단 드래그 바 표시 여부 (기본값: true) */ + showHandle?: boolean; + /** 외부에서 열림 상태를 제어하고 싶을 때 사용 */ + open?: boolean; + defaultOpen?: boolean; + onOpenChange?: (open: boolean) => void; +}; + +export default function BottomSheetModal({ + trigger, + children, + className = 'max-h-[80vh] w-full', + showCloseButton = false, + title, + dismissible = true, + showHandle = true, + open, + defaultOpen, + onOpenChange, +}: BottomSheetModalProps) { + return ( + + {trigger} + + +
+ {/* 스크린리더용 제목 (항상 필요) */} + + {title || '모달'} + + + {showCloseButton && ( +
+ + ✕ + +
+ )} + {children} +
+
+
+ ); +} diff --git a/src/global/components/modal/ConfirmModal.tsx b/src/global/components/modal/ConfirmModal.tsx new file mode 100644 index 00000000..a4d255dd --- /dev/null +++ b/src/global/components/modal/ConfirmModal.tsx @@ -0,0 +1,128 @@ +'use client'; + +import { + AlertDialog, + AlertDialogAction, + AlertDialogCancel, + AlertDialogContent, + AlertDialogDescription, + AlertDialogFooter, + AlertDialogHeader, + AlertDialogTitle, + AlertDialogTrigger, +} from '@/components/ui/alert-dialog'; +import { cn } from '@/lib/utils'; +import { useCallback, useEffect } from 'react'; + +type ConfirmModalProps = { + /** 트리거 버튼/노드 (예: ) */ + trigger: React.ReactNode; + /** 모달 제목 */ + title: string | React.ReactNode; + /** 모달 본문 */ + description?: string | React.ReactNode; + /** 취소 버튼 라벨 */ + cancelText?: string; + /** 확인 버튼 라벨 */ + confirmText?: string; + /** 취소 클릭 핸들러 */ + onCancel?: () => void; + /** 확인 클릭 핸들러 (Promise 허용) */ + onConfirm?: () => Promise | void; + /** 버튼 스타일을 커스터마이즈하고 싶다면 className 전달(선택) */ + cancelClassName?: string; + confirmClassName?: string; +}; + +export default function ConfirmModal({ + trigger, + title, + description, + cancelText = '취소', + confirmText = '확인', + onCancel, + onConfirm, + cancelClassName = '', + confirmClassName = '', +}: ConfirmModalProps) { + const handleConfirm = useCallback(async () => { + try { + await onConfirm?.(); + } catch (e) { + console.error(e); + } + }, [onConfirm]); + + const handleCancel = useCallback(() => { + onCancel?.(); + }, [onCancel]); + + useEffect(() => { + // data-scroll-locked 속성 제거 + const removeScrollLock = () => { + const body = document.body; + if (body.hasAttribute('data-scroll-locked')) { + body.removeAttribute('data-scroll-locked'); + } + }; + + removeScrollLock(); + const interval = setInterval(removeScrollLock, 100); + + return () => { + clearInterval(interval); + removeScrollLock(); + }; + }, []); + + return ( + + {trigger} + + + + + {title} + + {description ? ( + + {description} + + ) : null} + + + + + {cancelText} + + + + {confirmText} + + + + + ); +} diff --git a/src/global/components/photo/PhotoBox.tsx b/src/global/components/photo/PhotoBox.tsx new file mode 100644 index 00000000..46d9b67c --- /dev/null +++ b/src/global/components/photo/PhotoBox.tsx @@ -0,0 +1,142 @@ +'use client'; +import { cn } from '@/lib/utils'; +import { Check, Heart } from 'lucide-react'; +import { useEffect, useState } from 'react'; + +const FALLBACK_SRC = '/icon/error-image.svg'; + +interface PhotoBoxProps { + size?: number; // px + /** 부모 컨테이너 크기를 따라가기 여부 */ + responsive?: boolean; + likeCount?: number; + liked?: boolean; + mode?: 'default' | 'select'; + downloaded?: boolean; + pressed?: boolean; + disabled?: boolean; + imageSrc?: string; + imageAlt?: string; + onPress?: (pressed: boolean) => void; + onDisabledPress?: () => void; + pressable?: boolean; +} + +export default function PhotoBox({ + size = 82, + responsive = false, + likeCount, + liked = false, + downloaded = false, + pressed = false, + mode = 'default', + disabled = false, + imageSrc, + imageAlt = '사진', + onPress, + onDisabledPress, + pressable = true, +}: PhotoBoxProps) { + const showLike = likeCount !== undefined; + const [currentSrc, setCurrentSrc] = useState(imageSrc ?? FALLBACK_SRC); + + useEffect(() => { + setCurrentSrc(imageSrc ?? FALLBACK_SRC); + }, [imageSrc]); + + const handleImageError = (): void => { + if (currentSrc === FALLBACK_SRC) return; + setCurrentSrc(FALLBACK_SRC); + }; + + const handlePress = () => { + if (disabled) { + onDisabledPress?.(); + return; + } + + if (!pressable) return; + + const next = !pressed; + onPress?.(next); + }; + + return ( + + ); +} diff --git a/src/global/components/portal/BodyPortal.tsx b/src/global/components/portal/BodyPortal.tsx new file mode 100644 index 00000000..4deb590a --- /dev/null +++ b/src/global/components/portal/BodyPortal.tsx @@ -0,0 +1,19 @@ +'use client'; + +import type { ReactNode, ReactPortal } from 'react'; +import { createPortal } from 'react-dom'; + +interface BodyPortalProps { + children: ReactNode; + container?: Element; +} + +/** Render children into a DOM container (defaults to document.body) */ +export function BodyPortal({ + children, + container, +}: BodyPortalProps): ReactPortal | null { + if (typeof document === 'undefined') return null; + + return createPortal(children, container ?? document.body); +} diff --git a/src/global/components/toast/AlbumToast.tsx b/src/global/components/toast/AlbumToast.tsx new file mode 100644 index 00000000..cf91ca15 --- /dev/null +++ b/src/global/components/toast/AlbumToast.tsx @@ -0,0 +1,100 @@ +import { AlertCircle } from 'lucide-react'; +import { useEffect, useState } from 'react'; + +type AlbumToastProps = { + message: string; + style?: React.CSSProperties; + onDismiss?: () => void; + delay?: number; // 사라지기 시작하는 시간에 추가할 delay +}; + +export default function AlbumToast({ + message, + style, + onDismiss, + delay = 0, +}: AlbumToastProps) { + const bottomPx = style?.bottom ?? 88; + const [visible, setVisible] = useState(true); + const [shouldRender, setShouldRender] = useState(true); + + useEffect(() => { + const timer = setTimeout(() => { + setVisible(false); + // fade-out 트랜지션 후 실제로 unmount + setTimeout(() => { + setShouldRender(false); + onDismiss?.(); + }, 400); // 트랜지션 시간과 맞춤 + }, 4000 + delay); + return () => clearTimeout(timer); + }, [onDismiss, delay]); + + if (!shouldRender) return null; + + return ( +
+
+ + {message} +
+
+ ); +} + +export function AlbumToastList({ + toasts, + onRemove, +}: { + toasts: string[]; + onRemove?: (message: string) => void; +}) { + const TOAST_HEIGHT = 48; // px + const GAP = 12; // px + const [visibleToasts, setVisibleToasts] = useState< + Array<{ id: number; message: string }> + >([]); + + useEffect(() => { + // 새로운 토스트가 추가되면 기존 것에 추가 (중복 방지) + setVisibleToasts((prev) => { + const newToasts = toasts + .filter((msg) => !prev.some((t) => t.message === msg)) + .map((message) => ({ + id: Date.now() + Math.random(), + message, + })); + return newToasts.length > 0 ? [...prev, ...newToasts] : prev; + }); + }, [toasts]); + + const handleDismiss = (id: number, message: string) => { + setVisibleToasts((prev) => prev.filter((toast) => toast.id !== id)); + onRemove?.(message); + }; + + return ( + <> + {visibleToasts.map((toast, idx) => ( + handleDismiss(toast.id, toast.message)} + /> + ))} + + ); +} diff --git a/src/global/components/toast/Toast.tsx b/src/global/components/toast/Toast.tsx new file mode 100644 index 00000000..3698598b --- /dev/null +++ b/src/global/components/toast/Toast.tsx @@ -0,0 +1,74 @@ +import { CircleCheck } from 'lucide-react'; +import React from 'react'; +import { createRoot, Root } from 'react-dom/client'; + +const TOAST_CONTAINER_ID = '__toast-container'; + +// HTMLElement에 커스텀 프로퍼티 하나 더 붙여서 root 저장 +interface ToastContainerElement extends HTMLElement { + __root?: Root; +} + +const getContainer = (id: string): ToastContainerElement => { + let toastContainer = document.getElementById( + id, + ) as ToastContainerElement | null; + + if (!toastContainer) { + toastContainer = document.createElement('div') as ToastContainerElement; + toastContainer.id = id; + toastContainer.style.display = 'flex'; + toastContainer.style.justifyContent = 'center'; + document.body.appendChild(toastContainer); + } + + return toastContainer; +}; + +const renderToast = ( + component: React.ReactNode, + container: ToastContainerElement, +): void => { + // 이미 root가 있으면 그거 쓰고, 없으면 새로 만든다 + if (!container.__root) { + container.__root = createRoot(container); + } + + container.__root.render(component); +}; + +const unmountToast = ( + toastContainer: ToastContainerElement, + ms = 3000, +): void => { + setTimeout(() => { + // 만들어둔 root가 있으면 그걸로 언마운트 + toastContainer.__root?.unmount(); + toastContainer.remove(); + }, ms); +}; + +const Toast = { + alert: async (message: string): Promise => { + const toastContainer = getContainer(TOAST_CONTAINER_ID); + const { default: ToastView } = await import('./ToastView'); + + renderToast(, toastContainer); + unmountToast(toastContainer); + }, + check: async (message: string): Promise => { + const toastContainer = getContainer(TOAST_CONTAINER_ID); + const { default: ToastView } = await import('./ToastView'); + + renderToast( + } + />, + toastContainer, + ); + unmountToast(toastContainer); + }, +}; + +export default Toast; diff --git a/src/global/components/toast/ToastView.tsx b/src/global/components/toast/ToastView.tsx new file mode 100644 index 00000000..3f109960 --- /dev/null +++ b/src/global/components/toast/ToastView.tsx @@ -0,0 +1,18 @@ +import { AlertCircle } from 'lucide-react'; + +interface ToastViewProps { + lucideIcon?: React.ReactNode; + message: string; +} + +export default function ToastView({ lucideIcon, message }: ToastViewProps) { + return ( +
+
+ {lucideIcon ?? } + + {message} +
+
+ ); +} diff --git a/src/global/components/tooltip/BubbleTooltip.tsx b/src/global/components/tooltip/BubbleTooltip.tsx new file mode 100644 index 00000000..c152c7a7 --- /dev/null +++ b/src/global/components/tooltip/BubbleTooltip.tsx @@ -0,0 +1,35 @@ +import { cn } from '@/lib/utils'; + +interface BubbleTooltipProps { + /** 텍스트나 아이콘이 함께 포함된 문장 */ + message: string; + /** 기본은 가운데 정렬 */ + align?: 'center' | 'left' | 'right'; + className?: string; +} + +export default function BubbleTooltip({ + message, + align = 'center', + className = '', +}: BubbleTooltipProps) { + const alignClass = + align === 'center' ? 'mx-auto' : align === 'left' ? 'mr-auto' : 'ml-auto'; + + return ( +
+
+
+ {message} +
+ + {/* 꼬다리 */} +
+
+
+
+
+ ); +} diff --git a/src/global/constants/cookies.ts b/src/global/constants/cookies.ts new file mode 100644 index 00000000..d5327f47 --- /dev/null +++ b/src/global/constants/cookies.ts @@ -0,0 +1,2 @@ +export const ACCESS_TOKEN_KEY = 'ACCESS_TOKEN'; +export const REFRESH_TOKEN_KEY = 'REFRESH_TOKEN'; diff --git a/src/global/constants/images.ts b/src/global/constants/images.ts new file mode 100644 index 00000000..83f650cc --- /dev/null +++ b/src/global/constants/images.ts @@ -0,0 +1,2 @@ +export const DEFAULT_PROFILE_IMAGE = + 'https://say-cheese-profile.edge.naverncp.com/profile/sign_up_profile_1.jpg'; diff --git a/src/global/context/KakaoProvider.tsx b/src/global/context/KakaoProvider.tsx new file mode 100644 index 00000000..73c1695a --- /dev/null +++ b/src/global/context/KakaoProvider.tsx @@ -0,0 +1,29 @@ +'use client'; + +import Script from 'next/script'; + +interface KakaoProviderProps { + children: React.ReactNode; +} + +export default function KakaoProvider({ children }: KakaoProviderProps) { + const kakaoKey = process.env.NEXT_PUBLIC_KAKAO_JS_KEY || ''; + + return ( + <> +