diff --git a/app/(pages)/_components/ContactForm/ContactForm.jsx b/app/(pages)/_components/ContactForm/ContactForm.jsx new file mode 100644 index 0000000..f8fdf2e --- /dev/null +++ b/app/(pages)/_components/ContactForm/ContactForm.jsx @@ -0,0 +1,182 @@ +"use client"; + +import { useState } from "react"; +import styles from "./ContactForm.module.scss"; + +export default function ContactForm() { + const [firstName, setFirstName] = useState(""); + const [lastName, setLastName] = useState(""); + const [email, setEmail] = useState(""); + const [subject, setSubject] = useState(""); + const [message, setMessage] = useState(""); + + // Error handling + const [firstNameError, setFirstNameError] = useState(""); + const [lastNameError, setLastNameError] = useState(""); + const [emailError, setEmailError] = useState(""); + const [subjectError, setSubjectError] = useState(""); + const [messageError, setMessageError] = useState(""); + + // Prevent multiple submissions while loading + const [loading, setLoading] = useState(false); + + const handleSubmit = async (e) => { + console.log("submit"); + e.preventDefault(); // Prevent automatic page reload + + // Prevent multiple submissions while loading + if (loading) return; + + setLoading(true); + + // Input Validation + let valid = true; + + if (!firstName.trim()) { + setFirstNameError("Please provide a first name"); + valid = false; + } else { + setFirstNameError(""); + } + + if (!lastName.trim()) { + setLastNameError("Please provide a last name"); + valid = false; + } else { + setLastNameError(""); + } + + if (!email.trim()) { + setEmailError("Please provide an email"); + valid = false; + } else if (!/\S+@\S+\.\S+/.test(email)) { + setEmailError("Please provide a valid email"); + valid = false; + } else { + setEmailError(""); + } + + if (!subject.trim()) { + setSubjectError("Please provide a subject"); + valid = false; + } else { + setSubjectError(""); + } + + if (!message.trim()) { + setMessageError("Please provide a message"); + valid = false; + } else { + setMessageError(""); + } + + if (!valid) { + setLoading(false); // Reset loading state if validation fails + // return; + } + + const contactData = { + firstName, + lastName, + email, + subject, + message, + }; + + // Send to API + try { + const res = await fetch("/api/sendEmail", { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify(contactData), + }); + + if (!res.ok) throw new Error("Failed to send email"); + + const data = await res.json(); + alert(data.message || "Email sent successfully!"); + + // Reset form fields after successful submission + setFirstName(""); + setLastName(""); + setEmail(""); + setSubject(""); + setMessage(""); + } catch (error) { + alert("Something went wrong. Try again."); + } finally { + setLoading(false); + } + }; + + return ( +
+ ); +} diff --git a/app/(pages)/_components/ContactForm/ContactForm.module.scss b/app/(pages)/_components/ContactForm/ContactForm.module.scss new file mode 100644 index 0000000..ebeffa0 --- /dev/null +++ b/app/(pages)/_components/ContactForm/ContactForm.module.scss @@ -0,0 +1,134 @@ +@use "@/app/(pages)/_globals/mixins.scss"; + +.form { + display: flex; + flex-direction: column; + align-items: center; + width: 72.22%; + color: var(--dark-grey); + z-index: 1; + + @include mixins.phone { + width: 361px; + } +} + +.formField { + display: flex; + justify-content: space-between; + width: 100%; + + @include mixins.phone { + flex-direction: column; + justify-content: center; + align-items: center; + // gap: 30px; + } +} + +.formInput { + width: 44.92%; + margin-bottom: 20px; + text-align: left; + + @include mixins.phone { + width: 100%; + // margin: 0; + } +} + +.formInput label { + display: block; + font-size: 1rem; + font-family: var(--font2); + font-weight: 400; + line-height: 150%; +} + +.formInputText { + width: 100%; + padding: 12px; + border-radius: 10px; + border: 1px solid var(--dark-grey); + background-color: var(--white); + font-size: 1rem; + color: var(--dark-grey); + transition: + border 0.3s ease, + box-shadow 0.3s ease; +} + +.formInputText.error { + border-color: red; +} + +.formInputText:focus { + border-color: var(--light-blue); + box-shadow: var(--drop-shadow-small); + outline: none; +} + +.formMessage { + width: 100%; + border-radius: var(--card-border-radius); +} + +.formMessage label { + display: none; +} + +.message { + @include mixins.fontStyle(1rem, 400, 150%, var(--font2)); + border-radius: var(--card-border-radius); + border: solid 1px var(--dark-grey); + resize: none; + min-height: 112px; + width: 100%; + padding-top: 12px; + padding-left: 32px; + transition: + border 0.3s ease, + box-shadow 0.3s ease; +} + +.message.error { + border-color: red; +} + +.message:focus { + border-color: var(--light-blue); + box-shadow: var(--drop-shadow-small); + outline: none; +} + +.buttonArea { + display: flex; + flex-direction: row-reverse; + width: 100%; + padding-top: 50px; + + @include mixins.phone { + padding-top: 30px; + } +} + +.sendButton { + outline: none !important; + border: none !important; + font-weight: var(--bold); + font-family: var(--font2); + width: 6rem; + font-size: 1rem; + + @include mixins.phone { + width: 100%; + } +} + +/* Error messages */ +.errorText { + color: red; + font-size: 0.9rem; + margin-bottom: 0.5rem; + text-align: left; +} diff --git a/app/(pages)/contact-us/page.jsx b/app/(pages)/contact-us/page.jsx index 54fcc1a..dc6e4ec 100644 --- a/app/(pages)/contact-us/page.jsx +++ b/app/(pages)/contact-us/page.jsx @@ -1,82 +1,13 @@ "use client"; -import { useState } from "react"; import styles from "./page.module.scss"; import contactData from "../_data/contact-us.json"; import { FaInstagram } from "react-icons/fa"; import Image from "next/image"; +import ContactForm from "../_components/ContactForm/ContactForm"; export default function ContactUs() { const { heading1, heading2 } = contactData.questionsCard; - const [firstName, setFirstName] = useState(""); - const [lastName, setLastName] = useState(""); - const [email, setEmail] = useState(""); - const [subject, setSubject] = useState(""); - const [message, setMessage] = useState(""); - - // Error handling - const [firstNameError, setFirstNameError] = useState(""); - const [lastNameError, setLastNameError] = useState(""); - const [emailError, setEmailError] = useState(""); - const [subjectError, setSubjectError] = useState(""); - - const handleSubmit = async (e) => { - e.preventDefault(); // Prevent automatic page reload - - // Input Validation - let valid = true; - - if (!firstName.trim()) { - setFirstNameError("First Name is required."); - valid = false; - } else { - setFirstNameError(""); - } - - if (!lastName.trim()) { - setLastNameError("Last Name is required."); - valid = false; - } else { - setLastNameError(""); - } - - if (!email.trim()) { - setEmailError("Email is required."); - valid = false; - } else if (!/\S+@\S+\.\S+/.test(email)) { - setEmailError("Invalid email format."); - valid = false; - } else { - setEmailError(""); - } - - if (!subject.trim()) { - setSubjectError("Subject is required."); - valid = false; - } else { - setSubjectError(""); - } - - const contactData = { - firstName, - lastName, - email, - subject, - message, - }; - - if (valid) { - // Send to API - const res = await fetch("/api/sendEmail", { - method: "POST", - headers: { "Content-Type": "application/json" }, - body: JSON.stringify(contactData), - }); - - const data = await res.json(); - alert(data.message || "Something went wrong"); - } - }; return (