f833d429fc
Backend altyapısı: - 4 yeni Appwrite tablosu: blog_posts, testimonials, seo_pages, seo_settings - Appwrite Storage bucket: kovak-yazilim-media (görsel yüklemeleri) - Appwrite Auth ile session cookie tabanlı koruma Admin paneli (/admin): - Login akışı (email/password) + protected layout - Dashboard: sayım kartları + hızlı aksiyonlar - Blog CRUD: markdown content, kapak görseli, draft/published, SEO alanları - Services CRUD: lucide ikon seçici - Projects CRUD: teknoloji etiketleri, live URL - Testimonials CRUD: puanlama - SEO yöneticisi: global ayarlar + sayfa bazlı override - Mesaj inbox: status filtreleme + güncelleme - Medya kütüphanesi: Appwrite Storage upload/delete Public: - /blog ve /blog/[slug] sayfaları (markdown render) - Anasayfaya Testimonials bölümü - Tüm public sayfalarda generateMetadata + seo_pages override - Header'a Blog linki Route yapısı: - app/(site)/ — public site, Header/Footer ortak - app/admin/login — auth dışı - app/admin/(protected)/ — requireUser() korumalı 23 route üretiliyor, public static, admin dynamic.
138 lines
5.2 KiB
TypeScript
138 lines
5.2 KiB
TypeScript
import Link from "next/link";
|
||
import { Mail, Phone } from "lucide-react";
|
||
import { PageHeader } from "@/components/admin/form";
|
||
import { DeleteButton } from "@/components/admin/delete-button";
|
||
import { listMessages } from "@/lib/data";
|
||
import { deleteMessage, updateMessageStatus } from "@/lib/admin-actions";
|
||
import type { ContactMessageRow } from "@/lib/types";
|
||
|
||
const FILTERS = [
|
||
{ value: "", label: "Tümü" },
|
||
{ value: "new", label: "Yeni" },
|
||
{ value: "read", label: "Okundu" },
|
||
{ value: "replied", label: "Yanıtlandı" },
|
||
{ value: "archived", label: "Arşiv" },
|
||
];
|
||
|
||
export default async function MessagesAdminPage({
|
||
searchParams,
|
||
}: {
|
||
searchParams: Promise<{ filter?: string }>;
|
||
}) {
|
||
const sp = await searchParams;
|
||
const filter = sp.filter as ContactMessageRow["status"] | undefined;
|
||
const messages = await listMessages(filter || undefined);
|
||
|
||
return (
|
||
<div>
|
||
<PageHeader
|
||
title="İletişim mesajları"
|
||
description="Forma gelen mesajları yönetin."
|
||
/>
|
||
|
||
<div className="mt-6 flex flex-wrap gap-2">
|
||
{FILTERS.map((f) => {
|
||
const active = (filter ?? "") === f.value;
|
||
return (
|
||
<Link
|
||
key={f.value}
|
||
href={`/admin/iletisim${f.value ? `?filter=${f.value}` : ""}`}
|
||
className={`rounded-full border px-3 py-1.5 text-xs font-medium transition ${
|
||
active
|
||
? "border-[var(--navy)] bg-[var(--navy)] text-white"
|
||
: "border-[var(--border)] bg-white text-[var(--muted)] hover:border-[var(--navy)]"
|
||
}`}
|
||
>
|
||
{f.label}
|
||
</Link>
|
||
);
|
||
})}
|
||
</div>
|
||
|
||
<div className="mt-6 space-y-3">
|
||
{messages.length === 0 && (
|
||
<div className="rounded-2xl border border-dashed border-[var(--border)] bg-white p-12 text-center text-sm text-[var(--muted)]">
|
||
Bu filtrede mesaj yok.
|
||
</div>
|
||
)}
|
||
{messages.map((m) => (
|
||
<article
|
||
key={m.$id}
|
||
className="rounded-2xl border border-[var(--border)] bg-white p-5"
|
||
>
|
||
<header className="flex flex-wrap items-start justify-between gap-3">
|
||
<div>
|
||
<div className="flex items-center gap-2">
|
||
<p className="font-semibold text-[var(--navy)]">{m.name}</p>
|
||
<StatusBadge status={m.status} />
|
||
</div>
|
||
<div className="mt-1 flex flex-wrap items-center gap-3 text-xs text-[var(--muted)]">
|
||
<a href={`mailto:${m.email}`} className="inline-flex items-center gap-1 hover:text-[var(--navy)]">
|
||
<Mail className="size-3" />
|
||
{m.email}
|
||
</a>
|
||
{m.phone && (
|
||
<a href={`tel:${m.phone}`} className="inline-flex items-center gap-1 hover:text-[var(--navy)]">
|
||
<Phone className="size-3" />
|
||
{m.phone}
|
||
</a>
|
||
)}
|
||
<span>{new Date(m.$createdAt).toLocaleString("tr-TR")}</span>
|
||
</div>
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<form action={updateMessageStatus} className="flex items-center gap-1">
|
||
<input type="hidden" name="id" value={m.$id} />
|
||
<select
|
||
name="status"
|
||
defaultValue={m.status ?? "new"}
|
||
className="rounded-md border border-[var(--border)] bg-white px-2 py-1 text-xs"
|
||
>
|
||
<option value="new">Yeni</option>
|
||
<option value="read">Okundu</option>
|
||
<option value="replied">Yanıtlandı</option>
|
||
<option value="archived">Arşiv</option>
|
||
</select>
|
||
<button
|
||
type="submit"
|
||
className="rounded-md bg-[var(--navy)] px-2.5 py-1 text-xs font-medium text-white hover:bg-[var(--navy-700)]"
|
||
>
|
||
Güncelle
|
||
</button>
|
||
</form>
|
||
<form action={deleteMessage}>
|
||
<input type="hidden" name="id" value={m.$id} />
|
||
<DeleteButton />
|
||
</form>
|
||
</div>
|
||
</header>
|
||
{m.subject && (
|
||
<p className="mt-3 text-sm font-medium text-[var(--navy)]">
|
||
{m.subject}
|
||
</p>
|
||
)}
|
||
<p className="mt-2 whitespace-pre-wrap text-sm leading-relaxed text-[var(--foreground)]">
|
||
{m.message}
|
||
</p>
|
||
</article>
|
||
))}
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
function StatusBadge({ status }: { status?: string | null }) {
|
||
const map: Record<string, { label: string; cls: string }> = {
|
||
new: { label: "Yeni", cls: "bg-red-100 text-red-800" },
|
||
read: { label: "Okundu", cls: "bg-amber-100 text-amber-800" },
|
||
replied: { label: "Yanıtlandı", cls: "bg-green-100 text-green-800" },
|
||
archived: { label: "Arşiv", cls: "bg-gray-100 text-gray-700" },
|
||
};
|
||
const meta = map[status ?? "new"] ?? map.new;
|
||
return (
|
||
<span className={`rounded-full px-2 py-0.5 text-xs font-medium ${meta.cls}`}>
|
||
{meta.label}
|
||
</span>
|
||
);
|
||
}
|