dbc55e7527
Mevcut sorun: - Her görsel için medya sayfasına git, yükle, URL kopyala, forma yapıştır → 4 adım - Sürükle-bırak yok, progress yok, hangi dosyanın yüklendiği belirsiz Çözüm: MediaPicker component (tek/çoklu mode) API route'ları: - POST /api/admin/media/upload — session auth + Appwrite Storage upload - GET /api/admin/media/list — kütüphane modal için dosya listesi Component özellikleri: - Sürükle-bırak drop zone (hover state ile) - Multiple file upload (çoklu mode) - XHR ile gerçek progress bar (%) — Server Action ile alınamazdı - Görsel preview (single: aspect-video, multiple: aspect-square grid) - Hover'da × ile kaldırma - Multiple mode'da sırasını değiştirme - 'Kütüphaneden seç' modal — daha önce yüklenmiş görselleri grid'de göster, tıklayınca seç - Error handling (dosya boyutu, ağ hatası vb.) - Başarılı yüklemeyi 2 saniye gösterip kaybetme Form alanları → MediaPicker (URL field'ları kaldırıldı): - Blog: cover_image, seo_image - Hizmet: hero_image - Proje: image_url (kapak), gallery (çoklu) - Referans: image_url - Sektör: hero_image - Ekip: photo_url - SEO sayfa: og_image - SEO global: default_og_image - Site Settings: client_logos (çoklu) Backward compat: form data formatı aynı kalıyor — hidden input ile URL satır satır. admin-actions değişmedi. URL elle yapıştırmak hala mümkün (kütüphaneden URL kopyala).
141 lines
4.3 KiB
TypeScript
141 lines
4.3 KiB
TypeScript
import { Save } from "lucide-react";
|
||
import {
|
||
Checkbox,
|
||
Field,
|
||
FormActions,
|
||
FormShell,
|
||
GhostLink,
|
||
PageHeader,
|
||
PrimaryButton,
|
||
Textarea,
|
||
} from "@/components/admin/form";
|
||
import { MediaPicker } from "@/components/admin/media-picker";
|
||
import { saveService } from "@/lib/admin-actions";
|
||
import type { FaqItem, ServiceRow } from "@/lib/types";
|
||
|
||
const ICON_OPTIONS = [
|
||
"Globe",
|
||
"ShoppingCart",
|
||
"Smartphone",
|
||
"Code2",
|
||
"Users",
|
||
"TrendingUp",
|
||
"Share2",
|
||
"Megaphone",
|
||
"Layers",
|
||
];
|
||
|
||
function faqToText(items?: string[] | null): string {
|
||
if (!items) return "";
|
||
const parsed: FaqItem[] = [];
|
||
for (const raw of items) {
|
||
try {
|
||
const obj = JSON.parse(raw) as Partial<FaqItem>;
|
||
if (obj.q && obj.a) parsed.push({ q: obj.q, a: obj.a });
|
||
} catch {
|
||
/* ignore */
|
||
}
|
||
}
|
||
return parsed.map((it) => `${it.q}\n${it.a}`).join("\n---\n");
|
||
}
|
||
|
||
export function ServiceForm({ service }: { service?: ServiceRow }) {
|
||
return (
|
||
<div>
|
||
<PageHeader
|
||
title={service ? "Hizmeti düzenle" : "Yeni hizmet"}
|
||
backHref="/admin/hizmetler"
|
||
/>
|
||
<form action={saveService}>
|
||
{service && <input type="hidden" name="id" value={service.$id} />}
|
||
<FormShell>
|
||
<div className="grid gap-5 md:grid-cols-2">
|
||
<Field label="Başlık" name="title" required defaultValue={service?.title} />
|
||
<Field label="Slug" name="slug" defaultValue={service?.slug} />
|
||
<Field
|
||
label="Sıra"
|
||
name="order"
|
||
type="number"
|
||
defaultValue={service?.order ?? 0}
|
||
/>
|
||
<label className="block">
|
||
<span className="text-sm font-medium text-[var(--navy)]">İkon</span>
|
||
<select
|
||
name="icon"
|
||
defaultValue={service?.icon ?? "Layers"}
|
||
className="mt-1.5 w-full rounded-xl border border-[var(--border)] bg-white px-4 py-2.5 text-sm outline-none focus:border-[var(--sky)] focus:ring-2 focus:ring-[var(--sky)]/20"
|
||
>
|
||
{ICON_OPTIONS.map((i) => (
|
||
<option key={i} value={i}>
|
||
{i}
|
||
</option>
|
||
))}
|
||
</select>
|
||
</label>
|
||
<MediaPicker
|
||
label="Hero görsel"
|
||
name="hero_image"
|
||
defaultValue={service?.hero_image}
|
||
help="Detay sayfasının üst kısmında gösterilir (opsiyonel)."
|
||
/>
|
||
</div>
|
||
|
||
<div className="mt-5 space-y-5">
|
||
<Textarea
|
||
label="Kısa açıklama (kart için)"
|
||
name="description"
|
||
required
|
||
defaultValue={service?.description}
|
||
rows={3}
|
||
help="Listede ve anasayfa kartında gösterilir."
|
||
/>
|
||
|
||
<Textarea
|
||
label="Detay içerik (Markdown)"
|
||
name="content"
|
||
defaultValue={service?.content}
|
||
rows={10}
|
||
placeholder={"## Yaklaşım\n\nMarkdown desteklenir…"}
|
||
help="Hizmet detay sayfasında ana içerik olarak gösterilir."
|
||
/>
|
||
|
||
<Textarea
|
||
label="Özellikler"
|
||
name="features"
|
||
defaultValue={service?.features?.join(", ")}
|
||
rows={3}
|
||
placeholder="SEO uyumlu kod, Mobil responsive, Hızlı yüklenme, …"
|
||
help="Virgülle ayırın. Detay sayfasında checklist olarak gösterilir."
|
||
/>
|
||
|
||
<Textarea
|
||
label="SSS"
|
||
name="faq"
|
||
defaultValue={faqToText(service?.faq)}
|
||
rows={8}
|
||
placeholder={
|
||
"Soru 1?\nCevap 1 burada.\n---\nSoru 2?\nCevap 2 burada."
|
||
}
|
||
help="Her soru/cevap blokunu '---' ile ayırın. İlk satır soru, kalanı cevap."
|
||
/>
|
||
</div>
|
||
|
||
<div className="mt-5">
|
||
<Checkbox
|
||
label="Öne çıkar (Anasayfada göster)"
|
||
name="featured"
|
||
defaultChecked={service?.featured ?? false}
|
||
/>
|
||
</div>
|
||
<FormActions>
|
||
<GhostLink href="/admin/hizmetler">İptal</GhostLink>
|
||
<PrimaryButton>
|
||
<Save className="size-4" /> Kaydet
|
||
</PrimaryButton>
|
||
</FormActions>
|
||
</FormShell>
|
||
</form>
|
||
</div>
|
||
);
|
||
}
|