diff --git a/README.md b/README.md
index 0a3dbf5..5696ffd 100644
--- a/README.md
+++ b/README.md
@@ -1,11 +1,6 @@
# 🎉 Eventique – Event Management Website
-Welcome to **Eventique**, a stylish and modern event management platform built using React + Vite + Tailwind CSS + Firebase.
-Live Demo: 👉 [https://neweventique.netlify.app](https://neweventique.netlify.app)
-
----
-
## 🌟 Overview
Eventique is a fast, elegant, and responsive event management frontend built with modern tools.
@@ -56,35 +51,4 @@ npm install
npm run dev
```
----
-
-## 📸 Screenshots
-
-### 🔹 Home page
-
-
-### 🔹 Home page Logged In
-
-
-### 🔹 Services
-
-
---
-
-## 🌐 Live Demo
-👉 [https://neweventique.netlify.app](https://neweventique.netlify.app)
-
----
-
-## 🙌 Acknowledgements
-
-Built with ❤️ using React, Vite, and Tailwind CSS as part of a personal web development project.
-
----
-
-## ⭐ Show Your Support
-
-If you like this project, give it a ⭐ on GitHub or share it!
-
----
diff --git a/package-lock.json b/package-lock.json
index f7b6883..abd8012 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -8,7 +8,10 @@
"name": "event-management-ui",
"version": "0.0.0",
"dependencies": {
+ "axios": "^1.13.2",
"firebase": "^12.0.0",
+ "jwt-decode": "^4.0.0",
+ "lucide-react": "^0.556.0",
"react": "^19.1.0",
"react-dom": "^19.1.0",
"react-hot-toast": "^2.5.2",
@@ -699,6 +702,7 @@
"resolved": "https://registry.npmjs.org/@firebase/app/-/app-0.14.0.tgz",
"integrity": "sha512-APIAeKvRNFWKJLjIL8wLDjh7u8g6ZjaeVmItyqSjCdEkJj14UuVlus74D8ofsOMWh45HEwxwkd96GYbi+CImEg==",
"license": "Apache-2.0",
+ "peer": true,
"dependencies": {
"@firebase/component": "0.7.0",
"@firebase/logger": "0.5.0",
@@ -765,6 +769,7 @@
"resolved": "https://registry.npmjs.org/@firebase/app-compat/-/app-compat-0.5.0.tgz",
"integrity": "sha512-nUnNpOeRj0KZzVzHsyuyrmZKKHfykZ8mn40FtG28DeSTWeM5b/2P242Va4bmQpJsy5y32vfv50+jvdckrpzy7Q==",
"license": "Apache-2.0",
+ "peer": true,
"dependencies": {
"@firebase/app": "0.14.0",
"@firebase/component": "0.7.0",
@@ -780,7 +785,8 @@
"version": "0.9.3",
"resolved": "https://registry.npmjs.org/@firebase/app-types/-/app-types-0.9.3.tgz",
"integrity": "sha512-kRVpIl4vVGJ4baogMDINbyrIOtOxqhkZQg4jTq3l8Lw6WSk0xfpEYzezFu+Kl4ve4fbPl79dvwRtaFqAC/ucCw==",
- "license": "Apache-2.0"
+ "license": "Apache-2.0",
+ "peer": true
},
"node_modules/@firebase/auth": {
"version": "1.11.0",
@@ -1231,6 +1237,7 @@
"integrity": "sha512-0AZUyYUfpMNcztR5l09izHwXkZpghLgCUaAGjtMwXnCg3bj4ml5VgiwqOMOxJ+Nw4qN/zJAaOQBcJ7KGkWStqQ==",
"hasInstallScript": true,
"license": "Apache-2.0",
+ "peer": true,
"dependencies": {
"tslib": "^2.1.0"
},
@@ -2053,6 +2060,7 @@
"integrity": "sha512-AwAfQ2Wa5bCx9WP8nZL2uMZWod7J7/JSplxbTmBQ5ms6QpqNYm672H0Vu9ZVKVngQ+ii4R/byguVEUZQyeg44g==",
"dev": true,
"license": "MIT",
+ "peer": true,
"dependencies": {
"csstype": "^3.0.2"
}
@@ -2087,6 +2095,7 @@
"integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==",
"dev": true,
"license": "MIT",
+ "peer": true,
"bin": {
"acorn": "bin/acorn"
},
@@ -2197,6 +2206,12 @@
"dev": true,
"license": "Python-2.0"
},
+ "node_modules/asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+ "license": "MIT"
+ },
"node_modules/autoprefixer": {
"version": "10.4.21",
"resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.21.tgz",
@@ -2235,6 +2250,17 @@
"postcss": "^8.1.0"
}
},
+ "node_modules/axios": {
+ "version": "1.13.2",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.13.2.tgz",
+ "integrity": "sha512-VPk9ebNqPcy5lRGuSlKx752IlDatOjT9paPlm8A7yOuW2Fbvp4X3JznJtT4f0GzGLLiWE9W8onz51SqLYwzGaA==",
+ "license": "MIT",
+ "dependencies": {
+ "follow-redirects": "^1.15.6",
+ "form-data": "^4.0.4",
+ "proxy-from-env": "^1.1.0"
+ }
+ },
"node_modules/balanced-match": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
@@ -2299,6 +2325,7 @@
}
],
"license": "MIT",
+ "peer": true,
"dependencies": {
"caniuse-lite": "^1.0.30001726",
"electron-to-chromium": "^1.5.173",
@@ -2312,6 +2339,19 @@
"node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
}
},
+ "node_modules/call-bind-apply-helpers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
"node_modules/callsites": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
@@ -2498,6 +2538,18 @@
"integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
"license": "MIT"
},
+ "node_modules/combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "license": "MIT",
+ "dependencies": {
+ "delayed-stream": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
"node_modules/commander": {
"version": "4.1.1",
"resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz",
@@ -2556,7 +2608,8 @@
"version": "3.1.3",
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz",
"integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==",
- "license": "MIT"
+ "license": "MIT",
+ "peer": true
},
"node_modules/debug": {
"version": "4.4.1",
@@ -2583,6 +2636,15 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
"node_modules/didyoumean": {
"version": "1.2.2",
"resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz",
@@ -2597,6 +2659,20 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/dunder-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "gopd": "^1.2.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
"node_modules/eastasianwidth": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz",
@@ -2618,6 +2694,51 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/es-define-property": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-errors": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-object-atoms": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-set-tostringtag": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
+ "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "get-intrinsic": "^1.2.6",
+ "has-tostringtag": "^1.0.2",
+ "hasown": "^2.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
"node_modules/esbuild": {
"version": "0.25.5",
"resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.5.tgz",
@@ -2687,6 +2808,7 @@
"integrity": "sha512-zmxXPNMOXmwm9E0yQLi5uqXHs7uq2UIiqEKo3Gq+3fwo1XrJ+hijAZImyF7hclW3E6oHz43Yk3RP8at6OTKflQ==",
"dev": true,
"license": "MIT",
+ "peer": true,
"dependencies": {
"@eslint-community/eslint-utils": "^4.2.0",
"@eslint-community/regexpp": "^4.12.1",
@@ -3047,6 +3169,26 @@
"dev": true,
"license": "ISC"
},
+ "node_modules/follow-redirects": {
+ "version": "1.15.11",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.11.tgz",
+ "integrity": "sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ==",
+ "funding": [
+ {
+ "type": "individual",
+ "url": "https://github.com/sponsors/RubenVerborgh"
+ }
+ ],
+ "license": "MIT",
+ "engines": {
+ "node": ">=4.0"
+ },
+ "peerDependenciesMeta": {
+ "debug": {
+ "optional": true
+ }
+ }
+ },
"node_modules/foreground-child": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz",
@@ -3064,6 +3206,22 @@
"url": "https://github.com/sponsors/isaacs"
}
},
+ "node_modules/form-data": {
+ "version": "4.0.5",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz",
+ "integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==",
+ "license": "MIT",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "es-set-tostringtag": "^2.1.0",
+ "hasown": "^2.0.2",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
"node_modules/fraction.js": {
"version": "4.3.7",
"resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.3.7.tgz",
@@ -3097,7 +3255,6 @@
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
"integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
- "dev": true,
"license": "MIT",
"funding": {
"url": "https://github.com/sponsors/ljharb"
@@ -3112,6 +3269,43 @@
"node": "6.* || 8.* || >= 10.*"
}
},
+ "node_modules/get-intrinsic": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.2",
+ "es-define-property": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "es-object-atoms": "^1.1.1",
+ "function-bind": "^1.1.2",
+ "get-proto": "^1.0.1",
+ "gopd": "^1.2.0",
+ "has-symbols": "^1.1.0",
+ "hasown": "^2.0.2",
+ "math-intrinsics": "^1.1.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/get-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
+ "license": "MIT",
+ "dependencies": {
+ "dunder-proto": "^1.0.1",
+ "es-object-atoms": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
"node_modules/glob": {
"version": "10.4.5",
"resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz",
@@ -3194,6 +3388,18 @@
"csstype": "^3.0.10"
}
},
+ "node_modules/gopd": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
"node_modules/has-flag": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
@@ -3204,11 +3410,37 @@
"node": ">=8"
}
},
+ "node_modules/has-symbols": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-tostringtag": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
+ "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
+ "license": "MIT",
+ "dependencies": {
+ "has-symbols": "^1.0.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
"node_modules/hasown": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
"integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
- "dev": true,
"license": "MIT",
"dependencies": {
"function-bind": "^1.1.2"
@@ -3366,6 +3598,7 @@
"integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==",
"dev": true,
"license": "MIT",
+ "peer": true,
"bin": {
"jiti": "bin/jiti.js"
}
@@ -3404,6 +3637,15 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/jwt-decode": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-4.0.0.tgz",
+ "integrity": "sha512-+KJGIyHgkGuIq3IEBNftfhW/LfWhXUIY6OmyVWjliu5KH1y0fw7VQ8YndE2O4qZdMSd9SqbnC8GOcZEy0Om7sA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ }
+ },
"node_modules/keyv": {
"version": "4.5.4",
"resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
@@ -3490,6 +3732,24 @@
"dev": true,
"license": "ISC"
},
+ "node_modules/lucide-react": {
+ "version": "0.556.0",
+ "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-0.556.0.tgz",
+ "integrity": "sha512-iOb8dRk7kLaYBZhR2VlV1CeJGxChBgUthpSP8wom9jfj79qovgG6qcSdiy6vkoREKPnbUYzJsCn4o4PtG3Iy+A==",
+ "license": "ISC",
+ "peerDependencies": {
+ "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/math-intrinsics": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
"node_modules/merge2": {
"version": "1.4.1",
"resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
@@ -3527,6 +3787,27 @@
"url": "https://github.com/sponsors/jonschlinkert"
}
},
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "license": "MIT",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
"node_modules/minimatch": {
"version": "3.1.2",
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
@@ -3769,6 +4050,7 @@
"integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
"dev": true,
"license": "MIT",
+ "peer": true,
"engines": {
"node": ">=12"
},
@@ -3816,6 +4098,7 @@
}
],
"license": "MIT",
+ "peer": true,
"dependencies": {
"nanoid": "^3.3.11",
"picocolors": "^1.1.1",
@@ -3980,6 +4263,12 @@
"node": ">=12.0.0"
}
},
+ "node_modules/proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+ "license": "MIT"
+ },
"node_modules/punycode": {
"version": "2.3.1",
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
@@ -4016,6 +4305,7 @@
"resolved": "https://registry.npmjs.org/react/-/react-19.1.0.tgz",
"integrity": "sha512-FS+XFBNvn3GTAWq26joslQgWNoFu08F4kl0J4CgdNKADkdSGXQyTCnKteIAJy96Br6YbpEU1LSzV5dYtjMkMDg==",
"license": "MIT",
+ "peer": true,
"engines": {
"node": ">=0.10.0"
}
@@ -4025,6 +4315,7 @@
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.1.0.tgz",
"integrity": "sha512-Xs1hdnE+DyKgeHJeJznQmYMIBG3TKIHJJT95Q58nHLSrElKlGQqDTR2HQ9fx5CN/Gk6Vh/kupBTDLU11/nDk/g==",
"license": "MIT",
+ "peer": true,
"dependencies": {
"scheduler": "^0.26.0"
},
@@ -4687,6 +4978,7 @@
"integrity": "sha512-hxdyZDY1CM6SNpKI4w4lcUc3Mtkd9ej4ECWVHSMrOdSinVc2zYOAppHeGc/hzmRo3pxM5blMzkuWHOJA/3NiFw==",
"dev": true,
"license": "MIT",
+ "peer": true,
"dependencies": {
"esbuild": "^0.25.0",
"fdir": "^6.4.6",
diff --git a/package.json b/package.json
index 8b8e760..a536756 100644
--- a/package.json
+++ b/package.json
@@ -10,7 +10,10 @@
"preview": "vite preview"
},
"dependencies": {
+ "axios": "^1.13.2",
"firebase": "^12.0.0",
+ "jwt-decode": "^4.0.0",
+ "lucide-react": "^0.556.0",
"react": "^19.1.0",
"react-dom": "^19.1.0",
"react-hot-toast": "^2.5.2",
diff --git a/public/icono.png b/public/icono.png
new file mode 100644
index 0000000..39b2eb8
Binary files /dev/null and b/public/icono.png differ
diff --git a/public/logo.png b/public/logo.png
index c6b6b10..9c50812 100644
Binary files a/public/logo.png and b/public/logo.png differ
diff --git a/public/servicesData.json b/public/servicesData.json
index 273badb..2241f89 100644
--- a/public/servicesData.json
+++ b/public/servicesData.json
@@ -70,14 +70,5 @@
"cost": "Rs 22,000 onwards",
"location": "Pune, Maharashtra",
"duration": "3-5 hours"
- },
- {
- "id": 9,
- "image": "https://dittonmanor.com/wp-content/uploads/2024/09/how-to-plan-a-christmas-party.jpg",
- "title": "Festive Theme Parties",
- "description": "Celebrate festivals like Diwali, Christmas, or Holi with custom-themed decorations, joyful activities, snacks, and giveaways for guests.",
- "cost": "Rs 30,000 onwards",
- "location": "Ahmedabad, Gujarat",
- "duration": "Evening"
}
]
diff --git a/src/App.css b/src/App.css
index e69de29..fec81a4 100644
--- a/src/App.css
+++ b/src/App.css
@@ -0,0 +1,15 @@
+/* Animación para el snackbar */
+@keyframes fadeInDown {
+ from {
+ opacity: 0;
+ transform: translateY(-20px);
+ }
+ to {
+ opacity: 1;
+ transform: translateY(0);
+ }
+}
+
+.animate-fade-in-down {
+ animation: fadeInDown 0.3s ease-out;
+}
\ No newline at end of file
diff --git a/src/App.jsx b/src/App.jsx
index 1f9b4d7..e25ffd7 100644
--- a/src/App.jsx
+++ b/src/App.jsx
@@ -1,38 +1,133 @@
-import { BrowserRouter, Routes, Route } from 'react-router-dom';
+import { Routes, Route } from 'react-router-dom';
import Categories from './pages/Categories';
import Contact from './pages/Contact';
import Layout from './components/Layout';
import Home from './pages/Home';
-import Services from './pages/Services';
import SignIn from './pages/SignIn';
import EventDetails from './pages/EventDetails';
import SignUp from './pages/SignUp';
import Profile from './pages/Profile';
import Cart from './pages/Cart';
+import ClientRoute from './components/ClienteRoute';
+
+import AdminRoute from './components/AdminRoute';
+import AdminLayout from './pages/admin/Layout';
+import AdminDashboard from './pages/admin/dashboard';
+import EventsList from './pages/admin/events/index';
+import CreateEvent from './pages/admin/events/create';
+import EditEvent from './pages/admin/events/edit';
+
+import CreateEstablishment from './pages/admin/establishments/create';
+import EstablishmentsList from './pages/admin/establishments/index';
+import EditEstablishment from './pages/admin/establishments/edit';
+
+import CreateEntertainment from './pages/admin/entertainment/create';
+import EntertainmentsList from './pages/admin/entertainment/index';
+import EditEntertainment from './pages/admin/entertainment/edit';
+
+
+import CreateCatering from './pages/admin/catering/create';
+import CateringsList from './pages/admin/catering/index';
+import EditCatering from './pages/admin/catering/edit';
+
+import AdditionalList from './pages/admin/additional/index';
+import CreateAdditional from './pages/admin/additional/create';
+import EditAdditional from './pages/admin/additional/edit';
+
+import DecorationList from './pages/admin/decoration/index';
+import CreateDecoration from './pages/admin/decoration/create';
+import EditDecoration from './pages/admin/decoration/edit';
+
+import ClientList from './pages/admin/client/clientelist';
+
+import Services from "./pages/Services";
+import CategoryServices from "./pages/CategoriesServices";
+
+import ReservationForm from './pages/ReservationForm';
+
+
import './App.css'
function App() {
-
- return(
-
-
+ return (
+
- }/>
- }/>
- }/>
- }/>
- }/>
- }/>
- }/>
- }/>
- }/>
+ {/* Rutas públicas */}
+ } />
+ } />
+
+ } />
+ } />
+
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+
+ {/* Ruta de reserva solo para clientes */}
+
+
+
+ }
+ />
+
+ {/* Rutas admin*/}
+
+
+
+ }
+ >
+
+ } />
+ } />
+
+ {/* Rutas para Eventos */}
+ } />
+ } />
+ } />
+
+ {/* Rutas para Establecimientos */}
+ } />
+ } />
+ } />
+
+ {/* Rutas para Entretenimiento */}
+ } />
+ } />
+ } />
+
+ {/* Rutas para Catering */}
+ } />
+ } />
+ } />
+
+ {/* Rutas para Adicionales */}
+ } />
+ } />
+ } />
+
+
+ {/* Rutas para decoración */}
+ } />
+ } />
+ } />
+
+ {/* Rutas para obtener clientes */}
+ } />
+
-
-
- )
-
+
+ );
}
-export default App
+export default App;
\ No newline at end of file
diff --git a/src/api/additional.js b/src/api/additional.js
new file mode 100644
index 0000000..b56e149
--- /dev/null
+++ b/src/api/additional.js
@@ -0,0 +1,51 @@
+import api from './axios';
+
+export const getAdditionals = async () => {
+ try {
+ const response = await api.get('/additional');
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching additionals:", error);
+ throw error;
+ }
+ };
+
+export const getAdditionalById = async (id) => {
+ try {
+ const response = await api.get(`/additional/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching additionals:", error);
+ throw error;
+ }
+};
+
+export const createAdditional = async (additionalData) => {
+ try {
+ const response = await api.post('/additional', additionalData);
+ return response.data;
+ } catch (error) {
+ console.error("Error creating additional:", error);
+ throw error;
+ }
+};
+
+export const updateAdditional = async (id, additionalData) => {
+ try {
+ const response = await api.put(`/additional/${id}`, additionalData);
+ return response.data;
+ } catch (error) {
+ console.error("Error updating additional:", error);
+ throw error;
+ }
+};
+
+export const deleteAdditional = async (id) => {
+ try {
+ const response = await api.delete(`/additionals/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error("Error deleting additional:", error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/api/auth.js b/src/api/auth.js
new file mode 100644
index 0000000..11008b9
--- /dev/null
+++ b/src/api/auth.js
@@ -0,0 +1,7 @@
+import api from './axios';
+
+export const loginRequest = (email, password) =>
+ api.post('/User/login', { email, password });
+
+export const registerRequest = (userData) =>
+ api.post('/User/register', userData);
diff --git a/src/api/axios.js b/src/api/axios.js
new file mode 100644
index 0000000..060bd0d
--- /dev/null
+++ b/src/api/axios.js
@@ -0,0 +1,46 @@
+import axios from 'axios';
+
+const API_BASE_URL = 'http://localhost:8081';
+
+const api = axios.create({
+ baseURL: API_BASE_URL,
+ timeout: 10000,
+ headers: {
+ 'Content-Type': 'application/json',
+ }
+});
+
+// Interceptor para el token
+api.interceptors.request.use(
+ (config) => {
+ const token = localStorage.getItem('token');
+ if (token) {
+ config.headers.Authorization = `Bearer ${token}`;
+ console.log('🔐 Token agregado automáticamente');
+ }
+ return config;
+ },
+ (error) => {
+ console.error('❌ Error en interceptor:', error);
+ return Promise.reject(error);
+ }
+);
+
+// Interceptor para errores
+api.interceptors.response.use(
+ (response) => response,
+ (error) => {
+ if (error.response?.status === 401) {
+ console.log('🔒 Token inválido/vencido');
+ localStorage.removeItem('token');
+ localStorage.removeItem('user');
+ window.location.href = '/login';
+ }
+ if (error.response?.status === 403) {
+ console.log('🚫 No tienes permisos para esta acción');
+ }
+ return Promise.reject(error);
+ }
+);
+
+export default api;
\ No newline at end of file
diff --git a/src/api/catering.js b/src/api/catering.js
new file mode 100644
index 0000000..7720c4a
--- /dev/null
+++ b/src/api/catering.js
@@ -0,0 +1,51 @@
+import api from './axios';
+
+export const getCatering = async () => {
+ try {
+ const response = await api.get("/catering");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching catering:", error);
+ throw error;
+ }
+};
+
+export const getCateringById = async (id) => {
+ try {
+ const response = await api.get(`/catering/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error fetching catering with id ${id}:`, error);
+ throw error;
+ }
+};
+
+export const createCatering = async (cateringData) => {
+ try {
+ const response = await api.post("/catering", cateringData);
+ return response.data;
+ } catch (error) {
+ console.error("Error creating catering:", error);
+ throw error;
+ }
+};
+
+export const updateCatering = async (id, cateringData) => {
+ try {
+ const response = await api.put(`/catering/${id}`, cateringData);
+ return response.data;
+ } catch (error) {
+ console.error(`Error updating catering with id ${id}:`, error);
+ throw error;
+ }
+};
+
+export const deleteCatering = async (id) => {
+ try {
+ const response = await api.delete(`/catering/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error deleting catering with id ${id}:`, error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/api/decoration.js b/src/api/decoration.js
new file mode 100644
index 0000000..7d20922
--- /dev/null
+++ b/src/api/decoration.js
@@ -0,0 +1,51 @@
+import api from './axios';
+
+export const getDecorations = async () => {
+ try {
+ const response = await api.get('/decoration');
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
+
+export const getDecorationById = async (id) => {
+ try {
+ const response = await api.get(`/decoration/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
+
+export const createDecoration = async (decorationData) => {
+ try {
+ const response = await api.post('/decoration', decorationData);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
+
+export const updateDecoration = async (id, decorationData) => {
+ try {
+ const response = await api.put(`/decoration/${id}`, decorationData);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
+
+export const deleteDecoration = async (id) => {
+ try {
+ const response = await api.delete(`/decoration/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/api/entertainments.js b/src/api/entertainments.js
new file mode 100644
index 0000000..2458454
--- /dev/null
+++ b/src/api/entertainments.js
@@ -0,0 +1,51 @@
+import api from './axios';
+
+export const getEntertainments = async () => {
+ try {
+ const response = await api.get("/entertainment");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching entertainments:", error);
+ throw error;
+ }
+};
+
+export const getEntertainmentById = async (id) => {
+ try {
+ const response = await api.get(`/entertainment/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error fetching entertainment with id ${id}:`, error);
+ throw error;
+ }
+};
+
+export const createEntertainment = async (entertainmentData) => {
+ try {
+ const response = await api.post("/entertainment", entertainmentData);
+ return response.data;
+ } catch (error) {
+ console.error("Error creating entertainment:", error);
+ throw error;
+ }
+};
+
+export const updateEntertainment = async (id, entertainmentData) => {
+ try {
+ const response = await api.put(`/entertainment/${id}`, entertainmentData);
+ return response.data;
+ } catch (error) {
+ console.error(`Error updating entertainment with id ${id}:`, error);
+ throw error;
+ }
+};
+
+export const deleteEntertainment = async (id) => {
+ try {
+ const response = await api.delete(`/entertainment/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error deleting entertainment with id ${id}:`, error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/api/establishments.js b/src/api/establishments.js
new file mode 100644
index 0000000..91fe0c0
--- /dev/null
+++ b/src/api/establishments.js
@@ -0,0 +1,51 @@
+import api from './axios';
+
+export const getEstablishments = async () => {
+ try {
+ const response = await api.get("/establishments");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching establishments:", error);
+ throw error;
+ }
+};
+
+export const getEstablishmentById = async (id) => {
+ try {
+ const response = await api.get(`/establishments/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error fetching establishment ${id}:`, error);
+ throw error;
+ }
+};
+
+export const createEstablishment = async (establishmentData) => {
+ try {
+ const response = await api.post("/establishments", establishmentData);
+ return response.data;
+ } catch (error) {
+ console.error("Error creating establishment:", error);
+ throw error;
+ }
+};
+
+export const updateEstablishment = async (id, establishmentData) => {
+ try {
+ const response = await api.put(`/establishments/${id}`, establishmentData);
+ return response.data;
+ } catch (error) {
+ console.error(`Error updating establishment ${id}:`, error);
+ throw error;
+ }
+};
+
+export const deleteEstablishment = async (id) => {
+ try {
+ const response = await api.delete(`/establishments/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error deleting establishment ${id}:`, error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/api/events.js b/src/api/events.js
new file mode 100644
index 0000000..38e9819
--- /dev/null
+++ b/src/api/events.js
@@ -0,0 +1,52 @@
+import api from './axios';
+
+export const getEvents = async () => {
+ try {
+ const response = await api.get("/events");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching events:", error);
+ throw error;
+ }
+};
+
+export const getEventById = async (id) => {
+ try {
+ const response = await api.get(`/events/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error fetching event ${id}:`, error);
+ throw error;
+ }
+};
+
+export const createEvent = async (eventData) => {
+ try {
+
+ const response = await api.post("/events", eventData);
+ return response.data;
+ } catch (error) {
+ console.error("Error creating event:", error);
+ throw error;
+ }
+};
+
+export const updateEvent = async (id, eventData) => {
+ try {
+ const response = await api.put(`/events/${id}`, eventData);
+ return response.data;
+ } catch (error) {
+ console.error(`Error updating event ${id}:`, error);
+ throw error;
+ }
+};
+
+export const deleteEvent = async (id) => {
+ try {
+ const response = await api.delete(`/events/${id}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error deleting event ${id}:`, error);
+ throw error;
+ }
+};
diff --git a/src/api/reservation.js b/src/api/reservation.js
new file mode 100644
index 0000000..898cdfc
--- /dev/null
+++ b/src/api/reservation.js
@@ -0,0 +1,25 @@
+import api from './axios';
+
+export const getReservations = async () => {
+ try {
+ const response = await api.get('/reserve');
+ console.log("API Response:", response);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching reservations:", error);
+ throw error;
+ }
+}
+
+export const createReservations = async () => {
+ try {
+ const response = await api.get('/reserve');
+ console.log("API Response:", response);
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching reservations:", error);
+ throw error;
+ }
+
+
+};
\ No newline at end of file
diff --git a/src/api/services.js b/src/api/services.js
new file mode 100644
index 0000000..1592423
--- /dev/null
+++ b/src/api/services.js
@@ -0,0 +1,43 @@
+import api from './axios';
+
+//Obtner catering
+export const getCatering = async () => {
+ try {
+ const response = await api.get("/catering");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching catering:", error);
+ throw error;
+ }
+};
+
+export const getEntertainments = async () => {
+ try {
+ const response = await api.get("/entertainment");
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching entertainments:", error);
+ throw error;
+ }
+};
+
+export const getDecorations = async () => {
+ try {
+ const response = await api.get('/decoration');
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching decoration:", error);
+ throw error;
+ }
+};
+
+export const getAdditionals = async () => {
+ try {
+ const response = await api.get('/additional');
+ return response.data;
+ } catch (error) {
+ console.error("Error fetching additionals:", error);
+ throw error;
+ }
+ };
+
diff --git a/src/api/user.js b/src/api/user.js
new file mode 100644
index 0000000..a98c90f
--- /dev/null
+++ b/src/api/user.js
@@ -0,0 +1,65 @@
+import api from './axios';
+
+
+// Obtener usuarios por tipo (CLIENTE, ADMIN, etc.)
+export const getUsersByType = async (type) => {
+ try {
+ const response = await api.get(`/User/type/${type}`);
+ return response.data;
+ } catch (error) {
+ console.error(`Error fetching users by type ${type}:`, error);
+ throw error;
+ }
+};
+
+// Obtener solo usuarios con type === 'CLIENTE'
+export const getClients = async () => {
+ try {
+ return await getUsersByType('CLIENTE');
+ } catch (error) {
+ console.error("Error fetching clients:", error);
+ throw error;
+ }
+};
+
+
+
+// Obtener pagos del usuario
+export const getUserPayments = async () => {
+ try {
+ const response = await api.get("/payments");
+ return response;
+ } catch (error) {
+ console.error("Error fetching user payments:", error);
+ throw error;
+ }
+};
+
+
+
+//Corregir por endpoints reales
+
+// Obtener perfil del usuario
+export const getUserProfile = async () => {
+ try {
+ const response = await api.get("/api/users/profile");
+ return response;
+ } catch (error) {
+ console.error("Error fetching user profile:", error);
+ throw error;
+ }
+};
+
+
+
+
+// Actualizar perfil del usuario (cambiar por uno real)
+export const updateUserProfile = async (userData) => {
+ try {
+ const response = await api.put("/api/users/profile", userData);
+ return response;
+ } catch (error) {
+ console.error("Error updating user profile:", error);
+ throw error;
+ }
+};
\ No newline at end of file
diff --git a/src/components/AdminRoute.jsx b/src/components/AdminRoute.jsx
new file mode 100644
index 0000000..4beca9c
--- /dev/null
+++ b/src/components/AdminRoute.jsx
@@ -0,0 +1,18 @@
+import { Navigate } from "react-router-dom";
+import { useAuth } from "../hooks/useAuth";
+
+export default function AdminRoute({ children }) {
+ const { user, isAuthenticated } = useAuth();
+
+ // Si no está autenticado, redirigir a login
+ if (!isAuthenticated) {
+ return ;
+ }
+
+
+ if (user?.type !== "ADMIN") {
+ return ;
+ }
+
+ return children;
+}
\ No newline at end of file
diff --git a/src/components/ClienteRoute.jsx b/src/components/ClienteRoute.jsx
new file mode 100644
index 0000000..f728772
--- /dev/null
+++ b/src/components/ClienteRoute.jsx
@@ -0,0 +1,20 @@
+
+import { Navigate } from 'react-router-dom';
+import { useAuth } from '../hooks/useAuth';
+
+export default function ClientRoute({ children }) {
+ const { user, isAuthenticated } = useAuth();
+
+ if (!isAuthenticated) {
+ // Si no está autenticado, redirigir al login
+ return ;
+ }
+
+ // Si está autenticado pero es administrador, redirigir al dashboard de admin
+ if (user?.type === 'ADMIN') {
+ return ;
+ }
+
+ // Si es cliente, mostrar el componente hijo
+ return children;
+}
\ No newline at end of file
diff --git a/src/components/DiscountBanner.jsx b/src/components/DiscountBanner.jsx
index c3c241d..7269139 100644
--- a/src/components/DiscountBanner.jsx
+++ b/src/components/DiscountBanner.jsx
@@ -1,22 +1,29 @@
-// src/components/DiscountBanner.jsx
import { useState, useEffect } from "react";
-import { onAuthStateChanged } from "firebase/auth";
-import { auth } from "../firebase"; // adjust the path if needed
+
const DiscountBanner = () => {
const [showBanner, setShowBanner] = useState(true);
const [isLoggedIn, setIsLoggedIn] = useState(false);
- // Check user login state
useEffect(() => {
- const unsubscribe = onAuthStateChanged(auth, (user) => {
- setIsLoggedIn(!!user); // true if user exists, false otherwise
- });
+ const token = localStorage.getItem("token");
+ // Si hay token, asumimos que está logueado
+ setIsLoggedIn(!!token);
+
+ // Escuchar cambios en el localStorage para actualizar el estado
+ const handleStorageChange = (e) => {
+ if (e.key === "token") {
+ setIsLoggedIn(!!e.newValue);
+ }
+ };
- return () => unsubscribe();
+ window.addEventListener("storage", handleStorageChange);
+
+ return () => {
+ window.removeEventListener("storage", handleStorageChange);
+ };
}, []);
- // Don't render the banner if user is logged in or banner is dismissed
if (!showBanner || isLoggedIn) {
return null;
}
@@ -37,5 +44,4 @@ const DiscountBanner = () => {
);
};
-export default DiscountBanner;
-
+export default DiscountBanner;
\ No newline at end of file
diff --git a/src/components/Footer.jsx b/src/components/Footer.jsx
index 37baecf..11ef362 100644
--- a/src/components/Footer.jsx
+++ b/src/components/Footer.jsx
@@ -50,7 +50,7 @@ export default function Footer() {
{/* Bottom Bar */}
-
© 2025 EVENTIQUE. All rights reserved.
+
© 2025 EVENTIFY. All rights reserved.
diff --git a/src/components/Layout.jsx b/src/components/Layout.jsx
index 0167bb9..a96813b 100644
--- a/src/components/Layout.jsx
+++ b/src/components/Layout.jsx
@@ -1,4 +1,3 @@
-// src/Layout.jsx
import { Outlet } from "react-router-dom";
// import DiscountBanner from "./DiscountBanner";
import Footer from "./Footer";
diff --git a/src/components/Navbar.jsx b/src/components/Navbar.jsx
index 2397a78..7418d1b 100644
--- a/src/components/Navbar.jsx
+++ b/src/components/Navbar.jsx
@@ -1,51 +1,47 @@
-import { auth } from "../firebase";
import { Link, useNavigate } from "react-router-dom";
-import { useEffect, useState } from "react";
-import { onAuthStateChanged, signOut } from "firebase/auth";
+import { useState } from "react";
+import { useAuth } from "../hooks/useAuth";
export default function Navbar() {
- const [user, setUser] = useState(null);
+ const { user, logout, isAuthenticated } = useAuth();
const [isMenuOpen, setIsMenuOpen] = useState(false);
const navigate = useNavigate();
- useEffect(() => {
- if (isMenuOpen) {
+ const toggleMenu = () => {
+ setIsMenuOpen(!isMenuOpen);
+ if (!isMenuOpen) {
document.body.style.overflow = 'hidden';
} else {
document.body.style.overflow = 'unset';
}
- }, [isMenuOpen]);
-
- useEffect(() => {
- const unsubscribe = onAuthStateChanged(auth, (currentUser) => {
- setUser(currentUser);
- });
- return () => unsubscribe();
- }, []);
-
+ };
- const handleLogout = async () => {
- await signOut(auth);
- navigate("/sign-in");
+ const closeMenu = () => {
setIsMenuOpen(false);
+ document.body.style.overflow = 'unset';
};
- const toggleMenu = () => {
- setIsMenuOpen(!isMenuOpen);
+ const handleLogout = () => {
+ logout();
+ navigate("/sign-in");
+ closeMenu();
};
return (