GRV-Summit-Site/components/layout/SiteEntryPrompt.tsx
kirukib cb404ec079
Some checks failed
Deploy to Cloudflare Workers (OpenNext) / deploy (push) Has been cancelled
Align site colors with GRV brand book palette.
Centralize primary, secondary, tertiary, and neutral tokens and apply them across theme variables and UI components.

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-06-04 14:45:22 +03:00

127 lines
3.7 KiB
TypeScript

"use client";
import { useEffect, useState } from "react";
import Link from "next/link";
import { usePathname } from "next/navigation";
import { X } from "lucide-react";
import { dataConsent } from "@/content/consent";
import { Button } from "@/components/ui/button";
import { Checkbox } from "@/components/ui/checkbox";
import { Label } from "@/components/ui/label";
import { cn } from "@/lib/utils";
const copy = dataConsent.siteEntry;
function hasAcceptedConsent(): boolean {
if (typeof window === "undefined") return false;
return localStorage.getItem(copy.storageKey) === "accepted";
}
export function SiteEntryPrompt() {
const pathname = usePathname();
const [open, setOpen] = useState(false);
const [visible, setVisible] = useState(false);
const [checked, setChecked] = useState(false);
useEffect(() => {
if (hasAcceptedConsent()) {
setOpen(false);
setVisible(false);
return;
}
setOpen(true);
setChecked(false);
const show = requestAnimationFrame(() => setVisible(true));
return () => cancelAnimationFrame(show);
}, [pathname]);
const dismiss = () => {
setVisible(false);
window.setTimeout(() => setOpen(false), 280);
};
const handleAccept = () => {
if (!checked) return;
localStorage.setItem(copy.storageKey, "accepted");
dismiss();
};
const handleDecline = () => {
dismiss();
};
if (!open) return null;
return (
<div
role="dialog"
aria-labelledby="entry-consent-title"
aria-describedby="entry-consent-desc"
className={cn(
"fixed bottom-4 right-4 z-[200] w-[min(calc(100vw-2rem),24rem)] rounded-2xl border border-[#37a47a]/15 bg-white p-5 shadow-[0_12px_40px_rgba(13,61,38,0.18)] transition-all duration-500 ease-out",
visible ? "translate-y-0 opacity-100" : "translate-y-3 opacity-0"
)}
>
<button
type="button"
onClick={handleDecline}
className="absolute top-3 right-3 rounded-full p-1 text-[#5b5b5b] transition-colors hover:bg-[#37a47a]/8 hover:text-[#30614c]"
aria-label="Decline and close"
>
<X className="size-4" />
</button>
<h2
id="entry-consent-title"
className="pr-6 text-lg font-bold leading-snug text-[#30614c]"
>
{copy.title}
</h2>
<p id="entry-consent-desc" className="mt-2 text-sm leading-relaxed text-[#5b5b5b]">
{copy.description}
</p>
<div className="mt-4 flex items-start gap-3 rounded-lg border border-[#37a47a]/12 bg-[#e8f2ec]/80 p-3">
<Checkbox
id="site-entry-consent"
checked={checked}
onCheckedChange={(v) => setChecked(v === true)}
className="mt-0.5 border-[#37a47a]/40 data-[state=checked]:border-[#37a47a] data-[state=checked]:bg-[#37a47a]"
/>
<Label
htmlFor="site-entry-consent"
className="text-sm font-normal leading-snug text-[#5b5b5b]"
>
{copy.checkboxLabel}{" "}
<Link
href={copy.privacyHref}
className="font-medium text-[#37a47a] underline underline-offset-2"
>
{dataConsent.privacyLinkText}
</Link>
.
</Label>
</div>
<div className="mt-4 flex flex-col gap-2 sm:flex-row">
<Button
type="button"
disabled={!checked}
onClick={handleAccept}
className="rounded-full bg-[#37a47a] text-white hover:bg-[#30614c] disabled:opacity-50"
>
{copy.acceptCta}
</Button>
<Button
type="button"
variant="outline"
onClick={handleDecline}
className="rounded-full border-[#37a47a]/30 text-[#37a47a] hover:bg-[#37a47a]/6"
>
{copy.declineCta}
</Button>
</div>
</div>
);
}