d407f0b3e9
CI/CD / CI · API (dotnet build + test) (push) Successful in 40s
CI/CD / CI · Admin API (dotnet build) (push) Successful in 29s
CI/CD / CI · Dashboard (tsc) (push) Successful in 1m10s
CI/CD / CI · Admin Web (tsc) (push) Successful in 38s
CI/CD / CI · Website (tsc) (push) Successful in 45s
CI/CD / CI · Koja (tsc) (push) Successful in 49s
CI/CD / Deploy · all services (push) Has been cancelled
- web/website: manifest referenced /icon-192.png and /icon-512.png that didn't exist (broken favicon). Added the real transparent Meezi mark (32/180/192/512) + wired icons into metadata. OG image stays dynamic. - web/admin: had no metadata or icons at all. Added title template, favicon/apple icons (icons/ dir), themeColor, noindex. Dashboard + Koja already carry the real logo; web apps are now consistent. Mobile launcher icons will be handled with the Android packaging task. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
82 lines
2.3 KiB
TypeScript
82 lines
2.3 KiB
TypeScript
import type { Metadata, Viewport } from "next";
|
|
import { NextIntlClientProvider } from "next-intl";
|
|
import { getMessages, setRequestLocale } from "next-intl/server";
|
|
import { notFound } from "next/navigation";
|
|
import localFont from "next/font/local";
|
|
import { routing } from "@/i18n/routing";
|
|
import { Providers } from "@/components/providers";
|
|
import "../globals.css";
|
|
|
|
export const metadata: Metadata = {
|
|
title: { default: "مدیریت سامانه میزی", template: "%s — مدیریت میزی" },
|
|
description: "پنل مدیریت سامانه میزی",
|
|
icons: {
|
|
icon: [
|
|
{ url: "/icons/icon-32.png", sizes: "32x32", type: "image/png" },
|
|
{ url: "/icons/icon-48.png", sizes: "48x48", type: "image/png" },
|
|
{ url: "/icons/icon-192.png", sizes: "192x192", type: "image/png" },
|
|
],
|
|
shortcut: "/icons/icon-32.png",
|
|
apple: "/icons/icon-180.png",
|
|
},
|
|
robots: { index: false, follow: false }, // internal admin panel
|
|
};
|
|
|
|
export const viewport: Viewport = {
|
|
themeColor: "#0F6E56",
|
|
width: "device-width",
|
|
initialScale: 1,
|
|
};
|
|
|
|
const vazirmatn = localFont({
|
|
src: "../../fonts/Vazirmatn-Variable.woff2",
|
|
variable: "--font-vazirmatn",
|
|
display: "swap",
|
|
weight: "100 900",
|
|
});
|
|
|
|
const inter = localFont({
|
|
src: "../../fonts/Inter-Variable.woff2",
|
|
variable: "--font-inter",
|
|
display: "swap",
|
|
weight: "100 900",
|
|
});
|
|
|
|
export function generateStaticParams() {
|
|
return routing.locales.map((locale) => ({ locale }));
|
|
}
|
|
|
|
export default async function LocaleLayout({
|
|
children,
|
|
params: { locale },
|
|
}: {
|
|
children: React.ReactNode;
|
|
params: { locale: string };
|
|
}) {
|
|
if (!routing.locales.includes(locale as "fa" | "ar" | "en")) {
|
|
notFound();
|
|
}
|
|
|
|
setRequestLocale(locale);
|
|
const messages = await getMessages();
|
|
const dir = locale === "en" ? "ltr" : "rtl";
|
|
const fontClass =
|
|
locale === "en"
|
|
? inter.variable
|
|
: vazirmatn.variable;
|
|
|
|
return (
|
|
<html lang={locale} dir={dir}>
|
|
<body
|
|
className={`${fontClass} font-sans antialiased ${
|
|
locale === "en" ? "font-[family-name:var(--font-inter)]" : "font-[family-name:var(--font-vazirmatn)]"
|
|
}`}
|
|
>
|
|
<NextIntlClientProvider messages={messages}>
|
|
<Providers>{children}</Providers>
|
|
</NextIntlClientProvider>
|
|
</body>
|
|
</html>
|
|
);
|
|
}
|