Files
Ege Can Komur deff889f0c feat: WordPress tarzı rich editor (TipTap + slash menu + MediaPicker)
WordPress Gutenberg + Notion karışımı blok editor. 4 admin formunda
markdown textarea yerine gerçek WYSIWYG editor.

RichEditor component (components/admin/rich-editor.tsx):
- TipTap v3 (@tiptap/react + starter-kit + link + image + placeholder + underline)
- Üst toolbar (her zaman görünür):
  - B / I / U (bold, italic, underline)
  - H1 / H2 / H3
  - Bullet list / Ordered list / Quote / Code block
  - Link (URL prompt)
  - Görsel ekle (MediaPicker modal)
  - Undo / Redo
- Slash menu: '/' yazınca blok seçim menüsü açılır
  - Notion tarzı keyboard navigation (↓↑ Enter Esc)
  - 8 blok tipi: H1/H2/H3/ul/ol/quote/code/hr
- Image picker modal (toolbar görsel butonundan)
  - Mevcut MediaPicker'ı kullanır
  - 'Yeni görsel yükle' (progress bar ile) + 'Kütüphaneden seç' grid
- HTML çıktı (hidden input ile form'a)
- Mevcut content alanlarıyla backward compat

Formlarda değişiklik (4 dosya):
- app/admin/(protected)/blog/form.tsx → content
- app/admin/(protected)/hizmetler/form.tsx → content
- app/admin/(protected)/projeler/form.tsx → content
- app/admin/(protected)/sektorler/form.tsx → content

Public render (lib/content-render.ts):
- renderContent() yardımcısı:
  - İçerik '<' ile başlıyorsa → HTML (direkt döner)
  - Aksi halde → markdown (marked.parse)
- 4 detay sayfası bu helper'ı kullanıyor (blog/[slug], projeler/[slug],
  hizmetler/[slug], sektor/[slug])
- Eski markdown içerikler hala çalışıyor, yeni içerikler HTML olarak gelir

37 route, build temiz.
2026-05-20 18:34:44 +03:00

143 lines
4.5 KiB
TypeScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import {
Field,
FormActions,
FormShell,
GhostLink,
PageHeader,
PrimaryButton,
Select,
Textarea,
} from "@/components/admin/form";
import { MediaPicker } from "@/components/admin/media-picker";
import { RichEditor } from "@/components/admin/rich-editor";
import { saveBlogPost } from "@/lib/admin-actions";
import type { BlogPostRow } from "@/lib/types";
import { Save } from "lucide-react";
export function BlogForm({ post }: { post?: BlogPostRow }) {
return (
<div>
<PageHeader
title={post ? "Yazıyı düzenle" : "Yeni yazı"}
backHref="/admin/blog"
description="Markdown formatında içerik yazabilirsiniz."
/>
<form action={saveBlogPost}>
{post && <input type="hidden" name="id" value={post.$id} />}
<FormShell>
<div className="grid gap-5 md:grid-cols-2">
<Field
label="Başlık"
name="title"
required
defaultValue={post?.title}
placeholder="Yazı başlığı"
/>
<Field
label="Slug"
name="slug"
defaultValue={post?.slug}
placeholder="otomatik-uretilir"
help="Boş bırakırsanız başlıktan üretilir."
/>
<Field label="Yazar" name="author" defaultValue={post?.author} />
<Select
label="Durum"
name="status"
defaultValue={post?.status ?? "draft"}
options={[
{ value: "draft", label: "Taslak" },
{ value: "published", label: "Yayında" },
]}
/>
<Field
label="Yayın tarihi (ISO)"
name="published_at"
type="datetime-local"
defaultValue={post?.published_at?.slice(0, 16)}
help="Boş bırakırsanız yayına alındığı an kullanılır."
/>
<Field
label="Etiketler"
name="tags"
defaultValue={post?.tags?.join(", ")}
placeholder="seo, web tasarım, kocaeli"
help="Virgülle ayırın."
/>
</div>
<div className="mt-5 space-y-5">
<Textarea
label="Özet"
name="excerpt"
defaultValue={post?.excerpt}
rows={3}
placeholder="Liste/kart görünümünde gösterilecek kısa özet"
/>
<div>
<span className="text-sm font-medium text-[var(--navy)]">
İçerik
</span>
<div className="mt-1.5">
<RichEditor
name="content"
defaultValue={post?.content}
placeholder="Yazıya başlayın… `/` ile başlık, görsel, liste ekleyin"
minHeight={500}
/>
</div>
</div>
<MediaPicker
label="Kapak görseli"
name="cover_image"
defaultValue={post?.cover_image}
help="Yazı listesinde ve detay sayfasının üst kısmında gösterilir."
/>
<input
type="hidden"
name="cover_file_id"
defaultValue={post?.cover_file_id ?? ""}
/>
</div>
<h3 className="mt-8 text-sm font-semibold uppercase tracking-wider text-[var(--muted)]">
SEO
</h3>
<div className="mt-3 grid gap-5 md:grid-cols-2">
<Field
label="SEO başlığı"
name="seo_title"
defaultValue={post?.seo_title}
help="Boş bırakırsanız yazı başlığı kullanılır."
/>
<MediaPicker
label="SEO OG görseli"
name="seo_image"
defaultValue={post?.seo_image}
help="Sosyal medyada paylaşılınca görünecek görsel (1200×630 ideal)."
/>
</div>
<div className="mt-5">
<Textarea
label="SEO açıklaması"
name="seo_description"
defaultValue={post?.seo_description}
rows={2}
help="150160 karakter ideal."
/>
</div>
<FormActions>
<GhostLink href="/admin/blog">İptal</GhostLink>
<PrimaryButton>
<Save className="size-4" /> Kaydet
</PrimaryButton>
</FormActions>
</FormShell>
</form>
</div>
);
}