diff --git a/.eslintrc.cjs b/.eslintrc.cjs index 1978302..6582a95 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -13,7 +13,7 @@ module.exports = { parserOptions: { project: "./tsconfig.json", }, - ignorePatterns: ["dist", ".eslintrc.cjs"], + ignorePatterns: ["dist", ".eslintrc.cjs", "firebase-messaging-sw.js"], parser: "@typescript-eslint/parser", plugins: [ "react-refresh", @@ -36,7 +36,7 @@ module.exports = { "react/jsx-props-no-spreading": "off", "@typescript-eslint/no-explicit-any": "off", "@typescript-eslint/naming-convention": "off", - '@typescript-eslint/no-unused-vars': 'off', + "@typescript-eslint/no-unused-vars": "off", "jsx-a11y/click-events-have-key-events": "off", "no-underscore-dangle": "off", "no-alert": "off", diff --git a/.gitignore b/.gitignore index bc6d306..0b03a2d 100644 --- a/.gitignore +++ b/.gitignore @@ -23,4 +23,6 @@ dist-ssr *.sln *.sw? -*storybook.log \ No newline at end of file +*storybook.log + +.env .development \ No newline at end of file diff --git a/firebase-messaging-sw.js b/firebase-messaging-sw.js new file mode 100644 index 0000000..f7edbb9 --- /dev/null +++ b/firebase-messaging-sw.js @@ -0,0 +1,27 @@ +importScripts( + "https://www.gstatic.com/firebasejs/9.6.1/firebase-messaging-compat.js", +); + +firebase.initializeApp({ + apiKey: "AIzaSyAInigygScRLDilnWcnArBN8LMbQRpDZVk", + authDomain: "splanet-cef14.firebaseapp.com", + projectId: "splanet-cef14", + storageBucket: "splanet-cef14.appspot.com", + messagingSenderId: "995362943401", + appId: "1:995362943401:web:cef434d0e3f51d31a4d4b8", + measurementId: "G-LZJKRYBSJV", +}); + +const messaging = firebase.messaging(); + +// 알림 수신 대기 +messaging.onBackgroundMessage((payload) => { + console.log("Received background message ", payload); + // Customize notification + const notificationTitle = payload.notification.title; + const notificationOptions = { + body: payload.notification.body, + }; + + self.registration.showNotification(notificationTitle, notificationOptions); +}); diff --git a/package-lock.json b/package-lock.json index 8c0a6da..6271adc 100644 --- a/package-lock.json +++ b/package-lock.json @@ -20,7 +20,9 @@ "@mui/material": "^6.1.1", "@tanstack/react-query": "^5.56.2", "axios": "^1.7.7", + "date-fns": "^4.1.0", "events": "^3.3.0", + "firebase": "^11.0.1", "framer-motion": "^11.9.0", "js-cookie": "^3.0.5", "lucide-react": "^0.456.0", @@ -30,6 +32,7 @@ "postcss-import": "^16.1.0", "react": "^18.2.0", "react-cookie": "^7.2.2", + "react-datepicker": "^7.5.0", "react-dnd": "^16.0.1", "react-dnd-html5-backend": "^16.0.1", "react-dom": "^18.2.0", @@ -1082,6 +1085,665 @@ "node": "^12.22.0 || ^14.17.0 || >=16.0.0" } }, + "node_modules/@firebase/analytics": { + "version": "0.10.9", + "resolved": "https://registry.npmjs.org/@firebase/analytics/-/analytics-0.10.9.tgz", + "integrity": "sha512-FrvW6u6xDBKXUGYUy1WIUh0J9tvbppMsk90mig0JhHST8iLveKu/dIBVeVE/ZYZhmXy4fkI7SPSWvD1V0O4tXw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/installations": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/analytics-compat": { + "version": "0.2.15", + "resolved": "https://registry.npmjs.org/@firebase/analytics-compat/-/analytics-compat-0.2.15.tgz", + "integrity": "sha512-C5to422Sr8FkL0MPwXcIecbMnF4o2Ll7MtoWvIm4Q/LPJvvM+tWa1DiU+LzsCdsd1/CYE9EIW9Ma3ko9XnAAYw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/analytics": "0.10.9", + "@firebase/analytics-types": "0.8.2", + "@firebase/component": "0.6.10", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/analytics-types": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/@firebase/analytics-types/-/analytics-types-0.8.2.tgz", + "integrity": "sha512-EnzNNLh+9/sJsimsA/FGqzakmrAUKLeJvjRHlg8df1f97NLUlFidk9600y0ZgWOp3CAxn6Hjtk+08tixlUOWyw==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/app": { + "version": "0.10.15", + "resolved": "https://registry.npmjs.org/@firebase/app/-/app-0.10.15.tgz", + "integrity": "sha512-he6qlG3pmwL+LHdG/BrSMBQeJzzutciq4fpXN3lGa1uSwYSijJ24VtakS/bP2X9SiDf8jGywJ4u+OgXAenJsNg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "idb": "7.1.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/app-check": { + "version": "0.8.9", + "resolved": "https://registry.npmjs.org/@firebase/app-check/-/app-check-0.8.9.tgz", + "integrity": "sha512-YzVn1mMLzD2JboMPVVO0Pe20YOgWzrF+aXoAmmd0v3xec051n83YpxSUZbacL69uYvk0dHrEsbea44QtQ5WPDA==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/app-check-compat": { + "version": "0.3.16", + "resolved": "https://registry.npmjs.org/@firebase/app-check-compat/-/app-check-compat-0.3.16.tgz", + "integrity": "sha512-AxIGzLRXrTFNL+H6V+4BO0w/gERloROfRbWI/FoJUnQd0qPZIzyfdHZBbThFzFGLfDt/mVs2kdjYFx/l9I8NhQ==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app-check": "0.8.9", + "@firebase/app-check-types": "0.5.2", + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/app-check-interop-types": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@firebase/app-check-interop-types/-/app-check-interop-types-0.3.2.tgz", + "integrity": "sha512-LMs47Vinv2HBMZi49C09dJxp0QT5LwDzFaVGf/+ITHe3BlIhUiLNttkATSXplc89A2lAaeTqjgqVkiRfUGyQiQ==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/app-check-types": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@firebase/app-check-types/-/app-check-types-0.5.2.tgz", + "integrity": "sha512-FSOEzTzL5bLUbD2co3Zut46iyPWML6xc4x+78TeaXMSuJap5QObfb+rVvZJtla3asN4RwU7elaQaduP+HFizDA==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/app-compat": { + "version": "0.2.45", + "resolved": "https://registry.npmjs.org/@firebase/app-compat/-/app-compat-0.2.45.tgz", + "integrity": "sha512-5rYbXq1ndtMTg+07oH4WrkYuP+NZq61uzVwW1hlmybp/gr4cXq2SfaP9fc6/9IzTKmu3dh3H0fjj++HG7Z7o/w==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app": "0.10.15", + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/app-types": { + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/@firebase/app-types/-/app-types-0.9.2.tgz", + "integrity": "sha512-oMEZ1TDlBz479lmABwWsWjzHwheQKiAgnuKxE0pz0IXCVx7/rtlkx1fQ6GfgK24WCrxDKMplZrT50Kh04iMbXQ==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/auth": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/@firebase/auth/-/auth-1.8.0.tgz", + "integrity": "sha512-/O7UDWE5S5ux456fzNHSLx/0YN/Kykw/WyAzgDQ6wvkddZhSEmPX19EzxgsFldzhuFjsl5uOZTz8kzlosCiJjg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x", + "@react-native-async-storage/async-storage": "^1.18.1" + }, + "peerDependenciesMeta": { + "@react-native-async-storage/async-storage": { + "optional": true + } + } + }, + "node_modules/@firebase/auth-compat": { + "version": "0.5.15", + "resolved": "https://registry.npmjs.org/@firebase/auth-compat/-/auth-compat-0.5.15.tgz", + "integrity": "sha512-jz6k1ridPiecKI8CBRiqCM6IMOhwYp2MD+YvoxnMiK8nQLSTm57GvHETlPNX3WlbyQnCjMCOvrAhe27whyxAEg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/auth": "1.8.0", + "@firebase/auth-types": "0.12.2", + "@firebase/component": "0.6.10", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/auth-interop-types": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@firebase/auth-interop-types/-/auth-interop-types-0.2.3.tgz", + "integrity": "sha512-Fc9wuJGgxoxQeavybiuwgyi+0rssr76b+nHpj+eGhXFYAdudMWyfBHvFL/I5fEHniUM/UQdFzi9VXJK2iZF7FQ==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/auth-types": { + "version": "0.12.2", + "resolved": "https://registry.npmjs.org/@firebase/auth-types/-/auth-types-0.12.2.tgz", + "integrity": "sha512-qsEBaRMoGvHO10unlDJhaKSuPn4pyoTtlQuP1ghZfzB6rNQPuhp/N/DcFZxm9i4v0SogjCbf9reWupwIvfmH6w==", + "license": "Apache-2.0", + "peerDependencies": { + "@firebase/app-types": "0.x", + "@firebase/util": "1.x" + } + }, + "node_modules/@firebase/component": { + "version": "0.6.10", + "resolved": "https://registry.npmjs.org/@firebase/component/-/component-0.6.10.tgz", + "integrity": "sha512-OsNbEKyz9iLZSmMUhsl6+kCADzte00iisJIRUspnUqvDCX+RSGZOBIqekukv/jN177ovjApBQNFaxSYIDc/SyQ==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/data-connect": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@firebase/data-connect/-/data-connect-0.1.1.tgz", + "integrity": "sha512-RBJ7XE/a3oXFv31Jlw8cbMRdsxQoI8F3L7xm4n93ab+bIr1NQUiYGgW9L7TTw7obdNev91ZnW0xfqJtXcPA5yA==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/auth-interop-types": "0.2.3", + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/database": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@firebase/database/-/database-1.0.9.tgz", + "integrity": "sha512-EkiPSKSu2TJJGtOjyISASf3UFpFJDil1lMbfqnxilfbmIsilvC8DzgjuLoYD+eOitcug4wtU9Fh1tt2vgBhskA==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app-check-interop-types": "0.3.2", + "@firebase/auth-interop-types": "0.2.3", + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "faye-websocket": "0.11.4", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/database-compat": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@firebase/database-compat/-/database-compat-2.0.0.tgz", + "integrity": "sha512-2xlODKWwf/vNAxCmou0GFhymx2pqZKkhXMN9B5aiTjZ6+81sOxGim53ELY2lj+qKG2IvgiCYFc4X+ZJA2Ad5vg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/database": "1.0.9", + "@firebase/database-types": "1.0.6", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/database-types": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@firebase/database-types/-/database-types-1.0.6.tgz", + "integrity": "sha512-sMI7IynSZBsyGbUugc8PKE1jwKbnvaieAz/RxuM57PZQNCi6Rteiviwcw/jqZOX6igqYJwXWZ3UzKOZo2nUDRA==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app-types": "0.9.2", + "@firebase/util": "1.10.1" + } + }, + "node_modules/@firebase/firestore": { + "version": "4.7.4", + "resolved": "https://registry.npmjs.org/@firebase/firestore/-/firestore-4.7.4.tgz", + "integrity": "sha512-K2nq4w+NF8J1waGawY5OHLawP/Aw5CYxyDstVv1NZemGPcM3U+LZ9EPaXr1PatYIrPA7fS4DxZoWcbB0aGJ8Zg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "@firebase/webchannel-wrapper": "1.0.2", + "@grpc/grpc-js": "~1.9.0", + "@grpc/proto-loader": "^0.7.8", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/firestore-compat": { + "version": "0.3.39", + "resolved": "https://registry.npmjs.org/@firebase/firestore-compat/-/firestore-compat-0.3.39.tgz", + "integrity": "sha512-CsK8g34jNeHx95LISDRTcArJLonW+zJCqHI1Ez9WNiLAK2X8FeQ4UiD+RwOwxAIR+t2a6xED/5Fe6ZIqx7MuoQ==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/firestore": "4.7.4", + "@firebase/firestore-types": "3.0.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/firestore-types": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@firebase/firestore-types/-/firestore-types-3.0.2.tgz", + "integrity": "sha512-wp1A+t5rI2Qc/2q7r2ZpjUXkRVPtGMd6zCLsiWurjsQpqPgFin3AhNibKcIzoF2rnToNa/XYtyWXuifjOOwDgg==", + "license": "Apache-2.0", + "peerDependencies": { + "@firebase/app-types": "0.x", + "@firebase/util": "1.x" + } + }, + "node_modules/@firebase/functions": { + "version": "0.11.9", + "resolved": "https://registry.npmjs.org/@firebase/functions/-/functions-0.11.9.tgz", + "integrity": "sha512-dhO5IUfQRCsrc20YD20nSOX+QCT+cH6N86HlZOLz2XgyEFgzOdBQnUot4EabBJQRkMBI7fZWUrbYfRcnov53ug==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app-check-interop-types": "0.3.2", + "@firebase/auth-interop-types": "0.2.3", + "@firebase/component": "0.6.10", + "@firebase/messaging-interop-types": "0.2.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/functions-compat": { + "version": "0.3.15", + "resolved": "https://registry.npmjs.org/@firebase/functions-compat/-/functions-compat-0.3.15.tgz", + "integrity": "sha512-eiHpc6Sd9Y/SNhBsGi944SapiFbfTPKsiSUQ74QxNSs0yoxvABeIRolVMFk4TokP57NGmstGYpYte02XGNPcYw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/functions": "0.11.9", + "@firebase/functions-types": "0.6.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/functions-types": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/@firebase/functions-types/-/functions-types-0.6.2.tgz", + "integrity": "sha512-0KiJ9lZ28nS2iJJvimpY4nNccV21rkQyor5Iheu/nq8aKXJqtJdeSlZDspjPSBBiHRzo7/GMUttegnsEITqR+w==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/installations": { + "version": "0.6.10", + "resolved": "https://registry.npmjs.org/@firebase/installations/-/installations-0.6.10.tgz", + "integrity": "sha512-TuGSOMqkFrllxa0X/8VZIqBCRH4POndU/iWKWkRmkh12+/xKSpdp+y/kWaVbsySrelltan6LeYlcYPmLibWbwg==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/util": "1.10.1", + "idb": "7.1.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/installations-compat": { + "version": "0.2.10", + "resolved": "https://registry.npmjs.org/@firebase/installations-compat/-/installations-compat-0.2.10.tgz", + "integrity": "sha512-YTonkcVz3AK7RF8xFhvs5CwDuJ0xbzzCJIwXoV14gnzdYbMgy6vWlUUbzkvbtEDXzPRHB0n7aGZl56oy9dLOFw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/installations": "0.6.10", + "@firebase/installations-types": "0.5.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/installations-types": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@firebase/installations-types/-/installations-types-0.5.2.tgz", + "integrity": "sha512-que84TqGRZJpJKHBlF2pkvc1YcXrtEDOVGiDjovP/a3s6W4nlbohGXEsBJo0JCeeg/UG9A+DEZVDUV9GpklUzA==", + "license": "Apache-2.0", + "peerDependencies": { + "@firebase/app-types": "0.x" + } + }, + "node_modules/@firebase/logger": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@firebase/logger/-/logger-0.4.3.tgz", + "integrity": "sha512-Th42bWJg18EF5bJwhRosn2M/eYxmbWCwXZr4hHX7ltO0SE3QLrpgiMKeRBR/NW7vJke7i0n3i8esbCW2s93qBw==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/messaging": { + "version": "0.12.13", + "resolved": "https://registry.npmjs.org/@firebase/messaging/-/messaging-0.12.13.tgz", + "integrity": "sha512-YLa8PWl+BgiOVR5WOyzl21fVJFJeBRfniNuN25d9DBrQzppSAahuN6yS+vt1OIjvZNPN4pZ/lcRLYupbGu4W0w==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/installations": "0.6.10", + "@firebase/messaging-interop-types": "0.2.2", + "@firebase/util": "1.10.1", + "idb": "7.1.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/messaging-compat": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/@firebase/messaging-compat/-/messaging-compat-0.2.13.tgz", + "integrity": "sha512-9ootPClS6m2c2KIzo7AqSHaWzAw28zWcjQPjVv7WeQDu6wjufpbOg+7tuVzb+gqpF9Issa3lDoYOwlO0ZudO3g==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/messaging": "0.12.13", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/messaging-interop-types": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/@firebase/messaging-interop-types/-/messaging-interop-types-0.2.2.tgz", + "integrity": "sha512-l68HXbuD2PPzDUOFb3aG+nZj5KA3INcPwlocwLZOzPp9rFM9yeuI9YLl6DQfguTX5eAGxO0doTR+rDLDvQb5tA==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/performance": { + "version": "0.6.10", + "resolved": "https://registry.npmjs.org/@firebase/performance/-/performance-0.6.10.tgz", + "integrity": "sha512-x/mNYKGxq7A+QV0EiEZeD2S+E+kw+UcZ8FXuE7qDJyGGt/0Wd+bIIL7RakG/VrFt7/UYc//nKygDc7/Ig7sOmQ==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/installations": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/performance-compat": { + "version": "0.2.10", + "resolved": "https://registry.npmjs.org/@firebase/performance-compat/-/performance-compat-0.2.10.tgz", + "integrity": "sha512-0h1qYkF6I79DSSpHfTQFvb91fo8shmmwiPzWFYAPdPK02bSWpKwVssNYlZX2iUnumxerDMbl7dWN+Im/W3bnXA==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/performance": "0.6.10", + "@firebase/performance-types": "0.2.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/performance-types": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/@firebase/performance-types/-/performance-types-0.2.2.tgz", + "integrity": "sha512-gVq0/lAClVH5STrIdKnHnCo2UcPLjJlDUoEB/tB4KM+hAeHUxWKnpT0nemUPvxZ5nbdY/pybeyMe8Cs29gEcHA==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/remote-config": { + "version": "0.4.10", + "resolved": "https://registry.npmjs.org/@firebase/remote-config/-/remote-config-0.4.10.tgz", + "integrity": "sha512-jTRjy3TdqzVna19m5a1HEHE5BG4Z3BQTxBgvQRTmMKlHacx4QS0CToAas7R9M9UkxpgFcVuAE7FpWIOWQGCEWw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/installations": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/remote-config-compat": { + "version": "0.2.10", + "resolved": "https://registry.npmjs.org/@firebase/remote-config-compat/-/remote-config-compat-0.2.10.tgz", + "integrity": "sha512-fIi5OB2zk0zpChMV/tTd0oEZcZI8TlwQDlLlcrDpMOV5l5dqd0JNlWKh6Fwmh4izmytk+rZIAIpnak/NjGVesQ==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/remote-config": "0.4.10", + "@firebase/remote-config-types": "0.3.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/remote-config-types": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@firebase/remote-config-types/-/remote-config-types-0.3.2.tgz", + "integrity": "sha512-0BC4+Ud7y2aPTyhXJTMTFfrGGLqdYXrUB9sJVAB8NiqJswDTc4/2qrE/yfUbnQJhbSi6ZaTTBKyG3n1nplssaA==", + "license": "Apache-2.0" + }, + "node_modules/@firebase/storage": { + "version": "0.13.3", + "resolved": "https://registry.npmjs.org/@firebase/storage/-/storage-0.13.3.tgz", + "integrity": "sha512-B5HiJ7isYKaT4dOEV43f2ySdhQxzq+SQEm7lqXebJ8AYCsebdHrgGzrPR0LR962xGjPzJHFKx63gA8Be/P2MCw==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x" + } + }, + "node_modules/@firebase/storage-compat": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@firebase/storage-compat/-/storage-compat-0.3.13.tgz", + "integrity": "sha512-15kje7JALswRCBKsCSvKg5FbqUYykaIMqMbZRD7I6uVRWwdyTvez5MBQfMhBia2JcEmPiDpXhJTXH4PAWFiA8g==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/component": "0.6.10", + "@firebase/storage": "0.13.3", + "@firebase/storage-types": "0.8.2", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app-compat": "0.x" + } + }, + "node_modules/@firebase/storage-types": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/@firebase/storage-types/-/storage-types-0.8.2.tgz", + "integrity": "sha512-0vWu99rdey0g53lA7IShoA2Lol1jfnPovzLDUBuon65K7uKG9G+L5uO05brD9pMw+l4HRFw23ah3GwTGpEav6g==", + "license": "Apache-2.0", + "peerDependencies": { + "@firebase/app-types": "0.x", + "@firebase/util": "1.x" + } + }, + "node_modules/@firebase/util": { + "version": "1.10.1", + "resolved": "https://registry.npmjs.org/@firebase/util/-/util-1.10.1.tgz", + "integrity": "sha512-AIhFnCCjM8FmCqSNlNPTuOk3+gpHC1RkeNUBLtPbcqGYpN5MxI5q7Yby+rxycweOZOCboDzfIj8WyaY4tpQG/g==", + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/@firebase/vertexai": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@firebase/vertexai/-/vertexai-1.0.0.tgz", + "integrity": "sha512-48N3Lp/9GgiCCRfrSdHS+Y1IiMdYXvnHFO/f+HL1PgUtBq7WQ/fWmYOX3mzAN36zvytq13nb68ImF+GALopp+Q==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/app-check-interop-types": "0.3.2", + "@firebase/component": "0.6.10", + "@firebase/logger": "0.4.3", + "@firebase/util": "1.10.1", + "tslib": "^2.1.0" + }, + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "@firebase/app": "0.x", + "@firebase/app-types": "0.x" + } + }, + "node_modules/@firebase/webchannel-wrapper": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@firebase/webchannel-wrapper/-/webchannel-wrapper-1.0.2.tgz", + "integrity": "sha512-3F4iA2E+NtdMbOU0XC1cHE8q6MqpGIKRj62oGOF38S6AAx5VHR9cXmoDUSj7ejvTAT7m6jxuEeQkHeq0F+mU2w==", + "license": "Apache-2.0" + }, + "node_modules/@floating-ui/core": { + "version": "1.6.8", + "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.6.8.tgz", + "integrity": "sha512-7XJ9cPU+yI2QeLS+FCSlqNFZJq8arvswefkZrYI1yQBbftw6FyrZOxYSh+9S7z7TpeWlRt9zJ5IhM1WIL334jA==", + "license": "MIT", + "dependencies": { + "@floating-ui/utils": "^0.2.8" + } + }, + "node_modules/@floating-ui/dom": { + "version": "1.6.12", + "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.6.12.tgz", + "integrity": "sha512-NP83c0HjokcGVEMeoStg317VD9W7eDlGK7457dMBANbKA6GJZdc7rjujdgqzTaz93jkGgc5P/jeWbaCHnMNc+w==", + "license": "MIT", + "dependencies": { + "@floating-ui/core": "^1.6.0", + "@floating-ui/utils": "^0.2.8" + } + }, + "node_modules/@floating-ui/react": { + "version": "0.26.27", + "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.27.tgz", + "integrity": "sha512-jLP72x0Kr2CgY6eTYi/ra3VA9LOkTo4C+DUTrbFgFOExKy3omYVmwMjNKqxAHdsnyLS96BIDLcO2SlnsNf8KUQ==", + "license": "MIT", + "dependencies": { + "@floating-ui/react-dom": "^2.1.2", + "@floating-ui/utils": "^0.2.8", + "tabbable": "^6.0.0" + }, + "peerDependencies": { + "react": ">=16.8.0", + "react-dom": ">=16.8.0" + } + }, + "node_modules/@floating-ui/react-dom": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/@floating-ui/react-dom/-/react-dom-2.1.2.tgz", + "integrity": "sha512-06okr5cgPzMNBy+Ycse2A6udMi4bqwW/zgBF/rwjcNqWkyr82Mcg8b0vjX8OJpZFy/FKjJmw6wV7t44kK6kW7A==", + "license": "MIT", + "dependencies": { + "@floating-ui/dom": "^1.0.0" + }, + "peerDependencies": { + "react": ">=16.8.0", + "react-dom": ">=16.8.0" + } + }, + "node_modules/@floating-ui/utils": { + "version": "0.2.8", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.8.tgz", + "integrity": "sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==", + "license": "MIT" + }, "node_modules/@fullcalendar/common": { "version": "5.11.5", "resolved": "https://registry.npmjs.org/@fullcalendar/common/-/common-5.11.5.tgz", @@ -1135,6 +1797,37 @@ "@fullcalendar/core": "~6.1.15" } }, + "node_modules/@grpc/grpc-js": { + "version": "1.9.15", + "resolved": "https://registry.npmjs.org/@grpc/grpc-js/-/grpc-js-1.9.15.tgz", + "integrity": "sha512-nqE7Hc0AzI+euzUwDAy0aY5hCp10r734gMGRdU+qOPX0XSceI2ULrcXB5U2xSc5VkWwalCj4M7GzCAygZl2KoQ==", + "license": "Apache-2.0", + "dependencies": { + "@grpc/proto-loader": "^0.7.8", + "@types/node": ">=12.12.47" + }, + "engines": { + "node": "^8.13.0 || >=10.10.0" + } + }, + "node_modules/@grpc/proto-loader": { + "version": "0.7.13", + "resolved": "https://registry.npmjs.org/@grpc/proto-loader/-/proto-loader-0.7.13.tgz", + "integrity": "sha512-AiXO/bfe9bmxBjxxtYxFAXGZvMaN5s8kO+jBHAJCON8rJoB5YS/D6X7ZNc6XQkuHNmyl4CYaMI1fJ/Gn27RGGw==", + "license": "Apache-2.0", + "dependencies": { + "lodash.camelcase": "^4.3.0", + "long": "^5.0.0", + "protobufjs": "^7.2.5", + "yargs": "^17.7.2" + }, + "bin": { + "proto-loader-gen-types": "build/bin/proto-loader-gen-types.js" + }, + "engines": { + "node": ">=6" + } + }, "node_modules/@humanwhocodes/config-array": { "version": "0.11.14", "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.14.tgz", @@ -1676,6 +2369,70 @@ "url": "https://opencollective.com/popperjs" } }, + "node_modules/@protobufjs/aspromise": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz", + "integrity": "sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/base64": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz", + "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/codegen": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz", + "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/eventemitter": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz", + "integrity": "sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/fetch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz", + "integrity": "sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==", + "license": "BSD-3-Clause", + "dependencies": { + "@protobufjs/aspromise": "^1.1.1", + "@protobufjs/inquire": "^1.1.0" + } + }, + "node_modules/@protobufjs/float": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz", + "integrity": "sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/inquire": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz", + "integrity": "sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/path": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz", + "integrity": "sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/pool": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz", + "integrity": "sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw==", + "license": "BSD-3-Clause" + }, + "node_modules/@protobufjs/utf8": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz", + "integrity": "sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==", + "license": "BSD-3-Clause" + }, "node_modules/@react-dnd/asap": { "version": "5.0.2", "resolved": "https://registry.npmjs.org/@react-dnd/asap/-/asap-5.0.2.tgz", @@ -3006,7 +3763,6 @@ "version": "22.5.5", "resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.5.tgz", "integrity": "sha512-Xjs4y5UPO/CLdzpgR6GirZJx36yScjh73+2NlLlkFRSoQN8B0DpfXPdZGnvVmLRLOsqDpOfTNv7D9trgGhmOIA==", - "devOptional": true, "dependencies": { "undici-types": "~6.19.2" } @@ -3517,7 +4273,6 @@ "version": "5.0.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true, "engines": { "node": ">=8" } @@ -4057,7 +4812,6 @@ "version": "8.0.1", "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", - "dev": true, "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.1", @@ -4071,7 +4825,6 @@ "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, "dependencies": { "color-convert": "^2.0.1" }, @@ -4086,7 +4839,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, "dependencies": { "color-name": "~1.1.4" }, @@ -4097,14 +4849,12 @@ "node_modules/cliui/node_modules/color-name": { "version": "1.1.4", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" }, "node_modules/cliui/node_modules/wrap-ansi": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "dev": true, "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", @@ -4306,6 +5056,16 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/date-fns": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-4.1.0.tgz", + "integrity": "sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/kossnocorp" + } + }, "node_modules/debug": { "version": "4.3.7", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", @@ -4505,8 +5265,7 @@ "node_modules/emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, "node_modules/encodeurl": { "version": "2.0.0", @@ -4763,7 +5522,6 @@ "version": "3.2.0", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", - "dev": true, "engines": { "node": ">=6" } @@ -5807,6 +6565,18 @@ "reusify": "^1.0.4" } }, + "node_modules/faye-websocket": { + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "license": "Apache-2.0", + "dependencies": { + "websocket-driver": ">=0.5.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, "node_modules/file-entry-cache": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", @@ -5911,6 +6681,42 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/firebase": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/firebase/-/firebase-11.0.1.tgz", + "integrity": "sha512-qsFb8dMcQINEDhJteG7RP+GqwgSRvfyiexQqHd5JToDdm87i9I2rGC4XQsGawKGxzKwZ/ISdgwNWxXAFYdCC6A==", + "license": "Apache-2.0", + "dependencies": { + "@firebase/analytics": "0.10.9", + "@firebase/analytics-compat": "0.2.15", + "@firebase/app": "0.10.15", + "@firebase/app-check": "0.8.9", + "@firebase/app-check-compat": "0.3.16", + "@firebase/app-compat": "0.2.45", + "@firebase/app-types": "0.9.2", + "@firebase/auth": "1.8.0", + "@firebase/auth-compat": "0.5.15", + "@firebase/data-connect": "0.1.1", + "@firebase/database": "1.0.9", + "@firebase/database-compat": "2.0.0", + "@firebase/firestore": "4.7.4", + "@firebase/firestore-compat": "0.3.39", + "@firebase/functions": "0.11.9", + "@firebase/functions-compat": "0.3.15", + "@firebase/installations": "0.6.10", + "@firebase/installations-compat": "0.2.10", + "@firebase/messaging": "0.12.13", + "@firebase/messaging-compat": "0.2.13", + "@firebase/performance": "0.6.10", + "@firebase/performance-compat": "0.2.10", + "@firebase/remote-config": "0.4.10", + "@firebase/remote-config-compat": "0.2.10", + "@firebase/storage": "0.13.3", + "@firebase/storage-compat": "0.3.13", + "@firebase/util": "1.10.1", + "@firebase/vertexai": "1.0.0" + } + }, "node_modules/flat-cache": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", @@ -6096,7 +6902,6 @@ "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "dev": true, "engines": { "node": "6.* || 8.* || >= 10.*" } @@ -6463,6 +7268,12 @@ "node": ">= 0.8" } }, + "node_modules/http-parser-js": { + "version": "0.5.8", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.8.tgz", + "integrity": "sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q==", + "license": "MIT" + }, "node_modules/iconv-lite": { "version": "0.4.24", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", @@ -6475,6 +7286,12 @@ "node": ">=0.10.0" } }, + "node_modules/idb": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/idb/-/idb-7.1.1.tgz", + "integrity": "sha512-gchesWBzyvGHRO9W8tzUWFDycow5gwjvFKfyV9FF32Y7F50yZMp7mP+T2mJIWFx49zicqyC4uefHM17o6xKIVQ==", + "license": "ISC" + }, "node_modules/ignore": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", @@ -6753,7 +7570,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true, "engines": { "node": ">=8" } @@ -7174,12 +7990,24 @@ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", "dev": true }, + "node_modules/lodash.camelcase": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz", + "integrity": "sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==", + "license": "MIT" + }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", "dev": true }, + "node_modules/long": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/long/-/long-5.2.3.tgz", + "integrity": "sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q==", + "license": "Apache-2.0" + }, "node_modules/loose-envify": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", @@ -8153,6 +8981,30 @@ "react-is": "^16.13.1" } }, + "node_modules/protobufjs": { + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-7.4.0.tgz", + "integrity": "sha512-mRUWCc3KUU4w1jU8sGxICXH/gNS94DvI1gxqDvBzhj1JpcsimQkYiOJfwsPUykUI5ZaspFbSgmBLER8IrQ3tqw==", + "hasInstallScript": true, + "license": "BSD-3-Clause", + "dependencies": { + "@protobufjs/aspromise": "^1.1.2", + "@protobufjs/base64": "^1.1.2", + "@protobufjs/codegen": "^2.0.4", + "@protobufjs/eventemitter": "^1.1.0", + "@protobufjs/fetch": "^1.1.0", + "@protobufjs/float": "^1.0.2", + "@protobufjs/inquire": "^1.1.0", + "@protobufjs/path": "^1.1.2", + "@protobufjs/pool": "^1.1.0", + "@protobufjs/utf8": "^1.1.0", + "@types/node": ">=13.7.0", + "long": "^5.0.0" + }, + "engines": { + "node": ">=12.0.0" + } + }, "node_modules/proxy-addr": { "version": "2.0.7", "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", @@ -8301,6 +9153,32 @@ "react": ">= 16.3.0" } }, + "node_modules/react-datepicker": { + "version": "7.5.0", + "resolved": "https://registry.npmjs.org/react-datepicker/-/react-datepicker-7.5.0.tgz", + "integrity": "sha512-6MzeamV8cWSOcduwePHfGqY40acuGlS1cG//ePHT6bVbLxWyqngaStenfH03n1wbzOibFggF66kWaBTb1SbTtQ==", + "license": "MIT", + "dependencies": { + "@floating-ui/react": "^0.26.23", + "clsx": "^2.1.1", + "date-fns": "^3.6.0", + "prop-types": "^15.8.1" + }, + "peerDependencies": { + "react": "^16.9.0 || ^17 || ^18", + "react-dom": "^16.9.0 || ^17 || ^18" + } + }, + "node_modules/react-datepicker/node_modules/date-fns": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-3.6.0.tgz", + "integrity": "sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/kossnocorp" + } + }, "node_modules/react-dnd": { "version": "16.0.1", "resolved": "https://registry.npmjs.org/react-dnd/-/react-dnd-16.0.1.tgz", @@ -8629,7 +9507,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", - "dev": true, "engines": { "node": ">=0.10.0" } @@ -8779,7 +9656,6 @@ "version": "5.2.1", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "dev": true, "funding": [ { "type": "github", @@ -9074,7 +9950,6 @@ "version": "4.2.3", "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "dev": true, "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", @@ -9183,7 +10058,6 @@ "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "dev": true, "dependencies": { "ansi-regex": "^5.0.1" }, @@ -9270,6 +10144,12 @@ "url": "https://opencollective.com/unts" } }, + "node_modules/tabbable": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/tabbable/-/tabbable-6.2.0.tgz", + "integrity": "sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==", + "license": "MIT" + }, "node_modules/telejson": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/telejson/-/telejson-7.2.0.tgz", @@ -9570,8 +10450,7 @@ "node_modules/undici-types": { "version": "6.19.8", "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", - "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", - "devOptional": true + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==" }, "node_modules/unist-util-is": { "version": "6.0.0", @@ -9835,6 +10714,29 @@ "integrity": "sha512-66/V2i5hQanC51vBQKPH4aI8NMAcBW59FVBs+rC7eGHupMyfn34q7rZIE+ETlJ+XTevqfUhVVBgSUNSW2flEUQ==", "dev": true }, + "node_modules/websocket-driver": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz", + "integrity": "sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==", + "license": "Apache-2.0", + "dependencies": { + "http-parser-js": ">=0.5.1", + "safe-buffer": ">=5.1.0", + "websocket-extensions": ">=0.1.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/websocket-extensions": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/websocket-extensions/-/websocket-extensions-0.1.4.tgz", + "integrity": "sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==", + "license": "Apache-2.0", + "engines": { + "node": ">=0.8.0" + } + }, "node_modules/which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -10015,7 +10917,6 @@ "version": "5.0.8", "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", - "dev": true, "engines": { "node": ">=10" } @@ -10038,7 +10939,6 @@ "version": "17.7.2", "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", - "dev": true, "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", @@ -10056,7 +10956,6 @@ "version": "21.1.1", "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", - "dev": true, "engines": { "node": ">=12" } diff --git a/package.json b/package.json index adb206d..fb5feeb 100644 --- a/package.json +++ b/package.json @@ -28,6 +28,8 @@ "@tanstack/react-query": "^5.56.2", "axios": "^1.7.7", "events": "^3.3.0", + "date-fns": "^4.1.0", + "firebase": "^11.0.1", "framer-motion": "^11.9.0", "js-cookie": "^3.0.5", "lucide-react": "^0.456.0", @@ -37,6 +39,7 @@ "postcss-import": "^16.1.0", "react": "^18.2.0", "react-cookie": "^7.2.2", + "react-datepicker": "^7.5.0", "react-dnd": "^16.0.1", "react-dnd-html5-backend": "^16.0.1", "react-dom": "^18.2.0", diff --git a/public/firebase-messaging-sw.js b/public/firebase-messaging-sw.js new file mode 100644 index 0000000..6bdb526 --- /dev/null +++ b/public/firebase-messaging-sw.js @@ -0,0 +1,31 @@ +importScripts( + "https://www.gstatic.com/firebasejs/9.0.0/firebase-app-compat.js", +); +importScripts( + "https://www.gstatic.com/firebasejs/9.0.0/firebase-messaging-compat.js", +); + +// 환경 변수에서 Firebase 설정값 가져오기 +firebase.initializeApp({ + apiKey: "AIzaSyAInigygScRLDilnWcnArBN8LMbQRpDZVk", + authDomain: "splanet-cef14.firebaseapp.com", + projectId: "splanet-cef14", + storageBucket: "splanet-cef14.appspot.com", + messagingSenderId: "995362943401", + appId: "1:995362943401:web:cef434d0e3f51d31a4d4b8", + measurementId: "G-LZJKRYBSJV", +}); + +const messaging = firebase.messaging(); + +// 백그라운드 메시지 처리 +messaging.onBackgroundMessage((payload) => { + console.log("Received background message ", payload); + const notificationTitle = payload.notification.title; + const notificationOptions = { + body: payload.notification.body, + icon: "/icon.png", // 아이콘 이미지 경로 + }; + + self.registration.showNotification(notificationTitle, notificationOptions); +}); diff --git a/src/api/firebaseConfig.ts b/src/api/firebaseConfig.ts new file mode 100644 index 0000000..917a7bb --- /dev/null +++ b/src/api/firebaseConfig.ts @@ -0,0 +1,54 @@ +// eslint-disable-next-line import/no-extraneous-dependencies +import { initializeApp } from "firebase/app"; +// eslint-disable-next-line import/no-extraneous-dependencies +import { + getMessaging, + getToken, + onMessage as firebaseOnMessage, +} from "firebase/messaging"; +// 환경 변수에서 Firebase 설정값 가져오기 +const firebaseConfig = { + apiKey: import.meta.env.VITE_FIREBASE_API_KEY, + authDomain: import.meta.env.VITE_FIREBASE_AUTH_DOMAIN, + projectId: import.meta.env.VITE_FIREBASE_PROJECT_ID, + storageBucket: import.meta.env.VITE_FIREBASE_STORAGE_BUCKET, + messagingSenderId: import.meta.env.VITE_FIREBASE_MESSAGING_SENDER_ID, + appId: import.meta.env.VITE_FIREBASE_APP_ID, + measurementId: import.meta.env.VITE_FIREBASE_MEASUREMENT_ID, +}; +// Firebase 초기화 +const app = initializeApp(firebaseConfig); +const messaging = getMessaging(app); +// VAPID 키 가져오기 +const vapidKey = import.meta.env.VITE_FIREBASE_VAPID_KEY; +// FCM 토큰 요청 함수 +export const requestForToken = async () => { + try { + const currentToken = await getToken(messaging, { vapidKey }); + if (currentToken) { + console.log("FCM token:", currentToken); + return currentToken; + } + console.log( + "No registration token available. Request permission to generate one.", + ); + return null; + } catch (error) { + console.error("An error occurred while retrieving token. ", error); + return null; + } +}; +// 메시지 수신 리스너 설정 +export const setupOnMessageListener = () => { + firebaseOnMessage(messaging, (payload) => { + console.log("Message received: ", payload); + const notificationTitle = payload.notification?.title || "알림"; + const notificationOptions = { + body: payload.notification?.body || "새로운 알림이 도착했습니다.", + }; + // 브라우저 알림 표시 + // eslint-disable-next-line no-new + new Notification(notificationTitle, notificationOptions); + }); +}; +export { messaging }; diff --git a/src/api/instance/index.ts b/src/api/instance/index.ts index b770e47..e2d65d2 100644 --- a/src/api/instance/index.ts +++ b/src/api/instance/index.ts @@ -4,7 +4,7 @@ import type { AxiosInstance, AxiosRequestConfig } from "axios"; import { QueryClient } from "@tanstack/react-query"; import EventEmitter from "events"; // BASE URL 설정 -const API_URL = import.meta.env.VITE_API_URL || "http://localhost:5173"; +const API_URL = import.meta.env.VITE_API_URL; // 쿠키에서 특정 토큰을 가져오는 로직 const getCookie = (name: string) => { const value = `; ${document.cookie}`; diff --git a/src/assets/icon.png b/src/assets/icon.png new file mode 100644 index 0000000..bdc825f Binary files /dev/null and b/src/assets/icon.png differ diff --git a/src/components/features/CustomCalendar/CustomCalendar.styles.ts b/src/components/features/CustomCalendar/CustomCalendar.styles.ts index 9332300..ac20d86 100644 --- a/src/components/features/CustomCalendar/CustomCalendar.styles.ts +++ b/src/components/features/CustomCalendar/CustomCalendar.styles.ts @@ -190,3 +190,54 @@ export const eventItemStyles = (status: string, isDragging: boolean) => css` border-left-color: #ef4444; `} `; + +export const dropdownMenuStyles = css` + position: absolute; + top: 100%; + right: 0; + background-color: white; + list-style: none; + padding: 8px 0; + margin: 4px 0 0 0; + border: 1px solid #ddd; + border-radius: 8px; + box-shadow: 0 4px 8px rgba(0, 0, 0, 0.15); + z-index: 1000; + min-width: 100px; + animation: fadeIn 0.2s ease-in-out; + + @keyframes fadeIn { + from { + opacity: 0; + transform: translateY(-10px); + } + to { + opacity: 1; + transform: translateY(0); + } + } +`; + +export const dropdownItemStyles = css` + padding: 10px 12px; + color: blue; + border-radius: 8px; + font-size: 14px; + cursor: pointer; + text-align: left; + transition: background-color 0.2s; + display: block; + white-space: nowrap; + + &:hover { + padding: 7px 11px; /* hover 시 패딩을 줄여 크기 감소 */ + } +`; + +export const dropdownItemRedStyles = css` + ${dropdownItemStyles} + color: red; + &:hover { + padding: 7px 11px; /* hover 시 패딩을 줄여 크기 감소 */ + } +`; diff --git a/src/components/features/CustomCalendar/CustomCalendar.tsx b/src/components/features/CustomCalendar/CustomCalendar.tsx index 5232249..338da27 100644 --- a/src/components/features/CustomCalendar/CustomCalendar.tsx +++ b/src/components/features/CustomCalendar/CustomCalendar.tsx @@ -12,17 +12,66 @@ import dayGridPlugin from "@fullcalendar/daygrid"; import timeGridPlugin from "@fullcalendar/timegrid"; import interactionPlugin from "@fullcalendar/interaction"; import koLocale from "@fullcalendar/core/locales/ko"; +import styled from "@emotion/styled"; import breakpoints from "@/variants/breakpoints"; import { appContainerStyles, appTitleStyles, calendarStyles, eventItemStyles, + dropdownItemStyles, + dropdownItemRedStyles, + dropdownMenuStyles, } from "./CustomCalendar.styles"; import useDeletePlan from "@/api/hooks/useDeletePlan"; -import Modal from "./PlanModal"; +import Modal from "@/components/common/Modal/Modal"; import Button from "@/components/common/Button/Button"; +const ModalContainer = styled.div` + padding: 20px; + background-color: white; + border-radius: 12px; + display: flex; + flex-direction: column; + align-items: center; + gap: 15px; +`; + +const Title = styled.h2` + font-size: 1.8rem; + font-weight: bold; + color: #333; + margin-bottom: 20px; +`; + +const StyledInput = styled.input` + width: 100%; + padding: 12px; + margin-bottom: 10px; + border: 1px solid #ccc; + border-radius: 8px; + font-size: 1rem; + &:focus { + outline: none; + border-color: #6c63ff; + box-shadow: 0 0 0 2px rgba(108, 99, 255, 0.3); + } +`; + +const ToggleContainer = styled.div` + display: flex; + align-items: center; + width: 100%; + justify-content: space-between; + font-size: 1.2rem; + margin-bottom: 10px; +`; + +const ToggleSwitch = styled.input` + width: 20px; + height: 20px; +`; + export interface CalendarEvent { id: string; title: string; @@ -39,7 +88,6 @@ interface CustomCalendarProps { isReadOnly?: boolean; onPlanChange?: (plans: CalendarEvent[]) => void; onDeletePlan?: (planId: string) => void; - // onUpdatePlan?: (planId: string, planData: any) => void; } const VIEW_MODES = { @@ -55,72 +103,134 @@ const calculateEventStatus = (event: CalendarEvent) => { return "incomplete"; }; -const renderEventContent = ( - eventInfo: EventContentArg, - currentView: string, - handleDelete: (id: string) => void, +const EventContent = ({ + eventInfo, + handleDelete, + handleEdit, + isReadOnly, +}: { + eventInfo: EventContentArg; + handleDelete: (id: string) => void; handleEdit: ( id: string, title: string, description: string, accessibility: boolean | null, isCompleted: boolean | null, - ) => void, - isReadOnly: boolean, -) => { - if (currentView === "dayGridMonth") { - return
; - } - + ) => void; + isReadOnly: boolean; +}) => { const { event, timeText } = eventInfo; const description = event.extendedProps?.description || ""; const accessibility = event.extendedProps?.accessibility || false; const isCompleted = event.extendedProps?.isCompleted || false; + const [isDropdownOpen, setIsDropdownOpen] = useState(false); + const handleEventClick = (e: React.MouseEvent) => { + e.stopPropagation(); // 이벤트 버블링 방지 + setIsDropdownOpen(!isDropdownOpen); + }; + const handleKeyDown = (e: React.KeyboardEvent) => { + if (e.key === "Enter" || e.key === " ") { + e.preventDefault(); + setIsDropdownOpen(!isDropdownOpen); + } + }; + const handleOptionClick = (option: string) => { + if (option === "edit") { + handleEdit( + event.id, + event.title, + description, + accessibility, + isCompleted, + ); + } else if (option === "delete") { + handleDelete(event.id); + } + setIsDropdownOpen(false); + }; + // 드롭다운 외부 클릭 시 닫힘 처리 + useEffect(() => { + const handleClickOutside = (e: MouseEvent) => { + if (!(e.target instanceof Node)) return; + setIsDropdownOpen(false); + }; + if (isDropdownOpen) { + document.addEventListener("click", handleClickOutside); + } + return () => { + document.removeEventListener("click", handleClickOutside); + }; + }, [isDropdownOpen]); return ( -
+
{timeText}
{event.title}
{description}
- {!isReadOnly && ( -
- - -
+ 삭제 + + )}
); }; +// 이 함수도 컴포넌트 외부에 위치 +const renderEventContent = ( + eventInfo: EventContentArg, + currentView: string, + handleDelete: (id: string) => void, + handleEdit: ( + id: string, + title: string, + description: string, + accessibility: boolean | null, + isCompleted: boolean | null, + ) => void, + isReadOnly: boolean, +) => { + + if (currentView === "dayGridMonth") { + return
; + } + + return ( + + ); +}; const parseDate = (date: any) => { return typeof date === "string" || typeof date === "number" ? new Date(date) @@ -145,12 +255,10 @@ const CustomCalendar: React.FC = ({ const handleDelete = useCallback( (id: string) => { - { - if (onDeletePlan) { - onDeletePlan(id); - } else { - deletePlan(Number(id)); - } + if (onDeletePlan) { + onDeletePlan(id); + } else { + deletePlan(Number(id)); } }, [deletePlan, onDeletePlan], @@ -193,7 +301,8 @@ const CustomCalendar: React.FC = ({ }; const handleResize = useCallback(() => { - const currentMobile = window.innerWidth <= breakpoints.sm; + const currentMobile = + typeof window !== "undefined" && window.innerWidth <= breakpoints.sm; setIsMobile(currentMobile); const calendarApi = calendarRef.current?.getApi(); calendarApi?.changeView( @@ -236,6 +345,12 @@ const CustomCalendar: React.FC = ({ [onPlanChange, plans], ); + // useCallback으로 메모이제이션 + const eventContent = useCallback( + (eventInfo: EventContentArg) => + renderEventContent(eventInfo, currentView, handleDelete, handleEdit, isReadOnly), + [handleDelete, handleEdit, isReadOnly], + ); return (
{calendarOwner &&

{calendarOwner}

} @@ -281,20 +396,13 @@ const CustomCalendar: React.FC = ({ eventResizableFromStart={!isReadOnly} eventDrop={isReadOnly ? undefined : handleEventChange} eventResize={handleEventChange} - eventContent={(eventInfo) => - renderEventContent( - eventInfo, - currentView, - handleDelete, - handleEdit, - isReadOnly, - ) - } + eventContent={eventContent} selectable={false} selectMirror={false} dayMaxEvents weekends firstDay={1} + timeZone="UTC" events={parsedEvents} viewDidMount={({ view }) => setCurrentView(view.type)} datesSet={(dateInfo) => setCurrentDate(dateInfo.start)} @@ -308,46 +416,54 @@ const CustomCalendar: React.FC = ({ /> {isEditModalOpen && currentEditPlan && ( setIsEditModalOpen(false)}> -

플랜 수정

- - setCurrentEditPlan((prev) => - prev ? { ...prev, title: e.target.value } : prev, - ) - } - /> - - setCurrentEditPlan((prev) => - prev ? { ...prev, description: e.target.value } : prev, - ) - } - /> - 공개 여부: - - setCurrentEditPlan((prev) => - prev ? { ...prev, accessibility: e.target.checked } : prev, - ) - } - /> - 완료 여부: - - setCurrentEditPlan((prev) => - prev ? { ...prev, complete: e.target.checked } : prev, - ) - } - /> - + + 플랜 수정 + + setCurrentEditPlan((prev) => + prev ? { ...prev, title: e.target.value } : prev, + ) + } + /> + + setCurrentEditPlan((prev) => + prev ? { ...prev, description: e.target.value } : prev, + ) + } + /> + + 공개 여부 + + setCurrentEditPlan((prev) => ({ + ...prev, + accessibility: e.target.checked, + })) + } + /> + + + 완료 여부 + + setCurrentEditPlan((prev) => ({ + ...prev, + complete: e.target.checked, + })) + } + /> + + +
)}
diff --git a/src/components/features/CustomCalendar/PlanModal.tsx b/src/components/features/CustomCalendar/PlanModal.tsx deleted file mode 100644 index f8f7fae..0000000 --- a/src/components/features/CustomCalendar/PlanModal.tsx +++ /dev/null @@ -1,72 +0,0 @@ -import React, { ReactNode } from "react"; -import styled from "@emotion/styled"; - -interface ModalProps { - onClose: () => void; - children: ReactNode; -} - -const Overlay = styled.div` - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - background-color: rgba(0, 0, 0, 0.5); - display: flex; - align-items: center; - justify-content: center; - z-index: 1000; -`; - -const ModalContainer = styled.div` - background: white; - padding: 20px; - border-radius: 8px; - width: 90%; - max-width: 500px; - box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2); -`; - -const ModalHeader = styled.div` - display: flex; - justify-content: space-between; - align-items: center; - margin-bottom: 16px; -`; - -const CloseButton = styled.button` - background: none; - border: none; - font-size: 1.5rem; - cursor: pointer; - color: #333; -`; - -const ModalTitle = styled.h2` - font-size: 1.25rem; - font-weight: bold; -`; - -const ModalContent = styled.div` - margin-top: 10px; - display: flex; - flex-direction: column; - gap: 10px; -`; - -const Modal: React.FC = ({ onClose, children }) => { - return ( - - e.stopPropagation()}> - - 플랜 수정 - × - - {children} - - - ); -}; - -export default Modal; diff --git a/src/components/features/DatePicker/DatePicker.css b/src/components/features/DatePicker/DatePicker.css new file mode 100644 index 0000000..7711efd --- /dev/null +++ b/src/components/features/DatePicker/DatePicker.css @@ -0,0 +1,43 @@ +.custom-datepicker-wrapper { + position: relative; + z-index: 100; +} + +.custom-calendar { + border: 1px solid #ddd !important; + font-family: -apple-system, system-ui, "Segoe UI", Roboto, Helvetica, Arial, sans-serif; +} + +/* 시간 선택 컨테이너 스타일 */ +.react-datepicker__time-container { + border-left: 1px solid #ddd !important; + width: 100px !important; + display: block !important; +} + +.react-datepicker { + display: flex !important; +} + +.react-datepicker__time-box { + width: 100px !important; + margin: 0 !important; +} + +/* 시간 리스트 스타일 */ +.react-datepicker__time-list { + height: 200px !important; + overflow-y: scroll !important; + width: 100% !important; +} + +.react-datepicker__time-list-item { + padding: 5px 10px !important; + height: auto !important; +} + +/* 선택된 시간 스타일 */ +.react-datepicker__time-list-item--selected { + background-color: #216ba5 !important; + color: white !important; +} \ No newline at end of file diff --git a/src/components/features/DatePicker/DatePicker.tsx b/src/components/features/DatePicker/DatePicker.tsx new file mode 100644 index 0000000..3577c96 --- /dev/null +++ b/src/components/features/DatePicker/DatePicker.tsx @@ -0,0 +1,180 @@ +import { forwardRef, InputHTMLAttributes } from "react"; +import styled from "@emotion/styled"; +import DatePicker from "react-datepicker"; +import { Global, css } from "@emotion/react"; +import "react-datepicker/dist/react-datepicker.css"; + +// StyledInput 스타일 정의 +const StyledInput = styled.input` + width: 100%; + padding: 12px; + margin-bottom: 10px; + border: 1px solid #ccc; + border-radius: 8px; + font-size: 1rem; + &:focus { + outline: none; + border-color: #6c63ff; + box-shadow: 0 0 0 2px rgba(108, 99, 255, 0.3); + } +`; + +// 커스텀 입력 컴포넌트 타입 정의 +interface CustomDateInputProps extends InputHTMLAttributes { + onClick?: () => void; + value?: string; +} + +// DatePicker의 커스텀 입력 컴포넌트 +const CustomDateInput = forwardRef( + ({ value, onClick, placeholder }, ref) => ( + + ), +); + +interface ReactDatePickerProps { + placeholderText: string; + onDateChange: (date: Date | null) => void; + selectedDate?: Date | null; + showTimeSelect?: boolean; // 시간 선택 기능 + dateFormat?: string; // 날짜 포맷 설정 +} + +const ReactDatePicker = ({ + placeholderText, + onDateChange, + selectedDate, + showTimeSelect = false, + dateFormat = "yyyy/MM/dd", +}: ReactDatePickerProps) => { + const handleDateChange = (date: Date | null) => { + if (date) { + const utcDate = new Date( + date.getTime() - date.getTimezoneOffset() * 60000, + ); + onDateChange(utcDate); + } else { + onDateChange(null); + } + }; + + return ( + <> + + } + /> + + ); +}; + +export default ReactDatePicker; diff --git a/src/main.tsx b/src/main.tsx index af59181..ad55e7a 100644 --- a/src/main.tsx +++ b/src/main.tsx @@ -19,13 +19,14 @@ import "./index.css"; // }); if ("serviceWorker" in navigator) { - navigator.serviceWorker.getRegistrations().then((registrations) => { - registrations.forEach((registration) => { - if (registration.active && registration.scope.includes("mock")) { - registration.unregister(); // 서비스 워커 해제 - } + navigator.serviceWorker + .register("/firebase-messaging-sw.js") + .then((registration) => { + console.log("Service Worker 등록 성공:", registration); + }) + .catch((error) => { + console.error("Service Worker 등록 실패:", error); }); - }); } ReactDOM.createRoot(document.getElementById("root")!).render( diff --git a/src/pages/Main/MainPage.tsx b/src/pages/Main/MainPage.tsx index 400af6c..d34cb5a 100644 --- a/src/pages/Main/MainPage.tsx +++ b/src/pages/Main/MainPage.tsx @@ -1,4 +1,3 @@ -// mainPage.tsx import { useEffect, useRef } from "react"; import styled from "@emotion/styled"; import { useNavigate, useLocation } from "react-router-dom"; @@ -9,13 +8,15 @@ import { useGetPlans } from "@/api/hooks/useGetPlans"; import useCreatePlan from "@/api/hooks/useCreatePlan"; import Button from "@/components/common/Button/Button"; import RouterPath from "@/router/RouterPath"; +import { requestForToken, setupOnMessageListener } from "@/api/firebaseConfig"; // Import Firebase functions +import { apiClient } from "@/api/instance"; const PageContainer = styled.div` background-color: #ffffff; padding: 20px; `; -export default function MainlPage() { +export default function MainPage() { const location = useLocation(); const { data: Plans, isLoading, error, refetch } = useGetPlans(); const savePlanMutation = useCreatePlan(); @@ -30,7 +31,7 @@ export default function MainlPage() { }, [location, refetch]); useEffect(() => { - if (hasMounted.current) return; // strict모드로 인한 두 번 실행을 막기위해 + if (hasMounted.current) return; hasMounted.current = true; const savePlans = async () => { @@ -45,24 +46,26 @@ export default function MainlPage() { ); try { - for (const plan of parsedPlans) { - await savePlanMutation.mutateAsync({ - plan: { - title: plan.title, - description: plan.description, - startDate: plan.start.toISOString(), - endDate: plan.end.toISOString(), - accessibility: plan.accessibility ?? true, - isCompleted: plan.complete ?? false, - }, - }); - } + await Promise.all( + parsedPlans.map((plan) => + savePlanMutation.mutateAsync({ + plan: { + title: plan.title, + description: plan.description, + startDate: plan.start.toISOString(), + endDate: plan.end.toISOString(), + accessibility: plan.accessibility ?? true, + isCompleted: plan.complete ?? false, + }, + }), + ), + ); sessionStorage.removeItem("plans"); console.log("세션의 플랜이 저장되었습니다."); - isPlanSaved.current = true; // Set flag after saving + isPlanSaved.current = true; refetch(); - } catch (error) { - console.error("세션의 플랜 저장 실패:", error); + } catch (err) { + console.error("세션의 플랜 저장 실패:", err); } } }; @@ -71,6 +74,29 @@ export default function MainlPage() { // eslint-disable-next-line react-hooks/exhaustive-deps }, []); + // Notification functionality + useEffect(() => { + const registerFcmToken = async () => { + const permission = await Notification.requestPermission(); + if (permission === "granted") { + try { + const fcmToken = await requestForToken(); + if (fcmToken) { + await apiClient.post("/api/fcm/register", { token: fcmToken }); + console.log("FCM 토큰이 성공적으로 등록되었습니다."); + } + } catch (err) { + console.error("FCM 토큰 등록 중 오류 발생:", err); + } + } else { + console.log("알림 권한이 거부되었습니다."); + } + }; + + registerFcmToken(); + setupOnMessageListener(); // Set up the listener for foreground messages + }, []); + const handleModifyClick = () => { navigate(RouterPath.MAIN_MODIFY, { state: { plans: Plans } }); }; diff --git a/src/pages/Main/MainPageModify.tsx b/src/pages/Main/MainPageModify.tsx index 20a90c7..3320620 100644 --- a/src/pages/Main/MainPageModify.tsx +++ b/src/pages/Main/MainPageModify.tsx @@ -1,16 +1,48 @@ import { useState } from "react"; import styled from "@emotion/styled"; import { useLocation, useNavigate } from "react-router-dom"; +import ReactDatePicker from "@/components/features/DatePicker/DatePicker"; import CustomCalendar, { CalendarEvent, } from "@/components/features/CustomCalendar/CustomCalendar"; -import useModifyPlan from "@/api/hooks/useModifyPlans"; +import { apiClient } from "@/api/instance"; import useCreatePlan from "@/api/hooks/useCreatePlan"; import useDeletePlan from "@/api/hooks/useDeletePlan"; import Button from "@/components/common/Button/Button"; import Modal from "@/components/common/Modal/Modal"; import RouterPath from "@/router/RouterPath"; +const ModalContainer = styled.div` + padding: 20px; + background-color: white; + border-radius: 12px; + display: flex; + flex-direction: column; + align-items: center; + gap: 15px; +`; + +const Title = styled.h2` + font-size: 1.8rem; + font-weight: bold; + color: #333; + margin-bottom: 20px; +`; + +const StyledInput = styled.input` + width: 100%; + padding: 12px; + margin-bottom: 10px; + border: 1px solid #ccc; + border-radius: 8px; + font-size: 1rem; + &:focus { + outline: none; + border-color: #6c63ff; + box-shadow: 0 0 0 2px rgba(108, 99, 255, 0.3); + } +`; + const PageContainer = styled.div` max-width: 1200px; margin: 0 auto; @@ -39,7 +71,7 @@ export default function PlanModifyPage() { }); const navigate = useNavigate(); - const { mutate: modifyPlan } = useModifyPlan(); + const [pendingPlans, setPendingPlans] = useState(false); const { mutate: createPlan } = useCreatePlan(); const { mutate: deletePlan } = useDeletePlan(); @@ -54,8 +86,8 @@ export default function PlanModifyPage() { accessibility, isCompleted, } = newPlanData; - const utcStartDate = new Date(startDate).toISOString(); - const utcEndDate = new Date(endDate).toISOString(); + const utcStartDate = new Date(`${startDate}Z`).toISOString(); + const utcEndDate = new Date(`${endDate}Z`).toISOString(); createPlan( { @@ -76,9 +108,9 @@ export default function PlanModifyPage() { ...modifiedPlans, { ...newPlanData, - id: newPlanId, // 응답 데이터의 id 사용 - start: new Date(startDate), - end: new Date(endDate), + id: newPlanId, + start: new Date(utcStartDate), + end: new Date(utcEndDate), complete: isCompleted, }, ]); @@ -120,23 +152,30 @@ export default function PlanModifyPage() { }; const handleSaveAll = () => { - modifiedPlans.forEach((plan) => { - if (plan.id && !Number.isNaN(Number(plan.id))) { - modifyPlan({ - planId: Number(plan.id), - planData: { + setPendingPlans(true); + Promise.all( + modifiedPlans + .filter((plan) => plan.id && !Number.isNaN(Number(plan.id))) + .map((plan) => + apiClient.put(`/api/plans/${plan.id}`, { title: plan.title, description: plan.description, - startDate: plan.start.toISOString(), - endDate: plan.end.toISOString(), + startDate: new Date(plan.start).toISOString(), + endDate: new Date(plan.end).toISOString(), accessibility: plan.accessibility ?? true, isCompleted: plan.complete ?? false, - }, - }); - } - }); - alert("수정사항이 저장되었습니다."); - navigate(RouterPath.MAIN, { state: { refetchNeeded: true } }); + }), + ), + ) + .then(() => { + alert("수정사항이 저장되었습니다."); + setPendingPlans(false); + navigate(RouterPath.MAIN, { state: { refetchNeeded: true } }); + }) + .catch((error) => { + alert(`저장 중 오류 발생: ${error.message}`); + setPendingPlans(false); + }); }; return ( @@ -148,6 +187,7 @@ export default function PlanModifyPage() { onPlanChange={handlePlanChange} onDeletePlan={handleDeletePlan} /> + {pendingPlans &&

저장 중...

} + + 새로운 플랜 추가 + + setNewPlanData({ ...newPlanData, title: e.target.value }) + } + /> + + setNewPlanData({ ...newPlanData, description: e.target.value }) + } + /> + + setNewPlanData((prevData) => ({ + ...prevData, + startDate: date ? date.toISOString().slice(0, 16) : "", + })) + } + showTimeSelect + dateFormat="yyyy/MM/dd HH:mm" + /> + + setNewPlanData((prevData) => ({ + ...prevData, + endDate: date ? date.toISOString().slice(0, 16) : "", + })) + } + showTimeSelect + dateFormat="yyyy/MM/dd HH:mm" + /> + + )}