282 lines
8.7 KiB
TypeScript
282 lines
8.7 KiB
TypeScript
"use client";
|
|
|
|
import * as React from "react";
|
|
import { useState } from "react";
|
|
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardDescription,
|
|
CardFooter,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/components/ui/card";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/components/ui/select";
|
|
import { Textarea } from "../ui/textarea";
|
|
import { SubmitButton } from "../submit-button";
|
|
import Link from "next/link";
|
|
import { toast } from "@/hooks/use-toast";
|
|
import { sendContactEmail } from "@/actions/auth/contact-us";
|
|
import { TValidator } from "@/utils/validator";
|
|
|
|
export function ContactUsForm() {
|
|
const typeMessage = [
|
|
{ value: "1", label: "Request to become a user" },
|
|
{ value: "2", label: "OTP problem" },
|
|
{ value: "3", label: "Request for a feature" },
|
|
{ value: "4", label: "Other" },
|
|
];
|
|
|
|
// State untuk form data
|
|
const [formData, setFormData] = useState({
|
|
name: "",
|
|
email: "",
|
|
phone: "",
|
|
typeMessage: "",
|
|
message: "",
|
|
});
|
|
|
|
// State untuk error messages
|
|
const [errors, setErrors] = useState<Record<string, string>>({});
|
|
// Loading state
|
|
const [isSubmitting, setIsSubmitting] = useState(false);
|
|
|
|
// Handle input change
|
|
const handleChange = (
|
|
e: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>
|
|
) => {
|
|
const { id, value } = e.target;
|
|
setFormData((prev) => ({ ...prev, [id]: value }));
|
|
|
|
// Clear error when typing
|
|
if (errors[id]) {
|
|
setErrors((prev) => {
|
|
const newErrors = { ...prev };
|
|
delete newErrors[id];
|
|
return newErrors;
|
|
});
|
|
}
|
|
};
|
|
|
|
// Handle select change
|
|
const handleSelectChange = (value: string) => {
|
|
setFormData((prev) => ({ ...prev, typeMessage: value }));
|
|
|
|
// Clear error when selecting
|
|
if (errors.typeMessage) {
|
|
setErrors((prev) => {
|
|
const newErrors = { ...prev };
|
|
delete newErrors.typeMessage;
|
|
return newErrors;
|
|
});
|
|
}
|
|
};
|
|
|
|
// Handle form submission
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
e.preventDefault();
|
|
|
|
// Client-side validation
|
|
const validation = TValidator.validateContactForm(formData);
|
|
|
|
if (!validation.success) {
|
|
setErrors(validation.errors);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
setIsSubmitting(true);
|
|
|
|
// Call server action to send email
|
|
const result = await sendContactEmail(formData);
|
|
|
|
if (!result.success) {
|
|
// Handle server-side validation errors
|
|
if (result.errors) {
|
|
setErrors(result.errors);
|
|
} else {
|
|
toast({
|
|
title: "Error",
|
|
description:
|
|
result.error || "Failed to send message. Please try again.",
|
|
variant: "destructive",
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Success!
|
|
toast({
|
|
title: "Success",
|
|
description:
|
|
result.message || "Your message has been sent successfully!",
|
|
});
|
|
|
|
// Reset form and errors after successful submission
|
|
setFormData({
|
|
name: "",
|
|
email: "",
|
|
phone: "",
|
|
typeMessage: "",
|
|
message: "",
|
|
});
|
|
setErrors({});
|
|
} catch (error) {
|
|
console.error("Error submitting form:", error);
|
|
toast({
|
|
title: "Error",
|
|
description: "An unexpected error occurred. Please try again later.",
|
|
variant: "destructive",
|
|
});
|
|
} finally {
|
|
setIsSubmitting(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Card className="w-[500px] bg-[#171717] border-none text-white">
|
|
<CardHeader>
|
|
<CardTitle className="text-3xl font-bold">Contact Us</CardTitle>
|
|
<CardDescription className="text-gray-400">
|
|
Fill in the form below to contact the admin
|
|
</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<form className="space-y-4" onSubmit={handleSubmit}>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="name" className="text-sm text-gray-300">
|
|
Name
|
|
</Label>
|
|
<Input
|
|
id="name"
|
|
placeholder="John doe"
|
|
className={`bg-[#1C1C1C] border-gray-800 text-white placeholder:text-gray-500 focus:border-emerald-600 focus:ring-emerald-600 ${
|
|
errors.name ? "border-red-500" : ""
|
|
}`}
|
|
value={formData.name}
|
|
onChange={handleChange}
|
|
disabled={isSubmitting}
|
|
/>
|
|
{errors.name && (
|
|
<p className="text-red-500 text-xs mt-1">{errors.name}</p>
|
|
)}
|
|
</div>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="email" className="text-sm text-gray-300">
|
|
Email
|
|
</Label>
|
|
<Input
|
|
id="email"
|
|
placeholder="example@gmail.com"
|
|
className={`bg-[#1C1C1C] border-gray-800 text-white placeholder:text-gray-500 focus:border-emerald-600 focus:ring-emerald-600 ${
|
|
errors.email ? "border-red-500" : ""
|
|
}`}
|
|
value={formData.email}
|
|
onChange={handleChange}
|
|
disabled={isSubmitting}
|
|
/>
|
|
{errors.email && (
|
|
<p className="text-red-500 text-xs mt-1">{errors.email}</p>
|
|
)}
|
|
</div>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="phone" className="text-sm text-gray-300">
|
|
Phone
|
|
</Label>
|
|
<Input
|
|
id="phone"
|
|
placeholder="08123456789"
|
|
className={`bg-[#1C1C1C] border-gray-800 text-white placeholder:text-gray-500 focus:border-emerald-600 focus:ring-emerald-600 ${
|
|
errors.phone ? "border-red-500" : ""
|
|
}`}
|
|
value={formData.phone}
|
|
onChange={handleChange}
|
|
disabled={isSubmitting}
|
|
/>
|
|
{errors.phone && (
|
|
<p className="text-red-500 text-xs mt-1">{errors.phone}</p>
|
|
)}
|
|
</div>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="typemessage" className="text-sm text-gray-300">
|
|
Type message
|
|
</Label>
|
|
<Select
|
|
value={formData.typeMessage}
|
|
onValueChange={handleSelectChange}
|
|
disabled={isSubmitting}
|
|
>
|
|
<SelectTrigger
|
|
id="typemessage"
|
|
className={`bg-[#1C1C1C] border-gray-800 text-white focus:border-emerald-600 focus:ring-emerald-600 ${
|
|
errors.typeMessage ? "border-red-500" : ""
|
|
}`}
|
|
>
|
|
<SelectValue placeholder="Select" />
|
|
</SelectTrigger>
|
|
<SelectContent className="bg-[#1C1C1C] border-gray-800 text-white">
|
|
{typeMessage.map((message) => (
|
|
<SelectItem
|
|
key={message.value}
|
|
value={message.value}
|
|
className="focus:bg-emerald-600 focus:text-white"
|
|
>
|
|
{message.label}
|
|
</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
{errors.typeMessage && (
|
|
<p className="text-red-500 text-xs mt-1">{errors.typeMessage}</p>
|
|
)}
|
|
</div>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="message" className="text-sm text-gray-300">
|
|
Message
|
|
</Label>
|
|
<Textarea
|
|
id="message"
|
|
placeholder="Your message here..."
|
|
className={`resize-none h-24 bg-[#1C1C1C] border-gray-800 text-white placeholder:text-gray-500 focus:border-emerald-600 focus:ring-emerald-600 ${
|
|
errors.message ? "border-red-500" : ""
|
|
}`}
|
|
value={formData.message}
|
|
onChange={handleChange}
|
|
disabled={isSubmitting}
|
|
/>
|
|
{errors.message && (
|
|
<p className="text-red-500 text-xs mt-1">{errors.message}</p>
|
|
)}
|
|
</div>
|
|
<CardFooter className="flex flex-col items-center space-y-4 px-0">
|
|
<SubmitButton
|
|
type="submit"
|
|
className="w-full bg-emerald-600 hover:bg-emerald-700 text-white"
|
|
disabled={isSubmitting}
|
|
>
|
|
{isSubmitting ? "Sending..." : "Send"}
|
|
</SubmitButton>
|
|
<div className="text-center text-lg space-x-2">
|
|
<span className="text-gray-400">Already have an account?</span>
|
|
<Link
|
|
href="/sign-in"
|
|
className="text-white hover:text-emerald-500"
|
|
>
|
|
Login
|
|
</Link>
|
|
</div>
|
|
</CardFooter>
|
|
</form>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|