diff --git a/next.config.js b/next.config.js index 03a332a..b9f4bb2 100644 --- a/next.config.js +++ b/next.config.js @@ -2,6 +2,7 @@ module.exports = { + typescript: { // !! WARN !! // Dangerously allow production builds to successfully complete even if @@ -9,12 +10,13 @@ module.exports = { // !! WARN !! ignoreBuildErrors: true, }, - ignoreBuildErrors: true, eslint: { ignoreDuringBuilds: true, }, - typescript: { - ignoreBuildErrors: true, + i18n: { + locales: ['en'], // List all the languages you want to support + defaultLocale: 'en', // The default language of the website + localeDetection: true, // Enable or disable automatic locale detection based on browser settings }, images: { remotePatterns: [ diff --git a/package-lock.json b/package-lock.json index 929369c..91b5516 100644 --- a/package-lock.json +++ b/package-lock.json @@ -26,6 +26,7 @@ "google-map-react": "^2.2.1", "lodash": "^4.17.21", "next": "^13.4.3", + "next-i18next": "^15.3.1", "rc-slider": "^10.1.1", "react": "^18.2.0", "react-datepicker": "^4.11.0", @@ -59,12 +60,11 @@ } }, "node_modules/@babel/runtime": { - "version": "7.21.5", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.21.5.tgz", - "integrity": "sha512-8jI69toZqqcsnqGGqwGS4Qb1VwLOEp4hz+CXPywcvjs60u3B4Pom/U/7rm4W8tMOYEB+E9wgD0mW1l3r8qlI9Q==", - "license": "MIT", + "version": "7.25.6", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.6.tgz", + "integrity": "sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ==", "dependencies": { - "regenerator-runtime": "^0.13.11" + "regenerator-runtime": "^0.14.0" }, "engines": { "node": ">=6.9.0" @@ -579,6 +579,15 @@ "@types/react": "*" } }, + "node_modules/@types/hoist-non-react-statics": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.5.tgz", + "integrity": "sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg==", + "dependencies": { + "@types/react": "*", + "hoist-non-react-statics": "^3.3.0" + } + }, "node_modules/@types/js-cookie": { "version": "2.2.7", "resolved": "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-2.2.7.tgz", @@ -1348,6 +1357,16 @@ "toggle-selection": "^1.0.6" } }, + "node_modules/core-js": { + "version": "3.38.1", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.38.1.tgz", + "integrity": "sha512-OP35aUorbU3Zvlx7pjsFdu1rGNnD4pgw/CWoYzRY3t2EzoVT7shKHY1dlAy3f41cGIO7ZDPQimhGFTlEYkG/Hw==", + "hasInstallScript": true, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, "node_modules/cross-spawn": { "version": "7.0.3", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", @@ -2679,6 +2698,23 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/hoist-non-react-statics": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz", + "integrity": "sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==", + "dependencies": { + "react-is": "^16.7.0" + } + }, + "node_modules/html-parse-stringify": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/html-parse-stringify/-/html-parse-stringify-3.0.1.tgz", + "integrity": "sha512-KknJ50kTInJ7qIScF3jeaFRpMpE8/lfiTdzf/twXyPBLAGrLRTmkz3AdTnKeh40X8k9L2fdYwEp/42WGXIRGcg==", + "peer": true, + "dependencies": { + "void-elements": "3.1.0" + } + }, "node_modules/human-signals": { "version": "4.3.1", "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-4.3.1.tgz", @@ -2694,6 +2730,34 @@ "integrity": "sha512-ygGZLjmXfPHj+ZWh6LwbC37l43MhfztxetbFCoYTM2VjkIUpeHgSNn7QIyVFj7YQ1Wl9Cbw5sholVJPzWvC2MQ==", "license": "BSD-3-Clause" }, + "node_modules/i18next": { + "version": "23.15.1", + "resolved": "https://registry.npmjs.org/i18next/-/i18next-23.15.1.tgz", + "integrity": "sha512-wB4abZ3uK7EWodYisHl/asf8UYEhrI/vj/8aoSsrj/ZDxj4/UXPOa1KvFt1Fq5hkUHquNqwFlDprmjZ8iySgYA==", + "funding": [ + { + "type": "individual", + "url": "https://locize.com" + }, + { + "type": "individual", + "url": "https://locize.com/i18next.html" + }, + { + "type": "individual", + "url": "https://www.i18next.com/how-to/faq#i18next-is-awesome.-how-can-i-support-the-project" + } + ], + "peer": true, + "dependencies": { + "@babel/runtime": "^7.23.2" + } + }, + "node_modules/i18next-fs-backend": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/i18next-fs-backend/-/i18next-fs-backend-2.3.2.tgz", + "integrity": "sha512-LIwUlkqDZnUI8lnUxBnEj8K/FrHQTT/Sc+1rvDm9E8YvvY5YxzoEAASNx+W5M9DfD5s77lI5vSAFWeTp26B/3Q==" + }, "node_modules/ignore": { "version": "5.2.4", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz", @@ -3548,6 +3612,41 @@ } } }, + "node_modules/next-i18next": { + "version": "15.3.1", + "resolved": "https://registry.npmjs.org/next-i18next/-/next-i18next-15.3.1.tgz", + "integrity": "sha512-+pa2pZJb7B6k5PKW3TLVMmAodqkNaOBWVYlpWX56mgcEJz0UMW+MKSdKM9Z72CHp6Bp48g7OWwDnLqxXNp/84w==", + "funding": [ + { + "type": "individual", + "url": "https://locize.com/i18next.html" + }, + { + "type": "individual", + "url": "https://www.i18next.com/how-to/faq#i18next-is-awesome.-how-can-i-support-the-project" + }, + { + "type": "individual", + "url": "https://locize.com" + } + ], + "dependencies": { + "@babel/runtime": "^7.23.2", + "@types/hoist-non-react-statics": "^3.3.4", + "core-js": "^3", + "hoist-non-react-statics": "^3.3.2", + "i18next-fs-backend": "^2.3.2" + }, + "engines": { + "node": ">=14" + }, + "peerDependencies": { + "i18next": ">= 23.7.13", + "next": ">= 12.0.0", + "react": ">= 17.0.2", + "react-i18next": ">= 13.5.0" + } + }, "node_modules/next/node_modules/postcss": { "version": "8.4.14", "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.14.tgz", @@ -4213,6 +4312,28 @@ "react": ">=16.8.0" } }, + "node_modules/react-i18next": { + "version": "15.0.2", + "resolved": "https://registry.npmjs.org/react-i18next/-/react-i18next-15.0.2.tgz", + "integrity": "sha512-z0W3/RES9Idv3MmJUcf0mDNeeMOUXe+xoL0kPfQPbDoZHmni/XsIoq5zgT2MCFUiau283GuBUK578uD/mkAbLQ==", + "peer": true, + "dependencies": { + "@babel/runtime": "^7.25.0", + "html-parse-stringify": "^3.0.1" + }, + "peerDependencies": { + "i18next": ">= 23.2.3", + "react": ">= 16.8.0" + }, + "peerDependenciesMeta": { + "react-dom": { + "optional": true + }, + "react-native": { + "optional": true + } + } + }, "node_modules/react-icons": { "version": "5.3.0", "resolved": "https://registry.npmjs.org/react-icons/-/react-icons-5.3.0.tgz", @@ -4346,10 +4467,9 @@ } }, "node_modules/regenerator-runtime": { - "version": "0.13.11", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz", - "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==", - "license": "MIT" + "version": "0.14.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz", + "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==" }, "node_modules/regexp.prototype.flags": { "version": "1.5.0", @@ -5311,6 +5431,15 @@ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", "license": "MIT" }, + "node_modules/void-elements": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-3.1.0.tgz", + "integrity": "sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w==", + "peer": true, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/warning": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz", diff --git a/package.json b/package.json index 14b6411..eae0e20 100644 --- a/package.json +++ b/package.json @@ -33,6 +33,7 @@ "google-map-react": "^2.2.1", "lodash": "^4.17.21", "next": "^13.4.3", + "next-i18next": "^15.3.1", "rc-slider": "^10.1.1", "react": "^18.2.0", "react-datepicker": "^4.11.0", diff --git a/public/favicon.ico b/public/favicon.ico index 718d6fe..dbcafd7 100644 Binary files a/public/favicon.ico and b/public/favicon.ico differ diff --git a/src/app/(account-pages)/(components)/Nav.tsx b/src/app/(account-pages)/(components)/Nav.tsx index 1633962..5ea1fff 100644 --- a/src/app/(account-pages)/(components)/Nav.tsx +++ b/src/app/(account-pages)/(components)/Nav.tsx @@ -11,7 +11,6 @@ export const Nav = () => { const listNav: Route[] = [ "/account", "/my-trips", - "/account-password", "/passengers-list", ]; diff --git a/src/app/(account-pages)/account-password/page.tsx b/src/app/(account-pages)/account-password/page.tsx deleted file mode 100644 index 5bc23bb..0000000 --- a/src/app/(account-pages)/account-password/page.tsx +++ /dev/null @@ -1,33 +0,0 @@ -import React from "react"; -import Label from "@/components/Label"; -import ButtonPrimary from "@/shared/ButtonPrimary"; -import Input from "@/shared/Input"; - -const AccountPass = () => { - return ( -
- {/* HEADING */} -

Update your password

-
-
-
- - -
-
- - -
-
- - -
-
- Update password -
-
-
- ); -}; - -export default AccountPass; diff --git a/src/app/(account-pages)/account/page.tsx b/src/app/(account-pages)/account/page.tsx index e65cfca..ab4e60e 100644 --- a/src/app/(account-pages)/account/page.tsx +++ b/src/app/(account-pages)/account/page.tsx @@ -94,47 +94,62 @@ const AccountPage: FC = () => { const changeHandler = async (): Promise => { setError(""); setLoading({ change: true }); - + const formData = new FormData(); - formData.append("fullname", name); - formData.append("email", email); + + if (name !== user.fullname) { + formData.append("fullname", name); + } + + if (email !== user.email) { + formData.append("email", email); + } + if (imageURL) { formData.append("avatar", imageURL); } - - try { - const response = await axiosInstance.put( - `/api/account/profile/update/`, - formData, - { - headers: { - Authorization: `token ${user.token}`, - "Content-Type": "multipart/form-data", - }, + + if (formData.has("fullname") || formData.has("email") || formData.has("avatar")) { + try { + const response = await axiosInstance.put( + `/api/account/profile/update/`, + formData, + { + headers: { + Authorization: `token ${user.token}`, + "Content-Type": "multipart/form-data", + }, + } + ); + if (response.status === 200) { + console.log(response); + + toast.success("Updated successfully"); + setUser({ + ...user, + avatar: response.data.avatar, + email: response.data.email, + fullname: response.data.fullname, + phone_number: response.data.phone_number, + }); + } else { + setError("Something went wrong"); } - ); - if (response.status === 200) { - toast.success("Updated successfully") - setUser({ - ...user, - avatar: response.data.avatar, - email: response.data.email, - fullname: response.data.fullname, - phone_number: response.data.phone_number, - }); - } else { - setError("Something went wrong"); - } - } catch (error: unknown) { - if (error instanceof Error) { - setError(error.message); - } else { - setError("An unknown error occurred"); + } catch (error: unknown) { + if (error instanceof Error) { + setError(error.message); + } else { + setError("An unknown error occurred"); + } + } finally { + setLoading({ change: false }); } - } finally { + } else { + toast.info("No changes detected"); setLoading({ change: false }); } }; + const handleFileChange = async (e: ChangeEvent): Promise => { const file : File | undefined = e.target.files?.[0]; diff --git a/src/app/(account-pages)/my-trips/page.tsx b/src/app/(account-pages)/my-trips/page.tsx index e07b98d..f087405 100644 --- a/src/app/(account-pages)/my-trips/page.tsx +++ b/src/app/(account-pages)/my-trips/page.tsx @@ -9,7 +9,7 @@ import { DEMO_EXPERIENCES_LISTINGS, DEMO_STAY_LISTINGS, } from "@/data/listings"; -import React, { Fragment, useEffect, useState } from "react"; +import React, { Fragment, use, useEffect, useState } from "react"; import ButtonSecondary from "@/shared/ButtonSecondary"; import axiosInstance from "@/components/api/axios"; import StayCard2 from "@/components/StayCard2"; diff --git a/src/app/(account-pages)/passengers-list/PassengerTable.tsx b/src/app/(account-pages)/passengers-list/PassengerTable.tsx index 3401402..0a3052f 100644 --- a/src/app/(account-pages)/passengers-list/PassengerTable.tsx +++ b/src/app/(account-pages)/passengers-list/PassengerTable.tsx @@ -1,15 +1,17 @@ "use client"; import axiosInstance from "@/components/api/axios"; import { useUserContext } from "@/components/contexts/userContext"; +import { useRouter } from "next/navigation"; import React, { useEffect, useState } from "react"; import { IoMdTrash } from "react-icons/io"; import { MdEdit } from "react-icons/md"; const PassengerTable = ({ data }) => { const { user } = useUserContext(); - + const router = useRouter() const [show, setShow] = useState(true); + const deletHandler = async () => { try { const response = await axiosInstance.delete( @@ -43,13 +45,11 @@ const PassengerTable = ({ data }) => { {/* Action Icons */}
- + -
diff --git a/src/app/(account-pages)/passengers-list/[id]/page.tsx b/src/app/(account-pages)/passengers-list/[id]/page.tsx new file mode 100644 index 0000000..7862b5e --- /dev/null +++ b/src/app/(account-pages)/passengers-list/[id]/page.tsx @@ -0,0 +1,270 @@ +"use client"; + +import React, { useEffect, useState } from "react"; +import { FC } from "react"; +import ButtonPrimary from "@/shared/ButtonPrimary"; +import Input from "@/shared/Input"; +import FormItem from "@/app/add-listing/FormItem"; +import getImageURL from "@/components/api/getImageURL"; +import axiosInstance from "@/components/api/axios"; +import { useRouter } from "next/navigation"; +import { useUserContext } from "@/components/contexts/userContext"; +import { toast } from "react-toastify"; // Import toast + +export interface CommonLayoutProps { + params: { + id: string; + }; +} + +const EditPassenger: FC = ({ params }) => { + const { user } = useUserContext(); + const router = useRouter(); + + const [passenger, setPassenger] = useState({ + name: "", + passport: "", + number: "", + date: "", + image: "", + }); + + const [originalPassenger, setOriginalPassenger] = useState({ + name: "", + passport: "", + number: "", + date: "", + image: "", + }); + + const [loading, setLoading] = useState(false); + + useEffect(() => { + if (Object.keys(user).length) { + axiosInstance + .get(`/api/account/passengers/${params.id}/`, { + headers: { + Authorization: `token ${user.token}`, + "Content-Type": "application/json", + "X-CSRFToken": "oli8in4JPf6taFVzg7tsY2g9Xpmox45yOBp1LrgU20tUFL5K4VkOrx7quXvOLUwW", + }, + }) + .then((response) => { + const passengerData = { + name: response.data.fullname, + passport: response.data.passport_number, + date: response.data.birthdate, + number: response.data.phone_number.replace(/\D/g, ""), + image: response.data.passport_image, + }; + setPassenger(passengerData); + setOriginalPassenger(passengerData); // Save original data for comparison + }) + .catch((error) => { + toast.error(error.message); + }); + } + }, [user]); + + const handleFileChange = async (e: React.ChangeEvent) => { + setLoading(true); + const file = e.target.files?.[0]; + if (file) { + try { + const image = await getImageURL(file); + setPassenger((prev) => ({ ...prev, image: image.url })); + toast.success("Image uploaded successfully!"); + } catch (error) { + toast.error("Error uploading image."); + } finally { + setLoading(false); + } + } + }; + + const validateForm = () => { + let formIsValid = true; + + if (!passenger.name) { + formIsValid = false; + toast.error("Full Name is required."); + } + + if (!passenger.passport) { + formIsValid = false; + toast.error("Passport Number is required."); + } else if (!/^\d+$/.test(passenger.passport)) { + formIsValid = false; + toast.error("Passport Number must be numeric."); + } + + if (!passenger.date) { + formIsValid = false; + toast.error("Date of Birth is required."); + } + + if (!passenger.number) { + formIsValid = false; + toast.error("Phone Number is required."); + } else if (!/^\d+$/.test(passenger.number)) { + formIsValid = false; + toast.error("Phone Number must be numeric."); + } + + if (!passenger.image) { + formIsValid = false; + toast.error("Passport image is required."); + } + + return formIsValid; + }; + + const handleSavePassenger = async () => { + if (!validateForm()) return; + + const updatedFields: Partial = {}; + + // Only add fields that were changed + if (passenger.name !== originalPassenger.name) { + updatedFields.fullname = passenger.name; + } + if (passenger.passport !== originalPassenger.passport) { + updatedFields.passport_number = passenger.passport; + } + if (passenger.date !== originalPassenger.date) { + updatedFields.birthdate = passenger.date; + } + if (passenger.number !== originalPassenger.number) { + updatedFields.phone_number = passenger.number; + } + if (passenger.image !== originalPassenger.image) { + updatedFields.passport_image = passenger.image; + } + + if (Object.keys(updatedFields).length === 0) { + toast.info("No changes to update."); + return; + } + + try { + const response = await axiosInstance.patch( + `https://aqila.nwhco.ir/api/account/passengers/${params.id}/`, + updatedFields, + { + headers: { + Authorization: `token ${user.token}`, + "Content-Type": "application/json", + "X-CSRFToken": "oli8in4JPf6taFVzg7tsY2g9Xpmox45yOBp1LrgU20tUFL5K4VkOrx7quXvOLUwW", + }, + } + ); + + if (response.status === 200) { + toast.success("Passenger details updated successfully!"); + router.push("/passengers-list"); + } + } catch (error) { + toast.error("Error saving passenger details."); + } + }; + + return ( +
+
+
+
+ <> +

Passenger Information

+
+
+ + + setPassenger((prev) => ({ + ...prev, + name: e.target.value, + })) + } + placeholder="Full Name" + /> + + + + + setPassenger((prev) => ({ + ...prev, + passport: e.target.value, + })) + } + type="text" // Changed from 'number' to 'text' + placeholder="Passport Number" + /> + + + + + setPassenger((prev) => ({ + ...prev, + date: e.target.value, + })) + } + type="date" + placeholder="Date of Birth" + /> + + + + + setPassenger((prev) => ({ + ...prev, + number: e.target.value.replace(/\D/g, ""), // Ensure only numeric input + })) + } + type="text" // Keep as 'text' to prevent unwanted behavior + placeholder="Phone Number" + /> + + + + + {loading &&

Loading ...

} +
+
+ +
+ +
+ { + e.preventDefault(); + handleSavePassenger(); + }} + > + Continue + +
+
+
+
+ ); +}; + +export default EditPassenger; diff --git a/src/app/(account-pages)/passengers-list/page.tsx b/src/app/(account-pages)/passengers-list/page.tsx index 60762fc..04e54e3 100644 --- a/src/app/(account-pages)/passengers-list/page.tsx +++ b/src/app/(account-pages)/passengers-list/page.tsx @@ -6,10 +6,18 @@ import { IoPersonAddOutline } from "react-icons/io5"; import axiosInstance from "@/components/api/axios"; import Link from "next/link"; import { useUserContext } from "@/components/contexts/userContext"; +import { useRouter } from "next/navigation"; const PassengersList = () => { const [passengers , setPassenger ] = useState([]) const {user} = useUserContext() +const router = useRouter() + +useEffect(() => { + if (!Object.keys(user).length) { + router.replace("/"); + } +}, [user, router]); useEffect(()=>{ axiosInstance.get("/api/account/passengers/" ,{ diff --git a/src/app/(client-components)/(Header)/MainNav1.tsx b/src/app/(client-components)/(Header)/MainNav1.tsx index 77c7eaa..cee1d5c 100644 --- a/src/app/(client-components)/(Header)/MainNav1.tsx +++ b/src/app/(client-components)/(Header)/MainNav1.tsx @@ -20,6 +20,7 @@ export interface MainNav1Props { const MainNav1: FC = ({ className = "" }) => { const {user} = useUserContext() +console.log(Object.keys(user).length); return (
diff --git a/src/app/(client-components)/(HeroSearchForm)/GuestsInput.tsx b/src/app/(client-components)/(HeroSearchForm)/GuestsInput.tsx index 004800f..3cb55a9 100644 --- a/src/app/(client-components)/(HeroSearchForm)/GuestsInput.tsx +++ b/src/app/(client-components)/(HeroSearchForm)/GuestsInput.tsx @@ -80,7 +80,7 @@ const GuestsInput: FC = ({ {/* BUTTON SUBMIT OF FORM */} {hasButtonSubmit && (
- +
)}
diff --git a/src/app/(listing-detail)/(components)/MobileFooterSticky.tsx b/src/app/(listing-detail)/(components)/MobileFooterSticky.tsx index a6ad8fc..410dc5a 100644 --- a/src/app/(listing-detail)/(components)/MobileFooterSticky.tsx +++ b/src/app/(listing-detail)/(components)/MobileFooterSticky.tsx @@ -62,13 +62,9 @@ const MobileFooterSticky = ({ data }) => { className={`${ data?.status === "AVAILABLE" && passengers ? "" - : "opacity-60 cursor-not-allowed" - }`} - href={`${ - data?.status === "AVAILABLE" && passengers - ? `/add-listing/${id}` - : `/tours/${data?.slug}-${id}` + : "opacity-60 pointer-events-none" }`} + href={`/add-listing/${id}`} > Reserve diff --git a/src/app/add-listing/[[...stepIndex]]/page.tsx b/src/app/add-listing/[[...stepIndex]]/page.tsx index bf1ce2d..c1b59f5 100644 --- a/src/app/add-listing/[[...stepIndex]]/page.tsx +++ b/src/app/add-listing/[[...stepIndex]]/page.tsx @@ -1,5 +1,4 @@ -"use client"; - +"use client" import React, { useContext, useState, useEffect } from "react"; import { FC } from "react"; import ButtonPrimary from "@/shared/ButtonPrimary"; @@ -34,7 +33,7 @@ const CommonLayout: FC = ({ params }) => { }); const [passengerID, setPassengerID] = useState([]); const [selectedPassenger, setSelectedPassenger] = useState(null); - const [redirecting, setRedirecting] = useState(false); // New state for redirection + const [redirecting, setRedirecting] = useState(false); const tourID = params.stepIndex[0]; const totalPassengers = useContext(Context).passengers; @@ -43,32 +42,30 @@ const CommonLayout: FC = ({ params }) => { const backtHref = () => (index > 1 ? setIndex((prev) => prev - 1) : index); const nextBtnText = index === totalPassengers ? "Save Passengers" : "Continue"; - useEffect(()=>{ + useEffect(() => { Object.values(errors).forEach((error) => { toast.error(error, { position: "top-right", autoClose: 5000, - hideProgressBar: false, - closeOnClick: true, - pauseOnHover: true, - draggable: true, - progress: undefined, - theme: "light", }); }); - } , [errors]) - + }, [errors]); + const sendPassengers = async () => { + const dataToSend = { + tour_id: tourID, + passengers: { + passenger_ids: passengerID, + new_passengers: passengers, + }, + }; + + console.log("Data being sent:", dataToSend); + try { const response = await axiosInstance.post( "/api/tours/orders/purchase/", - { - tour_id: tourID, - passengers: { - passenger_ids: passengerID, - new_passengers: passengers, - }, - }, + dataToSend, { headers: { Authorization: `token ${user.token}`, @@ -77,7 +74,7 @@ const CommonLayout: FC = ({ params }) => { } ); console.log(response); - setRedirecting(true); + setRedirecting(true); } catch (error) { backtHref(); console.error("Error submitting passengers:", error); @@ -87,7 +84,7 @@ const CommonLayout: FC = ({ params }) => { useEffect(() => { if (redirecting) { router.replace("/my-trips"); - toast.success("Purchased Successfully ") + toast.success("Purchased Successfully "); } }, [redirecting, router]); @@ -100,15 +97,20 @@ const CommonLayout: FC = ({ params }) => { const nextHandler = () => { const validationErrors = validatePassenger(newPassenger); - if (Object.keys(validationErrors).length > 0 && passengerID.length < 0) { + if (Object.keys(validationErrors).length > 0 && passengerID.length === 0) { setErrors(validationErrors); console.log("Validation errors:", validationErrors, passengerID); - console.log(passengerID.length > 0); return; } - setSelectedPassenger(null); + if (selectedPassenger) { + // If an existing passenger is selected + setSelectedPassenger(null); + nextHref(); + return; + } + // Add new passenger setPassengers((prevPassengers) => [...prevPassengers, newPassenger]); setNewPassenger({ fullname: "", @@ -122,7 +124,6 @@ const CommonLayout: FC = ({ params }) => { nextHref(); }; - // Prevent rendering while redirecting if (redirecting) { return null; } @@ -151,9 +152,6 @@ const CommonLayout: FC = ({ params }) => {
- {/* {index > 1 && ( - Go back - )} */} {nextBtnText}
diff --git a/src/app/layout.tsx b/src/app/layout.tsx index d3d6e73..819b32f 100644 --- a/src/app/layout.tsx +++ b/src/app/layout.tsx @@ -12,6 +12,9 @@ import Footer from "@/components/Footer"; import FooterNav from "@/components/FooterNav"; import { UserProvider } from "@/components/contexts/userContext"; import { ToastContainer } from "react-toastify"; +import { appWithTranslation } from "next-i18next"; +import "react-toastify/dist/ReactToastify.css"; + const poppins = Poppins({ subsets: ["latin"], @@ -19,7 +22,7 @@ const poppins = Poppins({ weight: ["300", "400", "500", "600", "700"], }); -export default function RootLayout({ +function RootLayout({ children, params, }: { @@ -54,3 +57,4 @@ export default function RootLayout({ ); } +export default appWithTranslation(RootLayout) \ No newline at end of file diff --git a/src/app/page.tsx b/src/app/page.tsx index 12de2cc..52b9a3c 100644 --- a/src/app/page.tsx +++ b/src/app/page.tsx @@ -17,138 +17,9 @@ import axios from "axios"; import axiosInstance from "@/components/api/axios"; import TourSuggestion from "@/components/TourSuggestion"; import SectionDowloadApp from "./(home)/SectionDowloadApp"; +import "react-toastify/dist/ReactToastify.css"; + -const DEMO_CATS: TaxonomyType[] = [ - { - id: "1", - href: "/listing-stay-map", - name: "Mashhad", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "2", - href: "/listing-stay-map", - name: "Qum", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "3", - href: "/listing-stay-map", - name: "Shiraz", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "4", - href: "/listing-stay-map", - name: "Shar-e ray", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "5", - href: "/listing-stay-map", - name: "Karbala", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "6", - href: "/listing-stay-map", - name: "Najaf", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, - { - id: "7", - href: "/listing-stay-map", - name: "Kofeh", - taxonomy: "category", - count: 188288, - thumbnail: - "", - }, -]; - -const DEMO_CATS_2: TaxonomyType[] = [ - { - id: "1", - href: "/listing-stay-map", - name: "Enjoy the great cold", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/5764100/pexels-photo-5764100.jpeg?auto=compress&cs=tinysrgb&dpr=3&h=750&w=1260", - }, - { - id: "2", - href: "/listing-stay-map", - name: "Sleep in a floating way", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/2869499/pexels-photo-2869499.jpeg?auto=compress&cs=tinysrgb&dpr=2&h=750&w=1260", - }, - { - id: "3", - href: "/listing-stay-map", - name: "In the billionaire's house", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/7031413/pexels-photo-7031413.jpeg?auto=compress&cs=tinysrgb&dpr=2&h=750&w=1260", - }, - { - id: "4", - href: "/listing-stay-map", - name: "Cool in the deep forest", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/247532/pexels-photo-247532.jpeg?auto=compress&cs=tinysrgb&dpr=2&h=750&w=1260", - }, - { - id: "5", - href: "/listing-stay-map", - name: "In the billionaire's house", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/7031413/pexels-photo-7031413.jpeg?auto=compress&cs=tinysrgb&dpr=2&h=750&w=1260", - }, - { - id: "6", - href: "/listing-stay-map", - name: "In the billionaire's house", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/9828170/pexels-photo-9828170.jpeg?auto=compress&cs=tinysrgb&w=1600&lazy=load", - }, - { - id: "7", - href: "/listing-stay-map", - name: "Cool in the deep forest", - taxonomy: "category", - count: 188288, - thumbnail: - "https://images.pexels.com/photos/247532/pexels-photo-247532.jpeg?auto=compress&cs=tinysrgb&dpr=2&h=750&w=1260", - }, -]; function PageHome() { @@ -178,7 +49,6 @@ function PageHome() {
= ({ className = "", data = DEMO_STAY_LISTINGS, }) => { - const { countries, tours } = useContext(Context); + const { countries, tours } = useToursContext() const [countryTours, setCountryTours] = useState(tours.results || []); const [checked, setChecked] = useState<{ [key: string]: boolean }>({}); const searchParams = useSearchParams(); diff --git a/src/app/tours/[slug]/page.tsx b/src/app/tours/[slug]/page.tsx index d1aa8ac..ba30344 100644 --- a/src/app/tours/[slug]/page.tsx +++ b/src/app/tours/[slug]/page.tsx @@ -264,7 +264,7 @@ const ListingStayDetailPage: FC = ({}) => {
{iteneries?.map((item, index) => ( -
+
{totalIteneries !== index + 1 && ( @@ -470,7 +470,7 @@ const ListingStayDetailPage: FC = ({}) => {
    {details && details.travel_tips.map((item) => ( -
  • +
  • {item.title}

    @@ -511,7 +511,12 @@ const ListingStayDetailPage: FC = ({}) => { {details?.price} x {passengers} passengers - {details?.price * passengers} + + {" "} + {isNaN(details?.price * passengers) + ? "N/A" // Or any fallback value, like "0" or a string message + : (details?.price * passengers).toString()} +
Service charge @@ -520,17 +525,22 @@ const ListingStayDetailPage: FC = ({}) => {
Total - {details?.price * passengers} + + {isNaN(details?.price * passengers) + ? "N/A" // Or any fallback value, like "0" or a string message + : (details?.price * passengers).toString()} +
{/* SUBMIT */} Reserve @@ -544,13 +554,13 @@ const ListingStayDetailPage: FC = ({}) => {
- + />} {/*
= ({}) => {
{renderSidebar()}
- - +
); }; diff --git a/src/app/tours/page.tsx b/src/app/tours/page.tsx index 6ec71f0..2925942 100644 --- a/src/app/tours/page.tsx +++ b/src/app/tours/page.tsx @@ -5,7 +5,7 @@ export interface ListingStayMapPageProps {} const ListingStayMapPage: FC = ({}) => { return ( -
+
); diff --git a/src/components/CardCategory3.tsx b/src/components/CardCategory3.tsx index f320895..da774d1 100644 --- a/src/components/CardCategory3.tsx +++ b/src/components/CardCategory3.tsx @@ -1,56 +1,77 @@ -"use client" +"use client"; import React, { FC, useEffect, useState } from "react"; -import { TaxonomyType } from "@/data/types"; import convertNumbThousand from "@/utils/convertNumbThousand"; import Link from "next/link"; import Image from "next/image"; +// Define the TaxonomyType, CountryType, and TourType interfaces -export interface CardCategory3Props { +interface TaxonomyType { + count: number; + name: string; + href?: string; + thumbnail?: string; + city: { thumbnail: string }[]; // Assuming 'city' is an array of objects with 'thumbnail' +} + +interface CountryType { + name: string; + code: string; +} + +interface TourType { + destination_country: string; +} + +interface CardCategory3Props { className?: string; taxonomy: TaxonomyType; - countries : any - tours : any + countries: CountryType[]; + tours: { results: TourType[] }; // Assuming 'tours' is an object with 'results' as an array of tours } const CardCategory3: FC = ({ className = "", taxonomy, countries, - tours + tours, }) => { const { count, name, href = "/", thumbnail } = taxonomy; - const [countryTours , setCountryTours] = useState([]) - useEffect(()=>{ + // Set the state with proper typing for country tours + const [countryTours, setCountryTours] = useState([]); + // Update the useEffect hook to watch for changes in dependencies + useEffect(() => { const selected = countries?.find((country) => country.name === taxonomy.name); if (selected) { - - const selectedTours = tours?.results.filter( - (tour) => tour.destination_country === selected?.code + const selectedTours = tours?.results?.filter( + (tour) => tour.destination_country === selected.code ); - setCountryTours(selectedTours) + setCountryTours(selectedTours || []); // Set default as empty array if no tours found } + }, [countries, taxonomy.name, tours]); // Added dependencies for useEffect - } , []) - - - return ( - +
- places + {taxonomy.city[0]?.thumbnail ? ( + places + ) : ( +
+ )}
@@ -62,7 +83,7 @@ const CardCategory3: FC = ({ - {convertNumbThousand(countryTours.length || 0)} Tours + {convertNumbThousand(countryTours.length || 0)} Tours
diff --git a/src/components/TourSuggestion.tsx b/src/components/TourSuggestion.tsx index 0260858..1f4875e 100644 --- a/src/components/TourSuggestion.tsx +++ b/src/components/TourSuggestion.tsx @@ -13,7 +13,7 @@ import NextBtn from "./NextBtn"; import { variants } from "@/utils/animationVariants"; import { useWindowSize } from "react-use"; import axiosInstance from "./api/axios"; -import { Context } from "./contexts/tourDetails"; +import { Context, useToursContext } from "./contexts/tourDetails"; import CardCategory1 from "./CardCategory1"; export interface TourSuggestionProps { @@ -21,7 +21,6 @@ export interface TourSuggestionProps { itemClassName?: string; heading?: string; subHeading?: string; - categories?: TaxonomyType[]; categoryCardType?: "card3" | "card4" | "card5"; itemPerRow?: 4 | 5; sliderStyle?: "style1" | "style2"; @@ -107,7 +106,6 @@ const TourSuggestion: FC = ({ subHeading = "Popular places to recommends for you", className = "", itemClassName = "", - categories = DEMO_CATS, itemPerRow = 5, categoryCardType = "card3", sliderStyle = "style1", @@ -115,7 +113,7 @@ const TourSuggestion: FC = ({ const [currentIndex, setCurrentIndex] = useState(0); const [direction, setDirection] = useState(0); const [numberOfItems, setNumberOfitem] = useState(0); - const { tours , countries } = useContext(Context); + const { tours , countries } = useToursContext() diff --git a/src/components/contexts/tourDetails.tsx b/src/components/contexts/tourDetails.tsx index 4b207d3..cf3e25e 100644 --- a/src/components/contexts/tourDetails.tsx +++ b/src/components/contexts/tourDetails.tsx @@ -1,14 +1,50 @@ "use client"; import axiosInstance from "../api/axios"; -import React, { createContext, useContext, useEffect, useState } from "react"; +import React, { createContext, useContext, useEffect, useState, ReactNode } from "react"; -export const Context = createContext(); +interface Country { + id: number; + name: string; +} -export const ContextProvider = ({ children }) => { - const [details, setDetails] = useState(); - const [passengers, setPassengers] = useState(0); - const [tours, setTours] = useState([]); - const [countries, setCountries] = useState([]); +interface Tour { + id: number; + name: string; + description: string; + price: number; + duration: string; +} + +interface TourDetails { + id: number; + name: string; + description: string; + startDate: string; + endDate: string; + passengers: number; +} + +export const Context = createContext(undefined); + +interface ToursContextType { + details: TourDetails | undefined; + passengers: number; + getTourData: (item: number) => Promise; + setPassengers: React.Dispatch>; + setDetails: React.Dispatch>; + tours: Tour[]; + countries: Country[]; +} + +interface ContextProviderProps { + children: ReactNode; +} + +export const ContextProvider = ({ children }: ContextProviderProps) => { + const [details, setDetails] = useState(undefined); + const [passengers, setPassengers] = useState(0); + const [tours, setTours] = useState([]); + const [countries, setCountries] = useState([]); useEffect(() => { axiosInstance @@ -32,26 +68,15 @@ export const ContextProvider = ({ children }) => { }); }, []); - - const getTourData = async (item) => { - await axiosInstance - .get(`/api/tours/${item}/`) - .then((response) => { - setDetails(response.data); - }) - .catch((error) => { - console.error("Error fetching data:", error); - }); - // await axiosInstance - // .get(`/api/tours/${item}/passengers/`) - // .then((response) => { - // setPassengers(response.data); - // }) - // .catch((error) => { - // console.error("Error fetching data:", error); - // }); + const getTourData = async (item: number) => { + try { + const response = await axiosInstance.get(`/api/tours/${item}/`); + setDetails(response.data); + } catch (error) { + console.error("Error fetching tour data:", error); + } }; - + return ( { ); }; -export const useToursContext = () => { +// Custom hook +export const useToursContext = (): ToursContextType => { const context = useContext(Context); if (!context) { - throw new Error("useUserContext must be used within a UserProvider"); + throw new Error("useToursContext must be used within a ContextProvider"); } return context; }; diff --git a/src/hooks/i18n.js.ts b/src/hooks/i18n.js.ts new file mode 100644 index 0000000..a493b2e --- /dev/null +++ b/src/hooks/i18n.js.ts @@ -0,0 +1,8 @@ +const NextI18Next = require('next-i18next').default; + +const i18n = new NextI18Next({ + defaultLanguage: 'en', + otherLanguages: ['en'], +}); + +module.exports = i18n; diff --git a/src/shared/Logo.tsx b/src/shared/Logo.tsx index b6eff33..01f42c9 100644 --- a/src/shared/Logo.tsx +++ b/src/shared/Logo.tsx @@ -36,7 +36,11 @@ const Logo: React.FC = ({ src={img} alt="Logo" /> -

AQILA

+ +
+

Aqila

+

Traveling Agency

+
diff --git a/src/shared/Navigation/NavigationItem.tsx b/src/shared/Navigation/NavigationItem.tsx index 1be76ff..56949e5 100644 --- a/src/shared/Navigation/NavigationItem.tsx +++ b/src/shared/Navigation/NavigationItem.tsx @@ -240,7 +240,7 @@ const NavigationItem: FC = ({ menuItem }) => { target={item.targetBlank ? "_blank" : undefined} rel="noopener noreferrer" className="flex items-center font-normal text-neutral-6000 dark:text-neutral-300 py-2 px-4 rounded-md hover:text-neutral-700 hover:bg-neutral-100 dark:hover:bg-neutral-800 dark:hover:text-neutral-200 " - href={item.href || ""} + href={`/tours?country=${item.name}`} > {item.name} {item.type && ( diff --git a/yarn.lock b/yarn.lock index c0d5092..989a226 100644 --- a/yarn.lock +++ b/yarn.lock @@ -7,12 +7,12 @@ resolved "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz" integrity sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw== -"@babel/runtime@^7.1.2", "@babel/runtime@^7.10.1", "@babel/runtime@^7.18.3", "@babel/runtime@^7.20.7", "@babel/runtime@^7.21.0": - version "7.21.5" - resolved "https://registry.npmjs.org/@babel/runtime/-/runtime-7.21.5.tgz" - integrity sha512-8jI69toZqqcsnqGGqwGS4Qb1VwLOEp4hz+CXPywcvjs60u3B4Pom/U/7rm4W8tMOYEB+E9wgD0mW1l3r8qlI9Q== +"@babel/runtime@^7.1.2", "@babel/runtime@^7.10.1", "@babel/runtime@^7.18.3", "@babel/runtime@^7.20.7", "@babel/runtime@^7.21.0", "@babel/runtime@^7.23.2", "@babel/runtime@^7.25.0": + version "7.25.6" + resolved "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.6.tgz" + integrity sha512-VBj9MYyDb9tuLq7yzqjgzt6Q+IBQLrGZfdjOekyEirZPHxXWoTSGUTMrpsfi58Up73d13NfYLv8HT9vmznjzhQ== dependencies: - regenerator-runtime "^0.13.11" + regenerator-runtime "^0.14.0" "@emotion/is-prop-valid@^0.8.2": version "0.8.8" @@ -234,6 +234,14 @@ dependencies: "@types/react" "*" +"@types/hoist-non-react-statics@^3.3.4": + version "3.3.5" + resolved "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.5.tgz" + integrity sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg== + dependencies: + "@types/react" "*" + hoist-non-react-statics "^3.3.0" + "@types/js-cookie@^2.2.6": version "2.2.7" resolved "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-2.2.7.tgz" @@ -657,6 +665,11 @@ copy-to-clipboard@^3.3.1: dependencies: toggle-selection "^1.0.6" +core-js@^3: + version "3.38.1" + resolved "https://registry.npmjs.org/core-js/-/core-js-3.38.1.tgz" + integrity sha512-OP35aUorbU3Zvlx7pjsFdu1rGNnD4pgw/CWoYzRY3t2EzoVT7shKHY1dlAy3f41cGIO7ZDPQimhGFTlEYkG/Hw== + cross-spawn@^7.0.2, cross-spawn@^7.0.3: version "7.0.3" resolved "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz" @@ -1494,6 +1507,20 @@ has@^1.0.3: dependencies: function-bind "^1.1.1" +hoist-non-react-statics@^3.3.0, hoist-non-react-statics@^3.3.2: + version "3.3.2" + resolved "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz" + integrity sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw== + dependencies: + react-is "^16.7.0" + +html-parse-stringify@^3.0.1: + version "3.0.1" + resolved "https://registry.npmjs.org/html-parse-stringify/-/html-parse-stringify-3.0.1.tgz" + integrity sha512-KknJ50kTInJ7qIScF3jeaFRpMpE8/lfiTdzf/twXyPBLAGrLRTmkz3AdTnKeh40X8k9L2fdYwEp/42WGXIRGcg== + dependencies: + void-elements "3.1.0" + human-signals@^2.1.0: version "2.1.0" resolved "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz" @@ -1509,6 +1536,18 @@ hyphenate-style-name@^1.0.3: resolved "https://registry.npmjs.org/hyphenate-style-name/-/hyphenate-style-name-1.0.4.tgz" integrity sha512-ygGZLjmXfPHj+ZWh6LwbC37l43MhfztxetbFCoYTM2VjkIUpeHgSNn7QIyVFj7YQ1Wl9Cbw5sholVJPzWvC2MQ== +i18next-fs-backend@^2.3.2: + version "2.3.2" + resolved "https://registry.npmjs.org/i18next-fs-backend/-/i18next-fs-backend-2.3.2.tgz" + integrity sha512-LIwUlkqDZnUI8lnUxBnEj8K/FrHQTT/Sc+1rvDm9E8YvvY5YxzoEAASNx+W5M9DfD5s77lI5vSAFWeTp26B/3Q== + +"i18next@>= 23.2.3", "i18next@>= 23.7.13": + version "23.15.1" + resolved "https://registry.npmjs.org/i18next/-/i18next-23.15.1.tgz" + integrity sha512-wB4abZ3uK7EWodYisHl/asf8UYEhrI/vj/8aoSsrj/ZDxj4/UXPOa1KvFt1Fq5hkUHquNqwFlDprmjZ8iySgYA== + dependencies: + "@babel/runtime" "^7.23.2" + ignore@^5.2.0: version "5.2.4" resolved "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz" @@ -1991,7 +2030,18 @@ natural-compare@^1.4.0: resolved "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== -next@^13.4.3: +next-i18next@^15.3.1: + version "15.3.1" + resolved "https://registry.npmjs.org/next-i18next/-/next-i18next-15.3.1.tgz" + integrity sha512-+pa2pZJb7B6k5PKW3TLVMmAodqkNaOBWVYlpWX56mgcEJz0UMW+MKSdKM9Z72CHp6Bp48g7OWwDnLqxXNp/84w== + dependencies: + "@babel/runtime" "^7.23.2" + "@types/hoist-non-react-statics" "^3.3.4" + core-js "^3" + hoist-non-react-statics "^3.3.2" + i18next-fs-backend "^2.3.2" + +next@^13.4.3, "next@>= 12.0.0": version "13.4.3" resolved "https://registry.npmjs.org/next/-/next-13.4.3.tgz" integrity sha512-FV3pBrAAnAIfOclTvncw9dDohyeuEEXPe5KNcva91anT/rdycWbgtu3IjUj4n5yHnWK8YEPo0vrUecHmnmUNbA== @@ -2378,12 +2428,20 @@ react-hooks-global-state@^2.1.0: dependencies: zustand "4.0.0" +"react-i18next@>= 13.5.0": + version "15.0.2" + resolved "https://registry.npmjs.org/react-i18next/-/react-i18next-15.0.2.tgz" + integrity sha512-z0W3/RES9Idv3MmJUcf0mDNeeMOUXe+xoL0kPfQPbDoZHmni/XsIoq5zgT2MCFUiau283GuBUK578uD/mkAbLQ== + dependencies: + "@babel/runtime" "^7.25.0" + html-parse-stringify "^3.0.1" + react-icons@^5.3.0: version "5.3.0" resolved "https://registry.npmjs.org/react-icons/-/react-icons-5.3.0.tgz" integrity sha512-DnUk8aFbTyQPSkCfF8dbX6kQjXA9DktMeJqfjrg6cK9vwQVMxmcA3BfP4QoiztVmEHtwlTgLFsPuH2NskKT6eg== -react-is@^16.12.0, react-is@^16.13.1: +react-is@^16.12.0, react-is@^16.13.1, react-is@^16.7.0: version "16.13.1" resolved "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz" integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ== @@ -2445,7 +2503,7 @@ react-use@^17.4.0: ts-easing "^0.2.0" tslib "^2.1.0" -react@*, "react@^15.5.x || ^16.x || ^17.x || ^18.x", "react@^16 || ^17 || ^18", "react@^16.0.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || ^17 || ^18", "react@^16.8.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || 17 || 18", "react@^16.8.3 || ^17 || ^18", "react@^16.9.0 || ^17 || ^18", react@^18.0.0, react@^18.2.0, "react@>= 16", "react@>= 16.8.0 || 17.x.x || ^18.0.0-0", react@>=16.8, react@>=16.8.0, react@>=16.9.0, react@>=18: +react@*, "react@^15.5.x || ^16.x || ^17.x || ^18.x", "react@^16 || ^17 || ^18", "react@^16.0.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || ^17 || ^18", "react@^16.8.0 || ^17.0.0 || ^18.0.0", "react@^16.8.0 || 17 || 18", "react@^16.8.3 || ^17 || ^18", "react@^16.9.0 || ^17 || ^18", react@^18.0.0, react@^18.2.0, "react@>= 16", "react@>= 16.8.0", "react@>= 16.8.0 || 17.x.x || ^18.0.0-0", "react@>= 17.0.2", react@>=16.8, react@>=16.8.0, react@>=16.9.0, react@>=18: version "18.2.0" resolved "https://registry.npmjs.org/react/-/react-18.2.0.tgz" integrity sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ== @@ -2466,10 +2524,10 @@ readdirp@~3.6.0: dependencies: picomatch "^2.2.1" -regenerator-runtime@^0.13.11: - version "0.13.11" - resolved "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz" - integrity sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg== +regenerator-runtime@^0.14.0: + version "0.14.1" + resolved "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz" + integrity sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw== regexp.prototype.flags@^1.4.3, regexp.prototype.flags@^1.5.0: version "1.5.0" @@ -2986,6 +3044,11 @@ util-deprecate@^1.0.2: resolved "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz" integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== +void-elements@3.1.0: + version "3.1.0" + resolved "https://registry.npmjs.org/void-elements/-/void-elements-3.1.0.tgz" + integrity sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w== + warning@^4.0.2: version "4.0.3" resolved "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz"