9e45780ba0
First commit of the Next.js community site that powers deepseek-tui.com, deployed via Cloudflare Workers / OpenNext. This commit lands the scaffold and applies the visual + correctness pass requested by community feedback: - Palette: drop the cream/Anthropic-feel paper (#F4F1E8) for a DeepSeek-aligned cool white + soft gray (#FFFFFF / #F4F6FB), with indigo accents kept. Soften default hairlines so a pure-white background reads clean instead of harsh. - Mobile: add a hamburger menu (mobile-menu.tsx) so phones can reach Install / Docs / Activity / Roadmap / Contribute — previously the link list was hidden on phones with no replacement. Tighter hero, flexible button row, viewport-safe code blocks, columnar grids collapse cleanly under 768px, and the printed-almanac center rule is desktop-only now (it sliced through narrow viewports). - "How it works" diagram: replace the hand-rolled ASCII art (which misaligned under CJK monospace because Han characters take 2 columns vs Latin's 1, per dhh's note in WeChat) with a real mermaid diagram rendered client-side via dynamic import. Uses the mermaid.live standard syntax 庄表伟 recommended. - Issue #1104: the docs listed a `deepseek-cn` provider that the v0.8.16 binary doesn't accept (`ProviderArg` in crates/cli only has 9 variants; the 10th lives only in the legacy tui/config.rs). derive-facts.mjs now omits `deepseek-cn` until that variant is wired through the shared ProviderKind, and the install page's China-network recipe uses `base_url` / `DEEPSEEK_BASE_URL` (which actually works on v0.8.16) instead of the unsupported provider. Auto-deploys via .github/workflows/deploy-web.yml on push to main. Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
51 lines
1.4 KiB
TypeScript
51 lines
1.4 KiB
TypeScript
import { NextRequest, NextResponse } from "next/server";
|
|
import { locales, defaultLocale } from "@/lib/i18n/config";
|
|
|
|
const COOKIE = "NEXT_LOCALE";
|
|
|
|
function detectLocale(req: NextRequest): string {
|
|
// 1. Cookie
|
|
const cookie = req.cookies.get(COOKIE)?.value;
|
|
if (cookie && locales.includes(cookie as typeof locales[number])) return cookie;
|
|
|
|
// 2. Accept-Language header
|
|
const accept = req.headers.get("accept-language") ?? "";
|
|
if (/^zh/i.test(accept.split(",")[0])) return "zh";
|
|
|
|
return defaultLocale;
|
|
}
|
|
|
|
export function middleware(req: NextRequest) {
|
|
const { pathname } = req.nextUrl;
|
|
|
|
// Skip API routes, static files, _next
|
|
if (
|
|
pathname.startsWith("/api/") ||
|
|
pathname.startsWith("/_next/") ||
|
|
pathname.includes(".")
|
|
) {
|
|
return NextResponse.next();
|
|
}
|
|
|
|
// Check if locale is already in path
|
|
const seg = pathname.split("/")[1];
|
|
if (locales.includes(seg as typeof locales[number])) {
|
|
// Ensure cookie is set
|
|
const res = NextResponse.next();
|
|
res.cookies.set(COOKIE, seg, { path: "/", maxAge: 60 * 60 * 24 * 365 });
|
|
return res;
|
|
}
|
|
|
|
// Redirect bare paths to detected locale
|
|
const locale = detectLocale(req);
|
|
const url = req.nextUrl.clone();
|
|
url.pathname = `/${locale}${pathname}`;
|
|
const res = NextResponse.redirect(url);
|
|
res.cookies.set(COOKIE, locale, { path: "/", maxAge: 60 * 60 * 24 * 365 });
|
|
return res;
|
|
}
|
|
|
|
export const config = {
|
|
matcher: ["/((?!_next|api|favicon.ico|icon.svg|.*\\..*).*)"],
|
|
};
|