Drain the long-tail audit queue captured in alpha-uat-master.md.
- next-intl ripped out (zero useTranslations callers ever existed):
package.json, next.config.ts plugin wrap, src/i18n/, messages/, and
the layout NextIntlClientProvider all gone; <html lang="en"> hardcoded.
- RTL lint nudge added: warn-only no-restricted-syntax on physical
Tailwind utilities (ml-/mr-/pl-/pr-/text-left/text-right/border-l/
border-r/rounded-l-/rounded-r-) inside JSX className literals.
Existing ~1,000 sites grandfathered; new code trends toward logical.
- Icon-only button accessibility lint: jsx-a11y/control-has-associated-
label enabled at warn; 4 empty <th>/<td> action placeholders gain
sr-only labels.
- Currency: SUPPORTED_CURRENCIES drops the hardcoded English labels;
new currencyLabel(code, locale?) helper resolves via Intl.DisplayNames.
CurrencySelect + settings-manager migrated.
- Date locale sweep: 7 surfaces flip from toLocaleString('en-GB'|'en-US')
to toLocaleString(undefined, ...) so dates honour runtime locale.
- Dialog/Sheet width: 10 document/EOI/entity-form dialogs gain a
lg:max-w-4xl or lg:max-w-5xl step so wide desktops get breathing room.
- PaymentsSection collapsed-bar: slim one-line bar showing
"Payments - Not received yet" or "Payments - \$X received - N payments
- Expand"; per-interest collapse state persists in localStorage; the
RecordPayment flow auto-expands.
- muted-foreground opacity sweep: 10 text-bearing
text-muted-foreground/{60,70,80} hits dropped to plain
text-muted-foreground for AA contrast on muted bg. Icon-only
(aria-hidden) opacity hits left as-is.
- Micro-type bump: text-[10px] and text-[11px] -> text-xs (12px)
across 87 files in src/components + src/app. Pure mechanical sweep.
- Audit-doc cleanup: alpha-uat-master.md stale 2026-05-25 summary
rewritten with cumulative state through today. Items genuinely still
open are now a short long-tail list.
- New docs/marketing-site-followups.md: Umami Phase 4a/3/5, email
pixel E2E verification, and website-cutover work parked here so
they don't get lost in the CRM audit doc.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
88 lines
2.6 KiB
TypeScript
88 lines
2.6 KiB
TypeScript
import type { Metadata, Viewport } from 'next';
|
|
import Script from 'next/script';
|
|
import { headers } from 'next/headers';
|
|
import { Inter, JetBrains_Mono } from 'next/font/google';
|
|
import { Toaster } from 'sonner';
|
|
import { classifyFormFactor } from '@/lib/form-factor';
|
|
import { ReactGrabViewportSync } from '@/components/dev/react-grab-viewport-sync';
|
|
import { resolveAuthShellBranding } from '@/lib/email/auth-shell-branding';
|
|
import './globals.css';
|
|
|
|
const inter = Inter({
|
|
subsets: ['latin'],
|
|
variable: '--font-sans',
|
|
display: 'swap',
|
|
});
|
|
|
|
const jetbrainsMono = JetBrains_Mono({
|
|
subsets: ['latin'],
|
|
variable: '--font-mono',
|
|
display: 'swap',
|
|
});
|
|
|
|
export const viewport: Viewport = {
|
|
width: 'device-width',
|
|
initialScale: 1,
|
|
viewportFit: 'cover',
|
|
themeColor: '#1e2844',
|
|
};
|
|
|
|
/**
|
|
* Resolve the browser tab title from the first-port `branding_app_name`
|
|
* setting so a tenant's deploy sees their own brand in the title bar
|
|
* (and in `Cmd+T` browser history). Falls back to a generic label when
|
|
* the DB hasn't been seeded yet (e.g. fresh `pnpm dev` against an empty
|
|
* database during onboarding).
|
|
*/
|
|
export async function generateMetadata(): Promise<Metadata> {
|
|
const branding = await resolveAuthShellBranding();
|
|
const appName = branding?.appName?.trim() || 'CRM';
|
|
return {
|
|
title: {
|
|
default: appName,
|
|
template: `%s | ${appName}`,
|
|
},
|
|
description: `${appName} - marina management system`,
|
|
appleWebApp: {
|
|
capable: true,
|
|
statusBarStyle: 'black-translucent',
|
|
title: appName,
|
|
},
|
|
icons: {
|
|
icon: [
|
|
{ url: '/icon-192.png', sizes: '192x192', type: 'image/png' },
|
|
{ url: '/icon-512.png', sizes: '512x512', type: 'image/png' },
|
|
],
|
|
apple: '/apple-touch-icon.png',
|
|
},
|
|
manifest: '/manifest.json',
|
|
};
|
|
}
|
|
|
|
export default async function RootLayout({ children }: { children: React.ReactNode }) {
|
|
const headerList = await headers();
|
|
const formFactor = classifyFormFactor(headerList.get('user-agent'));
|
|
|
|
return (
|
|
<html lang="en" suppressHydrationWarning>
|
|
<head>
|
|
{process.env.NODE_ENV === 'development' && (
|
|
<Script
|
|
src="//unpkg.com/react-grab/dist/index.global.js"
|
|
crossOrigin="anonymous"
|
|
strategy="beforeInteractive"
|
|
/>
|
|
)}
|
|
</head>
|
|
<body
|
|
data-form-factor={formFactor}
|
|
className={`${inter.variable} ${jetbrainsMono.variable} font-sans antialiased`}
|
|
>
|
|
{children}
|
|
<Toaster richColors position="top-right" />
|
|
{process.env.NODE_ENV === 'development' && <ReactGrabViewportSync />}
|
|
</body>
|
|
</html>
|
|
);
|
|
}
|