cmux/web/app/[locale]/docs/api/page.tsx
Lawrence Chen cf75da8f8a
Internationalize website with next-intl for 19 languages (#1216)
* Add i18n framework with next-intl for 19 languages

Set up complete internationalization infrastructure:
- Install next-intl v4 with App Router support
- Create i18n config (routing, request, navigation)
- Add middleware for automatic locale detection from Accept-Language
- Restructure all routes under app/[locale]/
- Extract UI strings to messages/en.json
- Update all components to use useTranslations()
- Add language switcher dropdown in footer
- Support RTL for Arabic and Khmer
- Update sitemap with locale alternates
- Add generateStaticParams for all 19 locales

Languages: en, ja, zh-CN, zh-TW, ko, de, es, fr, it, da, pl, ru, bs, ar, no, pt-BR, th, tr, km

Locale detection: auto-detect from browser Accept-Language header,
with cookie persistence and locale prefix only for non-default (en).

* Add translations for de, fr, it, ja, zh-CN, zh-TW

* Add translations for ar, bs, da, es, km, no, pl, pt-BR, ru, th, tr

* Convert docs and legal pages to use useTranslations()

* Add i18n to keyboard shortcuts component

* Add i18n to wall-of-love, add missing blog posts to sitemap

* Add keyboard shortcuts and wallOfLove translations to all locales

* Update bun lockfile for next-intl dependency

* Fix t.rich() configPath: pass ReactNode not function for {var} interpolation

* Fix configPath: use rich text tag instead of plain interpolation for ReactNode

* Fix t.rich() interpolation: use rich text tags for all ReactNode placeholders

Changed {legacy}, {openShortcut}, {jumpShortcut} from plain variable
interpolation to <tag>content</tag> format so t.rich() gets proper
functions instead of values.

* Escape ICU curly braces in socketCallout rich text across all locales

* Fix i18n issues: Khmer RTL, zh-CN quality, locale-aware testimonials, hardcoded strings

- Fix Khmer (km) incorrectly marked as RTL (it's LTR, only Arabic is RTL)
- Fix zh-CN/zh-TW taglinePrefix to mention terminals and open source
- Add locale-aware testimonial translations: show original text, translate
  for non-matching locales, skip translation when locale matches original
- Translate hardcoded English table content in notifications page
- Add testimonial translations to all 19 locale files
- Remove unused setRequestLocale import and params from home page

* Address PR review comments: metadata localization, blog fixes, legal pages, accessibility

- Convert hardcoded metadata to generateMetadata with getTranslations on all docs, blog, community, and wall-of-love pages
- Fix blog canonical/OG URLs to be locale-aware
- Fix introducing-cmux .split(": ") by using separate label/desc translation keys
- Revert legal page titles to English (legal content stays English-only)
- Add focus-visible ring to language switcher for keyboard accessibility
- Preserve query string and hash when switching locale
- Convert site-footer to server component (remove unnecessary "use client")
- Remove .toLowerCase() on translated text in community page
- Add /docs/browser-automation and /wall-of-love to sitemap
- Fix keyboard-shortcuts jump link visibility with trimmed query
- Deduplicate blogSlugs by importing from blog-posts.ts
- Add typingCodingAgents/typingMultitasking translation keys to all locales
- Fix Spanish accent/tilde issues in es.json testimonials
- Fix nested <a> tag in homepage keyboard shortcuts feature
- Remove unused setRequestLocale import from homepage

* Convert remaining layout/index metadata to generateMetadata

- Root layout: locale-aware title, description, OG, and Twitter card metadata
- Docs layout: translated title template
- Blog layout: translated title template
- Blog index: locale-aware metadata

* Add translated metadata keys to all locales, fix docs redirect

- Add meta.title/description/ogDescription to all 18 non-English locales
- Add docs.layoutTitle, blog.layoutTitle/metaTitle/metaDescription to all locales
- Add blog post metadata (zenOfCmux, cmdShiftU, showHnLaunch, introducingCmux) to all locales
- Add community.metaTitle/metaDescription to all locales
- Fix docs index redirect to preserve locale prefix

* Add translated docs page metaTitle keys to all locales
2026-03-12 05:36:58 -07:00

485 lines
14 KiB
TypeScript

import { useTranslations } from "next-intl";
import { getTranslations } from "next-intl/server";
import { CodeBlock } from "../../components/code-block";
import { Callout } from "../../components/callout";
export async function generateMetadata({ params }: { params: Promise<{ locale: string }> }) {
const { locale } = await params;
const t = await getTranslations({ locale, namespace: "docs.api" });
return {
title: t("metaTitle"),
description: t("metaDescription"),
};
}
function Cmd({
name,
desc,
cli,
socket,
}: {
name: string;
desc: string;
cli: string;
socket: string;
}) {
return (
<div className="mb-6">
<h4>{name}</h4>
<p>{desc}</p>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-2">
<CodeBlock title="CLI" lang="bash">{cli}</CodeBlock>
<CodeBlock title="Socket" lang="json">{socket}</CodeBlock>
</div>
</div>
);
}
export default function ApiPage() {
const t = useTranslations("docs.api");
return (
<>
<h1>{t("title")}</h1>
<p>{t("intro")}</p>
<h2>{t("socket")}</h2>
<table>
<thead>
<tr>
<th>{t("buildHeader")}</th>
<th>{t("pathHeader")}</th>
</tr>
</thead>
<tbody>
<tr>
<td>{t("release")}</td>
<td>
<code>/tmp/cmux.sock</code>
</td>
</tr>
<tr>
<td>{t("debug")}</td>
<td>
<code>/tmp/cmux-debug.sock</code>
</td>
</tr>
<tr>
<td>{t("taggedDebug")}</td>
<td>
<code>/tmp/cmux-debug-&lt;tag&gt;.sock</code>
</td>
</tr>
</tbody>
</table>
<p>{t("socketOverride")}</p>
<CodeBlock lang="json">{`{"id":"req-1","method":"workspace.list","params":{}}
// Response:
{"id":"req-1","ok":true,"result":{"workspaces":[...]}}`}</CodeBlock>
<Callout>
{t.rich("socketCallout", {
legacy: (chunks) => <code>{chunks}</code>,
})}
</Callout>
<h2>{t("accessModes")}</h2>
<table>
<thead>
<tr>
<th>{t("modeHeader")}</th>
<th>{t("descriptionHeader")}</th>
<th>{t("howToEnableHeader")}</th>
</tr>
</thead>
<tbody>
<tr>
<td>
<strong>Off</strong>
</td>
<td>{t("offMode")}</td>
<td>{t("offEnable")}</td>
</tr>
<tr>
<td>
<strong>cmux processes only</strong>
</td>
<td>{t("cmuxOnlyMode")}</td>
<td>{t("cmuxOnlyEnable")}</td>
</tr>
<tr>
<td>
<strong>allowAll</strong>
</td>
<td>{t("allowAllMode")}</td>
<td>{t("allowAllEnable")}</td>
</tr>
</tbody>
</table>
<Callout type="warn">
{t("accessCallout")}
</Callout>
<h2>{t("cliOptions")}</h2>
<table>
<thead>
<tr>
<th>{t("flagHeader")}</th>
<th>{t("descriptionHeader")}</th>
</tr>
</thead>
<tbody>
<tr>
<td>
<code>--socket PATH</code>
</td>
<td>{t("customSocketPath")}</td>
</tr>
<tr>
<td>
<code>--json</code>
</td>
<td>{t("outputJson")}</td>
</tr>
<tr>
<td>
<code>--window ID</code>
</td>
<td>{t("targetWindow")}</td>
</tr>
<tr>
<td>
<code>--workspace ID</code>
</td>
<td>{t("targetWorkspace")}</td>
</tr>
<tr>
<td>
<code>--surface ID</code>
</td>
<td>{t("targetSurface")}</td>
</tr>
<tr>
<td>
<code>--id-format refs|uuids|both</code>
</td>
<td>{t("idFormat")}</td>
</tr>
</tbody>
</table>
<h2>{t("workspaceCommands")}</h2>
<Cmd
name="list-workspaces"
desc={t("listWorkspacesDesc")}
cli={`cmux list-workspaces
cmux list-workspaces --json`}
socket={`{"id":"ws-list","method":"workspace.list","params":{}}`}
/>
<Cmd
name="new-workspace"
desc={t("newWorkspaceDesc")}
cli={`cmux new-workspace`}
socket={`{"id":"ws-new","method":"workspace.create","params":{}}`}
/>
<Cmd
name="select-workspace"
desc={t("selectWorkspaceDesc")}
cli={`cmux select-workspace --workspace <id>`}
socket={`{"id":"ws-select","method":"workspace.select","params":{"workspace_id":"<id>"}}`}
/>
<Cmd
name="current-workspace"
desc={t("currentWorkspaceDesc")}
cli={`cmux current-workspace
cmux current-workspace --json`}
socket={`{"id":"ws-current","method":"workspace.current","params":{}}`}
/>
<Cmd
name="close-workspace"
desc={t("closeWorkspaceDesc")}
cli={`cmux close-workspace --workspace <id>`}
socket={`{"id":"ws-close","method":"workspace.close","params":{"workspace_id":"<id>"}}`}
/>
<h2>{t("splitCommands")}</h2>
<Cmd
name="new-split"
desc={t("newSplitDesc")}
cli={`cmux new-split right
cmux new-split down`}
socket={`{"id":"split-new","method":"surface.split","params":{"direction":"right"}}`}
/>
<Cmd
name="list-surfaces"
desc={t("listSurfacesDesc")}
cli={`cmux list-surfaces
cmux list-surfaces --json`}
socket={`{"id":"surface-list","method":"surface.list","params":{}}`}
/>
<Cmd
name="focus-surface"
desc={t("focusSurfaceDesc")}
cli={`cmux focus-surface --surface <id>`}
socket={`{"id":"surface-focus","method":"surface.focus","params":{"surface_id":"<id>"}}`}
/>
<h2>{t("inputCommands")}</h2>
<Cmd
name="send"
desc={t("sendDesc")}
cli={`cmux send "echo hello"
cmux send "ls -la\\n"`}
socket={`{"id":"send-text","method":"surface.send_text","params":{"text":"echo hello\\n"}}`}
/>
<Cmd
name="send-key"
desc={t("sendKeyDesc")}
cli={`cmux send-key enter`}
socket={`{"id":"send-key","method":"surface.send_key","params":{"key":"enter"}}`}
/>
<Cmd
name="send-surface"
desc={t("sendSurfaceDesc")}
cli={`cmux send-surface --surface <id> "command"`}
socket={`{"id":"send-surface","method":"surface.send_text","params":{"surface_id":"<id>","text":"command"}}`}
/>
<Cmd
name="send-key-surface"
desc={t("sendKeySurfaceDesc")}
cli={`cmux send-key-surface --surface <id> enter`}
socket={`{"id":"send-key-surface","method":"surface.send_key","params":{"surface_id":"<id>","key":"enter"}}`}
/>
<h2>{t("notificationCommands")}</h2>
<Cmd
name="notify"
desc={t("notifyDesc")}
cli={`cmux notify --title "Title" --body "Body"
cmux notify --title "T" --subtitle "S" --body "B"`}
socket={`{"id":"notify","method":"notification.create","params":{"title":"Title","subtitle":"S","body":"Body"}}`}
/>
<Cmd
name="list-notifications"
desc={t("listNotificationsDesc")}
cli={`cmux list-notifications
cmux list-notifications --json`}
socket={`{"id":"notif-list","method":"notification.list","params":{}}`}
/>
<Cmd
name="clear-notifications"
desc={t("clearNotificationsDesc")}
cli={`cmux clear-notifications`}
socket={`{"id":"notif-clear","method":"notification.clear","params":{}}`}
/>
<h2>{t("sidebarMetadata")}</h2>
<p>{t("sidebarMetadataDesc")}</p>
<Cmd
name="set-status"
desc={t("setStatusDesc")}
cli={`cmux set-status build "compiling" --icon hammer --color "#ff9500"
cmux set-status deploy "v1.2.3" --workspace workspace:2`}
socket={`set_status build compiling --icon=hammer --color=#ff9500 --tab=<workspace-uuid>`}
/>
<Cmd
name="clear-status"
desc={t("clearStatusDesc")}
cli={`cmux clear-status build`}
socket={`clear_status build --tab=<workspace-uuid>`}
/>
<Cmd
name="list-status"
desc={t("listStatusDesc")}
cli={`cmux list-status`}
socket={`list_status --tab=<workspace-uuid>`}
/>
<Cmd
name="set-progress"
desc={t("setProgressDesc")}
cli={`cmux set-progress 0.5 --label "Building..."
cmux set-progress 1.0 --label "Done"`}
socket={`set_progress 0.5 --label=Building... --tab=<workspace-uuid>`}
/>
<Cmd
name="clear-progress"
desc={t("clearProgressDesc")}
cli={`cmux clear-progress`}
socket={`clear_progress --tab=<workspace-uuid>`}
/>
<Cmd
name="log"
desc={t("logDesc")}
cli={`cmux log "Build started"
cmux log --level error --source build "Compilation failed"
cmux log --level success -- "All 42 tests passed"`}
socket={`log --level=error --source=build --tab=<workspace-uuid> -- Compilation failed`}
/>
<Cmd
name="clear-log"
desc={t("clearLogDesc")}
cli={`cmux clear-log`}
socket={`clear_log --tab=<workspace-uuid>`}
/>
<Cmd
name="list-log"
desc={t("listLogDesc")}
cli={`cmux list-log
cmux list-log --limit 5`}
socket={`list_log --limit=5 --tab=<workspace-uuid>`}
/>
<Cmd
name="sidebar-state"
desc={t("sidebarStateDesc")}
cli={`cmux sidebar-state
cmux sidebar-state --workspace workspace:2`}
socket={`sidebar_state --tab=<workspace-uuid>`}
/>
<h2>{t("utilityCommands")}</h2>
<Cmd
name="ping"
desc={t("pingDesc")}
cli={`cmux ping`}
socket={`{"id":"ping","method":"system.ping","params":{}}
// Response: {"id":"ping","ok":true,"result":{"pong":true}}`}
/>
<Cmd
name="capabilities"
desc={t("capabilitiesDesc")}
cli={`cmux capabilities
cmux capabilities --json`}
socket={`{"id":"caps","method":"system.capabilities","params":{}}`}
/>
<Cmd
name="identify"
desc={t("identifyDesc")}
cli={`cmux identify
cmux identify --json`}
socket={`{"id":"identify","method":"system.identify","params":{}}`}
/>
<h2>{t("envVariables")}</h2>
<table>
<thead>
<tr>
<th>{t("variableHeader")}</th>
<th>{t("descriptionHeader")}</th>
</tr>
</thead>
<tbody>
<tr>
<td>
<code>CMUX_SOCKET_PATH</code>
</td>
<td>{t("socketPathDesc")}</td>
</tr>
<tr>
<td>
<code>CMUX_SOCKET_ENABLE</code>
</td>
<td>{t("socketEnableDesc")}</td>
</tr>
<tr>
<td>
<code>CMUX_SOCKET_MODE</code>
</td>
<td>{t("socketModeDesc")}</td>
</tr>
<tr>
<td>
<code>CMUX_WORKSPACE_ID</code>
</td>
<td>{t("workspaceIdDesc")}</td>
</tr>
<tr>
<td>
<code>CMUX_SURFACE_ID</code>
</td>
<td>{t("surfaceIdDesc")}</td>
</tr>
<tr>
<td>
<code>TERM_PROGRAM</code>
</td>
<td>{t("termProgramDesc")}</td>
</tr>
<tr>
<td>
<code>TERM</code>
</td>
<td>{t("termDesc")}</td>
</tr>
</tbody>
</table>
<Callout>
{t("envCallout")}
</Callout>
<h2>{t("detectingCmux")}</h2>
<CodeBlock title="bash" lang="bash">{`# Prefer explicit socket path if set
SOCK="\${CMUX_SOCKET_PATH:-/tmp/cmux.sock}"
[ -S "$SOCK" ] && echo "Socket available"
# Check for the CLI
command -v cmux &>/dev/null && echo "cmux available"
# In cmux-managed terminals these are auto-set
[ -n "\${CMUX_WORKSPACE_ID:-}" ] && [ -n "\${CMUX_SURFACE_ID:-}" ] && echo "Inside cmux surface"
# Distinguish from regular Ghostty
[ "$TERM_PROGRAM" = "ghostty" ] && [ -n "\${CMUX_WORKSPACE_ID:-}" ] && echo "In cmux"`}</CodeBlock>
<h2>{t("examples")}</h2>
<h3>{t("pythonClient")}</h3>
<CodeBlock title="python" lang="python">{`import json
import os
import socket
SOCKET_PATH = os.environ.get("CMUX_SOCKET_PATH", "/tmp/cmux.sock")
def rpc(method, params=None, req_id=1):
payload = {"id": req_id, "method": method, "params": params or {}}
with socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) as sock:
sock.connect(SOCKET_PATH)
sock.sendall(json.dumps(payload).encode("utf-8") + b"\\n")
return json.loads(sock.recv(65536).decode("utf-8"))
# List workspaces
print(rpc("workspace.list", req_id="ws"))
# Send notification
print(rpc(
"notification.create",
{"title": "Hello", "body": "From Python!"},
req_id="notify"
))`}</CodeBlock>
<h3>{t("shellScript")}</h3>
<CodeBlock title="bash" lang="bash">{`#!/bin/bash
SOCK="\${CMUX_SOCKET_PATH:-/tmp/cmux.sock}"
cmux_cmd() {
printf "%s\\n" "$1" | nc -U "$SOCK"
}
cmux_cmd '{"id":"ws","method":"workspace.list","params":{}}'
cmux_cmd '{"id":"notify","method":"notification.create","params":{"title":"Done","body":"Task complete"}}'`}</CodeBlock>
<h3>{t("buildScriptNotification")}</h3>
<CodeBlock title="bash" lang="bash">{`#!/bin/bash
npm run build
if [ $? -eq 0 ]; then
cmux notify --title "✓ Build Success" --body "Ready to deploy"
else
cmux notify --title "✗ Build Failed" --body "Check the logs"
fi`}</CodeBlock>
</>
);
}