feat: Çözümler bölümü + mobil menü; admin parser düzeltmeleri

- Çözümler: solutions tablosu, /cozumler liste + detay sayfası, anasayfa
  bölümü, tam admin CRUD (/admin/cozumler), header & footer linkleri,
  projelerde solution_slug ilişkisi, services-grid genelleştirildi
- Mobil menü (hamburger drawer) eklendi — header artık < lg'de gezilebilir
- Site ayarları parser: textarea CRLF (\r\n) normalizasyonu — neden biz,
  süreç adımları, değerler ve SSS blokları artık doğru parçalanıyor
- homepage_faq + garanti (title/description/items) saveSiteSettings'e
  bağlandı (daha önce hiç kaydedilmiyordu)
This commit is contained in:
egecankomur
2026-06-02 18:21:58 +03:00
parent f49df9cbeb
commit 2e001680bf
21 changed files with 1191 additions and 27 deletions
+17
View File
@@ -7,6 +7,7 @@ import type {
IndustryRow,
ProjectRow,
ServiceRow,
SolutionRow,
SeoPageRow,
SeoSettingsRow,
SiteSettingsRow,
@@ -44,14 +45,22 @@ export async function listServices(opts?: { featured?: boolean }) {
return safeList<ServiceRow>(TABLES.services, q);
}
export async function listSolutions(opts?: { featured?: boolean }) {
const q = [Q.orderAsc("order"), Q.limit(50)];
if (opts?.featured) q.unshift(Q.equal("featured", true));
return safeList<SolutionRow>(TABLES.solutions, q);
}
export async function listProjects(opts?: {
featured?: boolean;
limit?: number;
serviceSlug?: string;
solutionSlug?: string;
}) {
const q = [Q.orderDesc("year"), Q.limit(opts?.limit ?? 50)];
if (opts?.featured) q.unshift(Q.equal("featured", true));
if (opts?.serviceSlug) q.unshift(Q.equal("service_slug", opts.serviceSlug));
if (opts?.solutionSlug) q.unshift(Q.equal("solution_slug", opts.solutionSlug));
return safeList<ProjectRow>(TABLES.projects, q);
}
@@ -63,6 +72,14 @@ export async function getServiceBySlug(slug: string): Promise<ServiceRow | null>
return res[0] ?? null;
}
export async function getSolutionBySlug(slug: string): Promise<SolutionRow | null> {
const res = await safeList<SolutionRow>(TABLES.solutions, [
Q.equal("slug", slug),
Q.limit(1),
]);
return res[0] ?? null;
}
export async function getProjectBySlug(slug: string): Promise<ProjectRow | null> {
const res = await safeList<ProjectRow>(TABLES.projects, [
Q.equal("slug", slug),