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.
30 lines
940 B
TypeScript
30 lines
940 B
TypeScript
import type { Metadata } from "next";
|
||
import { SectionTitle } from "@/components/section-title";
|
||
import { ProjectsGrid } from "@/components/projects-grid";
|
||
import { listProjects } from "@/lib/data";
|
||
import { buildMetadata } from "@/lib/seo";
|
||
|
||
export async function generateMetadata(): Promise<Metadata> {
|
||
return buildMetadata("/projeler", {
|
||
title: "Projeler",
|
||
description: "Tamamladığımız web, mobil ve CRM projelerinden seçkiler.",
|
||
});
|
||
}
|
||
|
||
export default async function ProjectsPage() {
|
||
const projects = await listProjects();
|
||
|
||
return (
|
||
<div className="mx-auto max-w-7xl px-6 py-20">
|
||
<SectionTitle
|
||
eyebrow="Çalışmalar"
|
||
title="Müşterilerimiz için ürettiğimiz işler"
|
||
description="Strateji, tasarım ve mühendislik bir araya geldiğinde ortaya çıkan ürünler."
|
||
/>
|
||
<div className="mt-14">
|
||
<ProjectsGrid projects={projects} />
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|