|
import { render, screen } from "@testing-library/react";
|
|
import { test, expect, describe, vi } from "vitest";
|
|
import { useTranslation } from "react-i18next";
|
|
import translations from "../../src/i18n/translation.json";
|
|
import { UserAvatar } from "../../src/components/features/sidebar/user-avatar";
|
|
|
|
vi.mock("@nextui-org/react", () => ({
|
|
Tooltip: ({ content, children }: { content: string; children: React.ReactNode }) => (
|
|
<div>
|
|
{children}
|
|
<div>{content}</div>
|
|
</div>
|
|
),
|
|
}));
|
|
|
|
const supportedLanguages = ['en', 'ja', 'zh-CN', 'zh-TW', 'ko-KR', 'de', 'no', 'it', 'pt', 'es', 'ar', 'fr', 'tr'];
|
|
|
|
|
|
function checkTranslationExists(key: string) {
|
|
const missingTranslations: string[] = [];
|
|
|
|
const translationEntry = (translations as Record<string, Record<string, string>>)[key];
|
|
if (!translationEntry) {
|
|
throw new Error(`Translation key "${key}" does not exist in translation.json`);
|
|
}
|
|
|
|
for (const lang of supportedLanguages) {
|
|
if (!translationEntry[lang]) {
|
|
missingTranslations.push(lang);
|
|
}
|
|
}
|
|
|
|
return missingTranslations;
|
|
}
|
|
|
|
|
|
function findDuplicateKeys(obj: Record<string, any>) {
|
|
const seen = new Set<string>();
|
|
const duplicates = new Set<string>();
|
|
|
|
|
|
for (const key in obj) {
|
|
if (seen.has(key)) {
|
|
duplicates.add(key);
|
|
} else {
|
|
seen.add(key);
|
|
}
|
|
}
|
|
|
|
return Array.from(duplicates);
|
|
}
|
|
|
|
vi.mock("react-i18next", () => ({
|
|
useTranslation: () => ({
|
|
t: (key: string) => {
|
|
const translationEntry = (translations as Record<string, Record<string, string>>)[key];
|
|
return translationEntry?.ja || key;
|
|
},
|
|
}),
|
|
}));
|
|
|
|
describe("Landing page translations", () => {
|
|
test("should render Japanese translations correctly", () => {
|
|
|
|
const TestComponent = () => {
|
|
const { t } = useTranslation();
|
|
return (
|
|
<div>
|
|
<UserAvatar onClick={() => {}} />
|
|
<div data-testid="main-content">
|
|
<h1>{t("LANDING$TITLE")}</h1>
|
|
<button>{t("VSCODE$OPEN")}</button>
|
|
<button>{t("SUGGESTIONS$INCREASE_TEST_COVERAGE")}</button>
|
|
<button>{t("SUGGESTIONS$AUTO_MERGE_PRS")}</button>
|
|
<button>{t("SUGGESTIONS$FIX_README")}</button>
|
|
<button>{t("SUGGESTIONS$CLEAN_DEPENDENCIES")}</button>
|
|
</div>
|
|
<div data-testid="tabs">
|
|
<span>{t("WORKSPACE$TERMINAL_TAB_LABEL")}</span>
|
|
<span>{t("WORKSPACE$BROWSER_TAB_LABEL")}</span>
|
|
<span>{t("WORKSPACE$JUPYTER_TAB_LABEL")}</span>
|
|
<span>{t("WORKSPACE$CODE_EDITOR_TAB_LABEL")}</span>
|
|
</div>
|
|
<div data-testid="workspace-label">{t("WORKSPACE$TITLE")}</div>
|
|
<button data-testid="new-project">{t("PROJECT$NEW_PROJECT")}</button>
|
|
<div data-testid="status">
|
|
<span>{t("TERMINAL$WAITING_FOR_CLIENT")}</span>
|
|
<span>{t("STATUS$CONNECTED")}</span>
|
|
<span>{t("STATUS$CONNECTED_TO_SERVER")}</span>
|
|
</div>
|
|
<div data-testid="time">
|
|
<span>{`5 ${t("TIME$MINUTES_AGO")}`}</span>
|
|
<span>{`2 ${t("TIME$HOURS_AGO")}`}</span>
|
|
<span>{`3 ${t("TIME$DAYS_AGO")}`}</span>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
render(<TestComponent />);
|
|
|
|
|
|
expect(screen.getByText("開発を始めましょう!")).toBeInTheDocument();
|
|
expect(screen.getByText("VS Codeで開く")).toBeInTheDocument();
|
|
expect(screen.getByText("テストカバレッジを向上させる")).toBeInTheDocument();
|
|
expect(screen.getByText("Dependabot PRを自動マージ")).toBeInTheDocument();
|
|
expect(screen.getByText("READMEを改善")).toBeInTheDocument();
|
|
expect(screen.getByText("依存関係を整理")).toBeInTheDocument();
|
|
|
|
|
|
const userAvatar = screen.getByTestId("user-avatar");
|
|
userAvatar.focus();
|
|
expect(screen.getByText("アカウント設定")).toBeInTheDocument();
|
|
|
|
|
|
const tabs = screen.getByTestId("tabs");
|
|
expect(tabs).toHaveTextContent("ターミナル");
|
|
expect(tabs).toHaveTextContent("ブラウザ");
|
|
expect(tabs).toHaveTextContent("Jupyter");
|
|
expect(tabs).toHaveTextContent("コードエディタ");
|
|
|
|
|
|
expect(screen.getByTestId("workspace-label")).toHaveTextContent("ワークスペース");
|
|
expect(screen.getByTestId("new-project")).toHaveTextContent("新規プロジェクト");
|
|
|
|
|
|
const status = screen.getByTestId("status");
|
|
expect(status).toHaveTextContent("クライアントの準備を待機中");
|
|
expect(status).toHaveTextContent("接続済み");
|
|
expect(status).toHaveTextContent("サーバーに接続済み");
|
|
|
|
|
|
expect(screen.getByText("アカウント設定")).toBeInTheDocument();
|
|
|
|
|
|
const time = screen.getByTestId("time");
|
|
expect(time).toHaveTextContent("5 分前");
|
|
expect(time).toHaveTextContent("2 時間前");
|
|
expect(time).toHaveTextContent("3 日前");
|
|
});
|
|
|
|
test("all translation keys should have translations for all supported languages", () => {
|
|
|
|
const translationKeys = [
|
|
"LANDING$TITLE",
|
|
"VSCODE$OPEN",
|
|
"SUGGESTIONS$INCREASE_TEST_COVERAGE",
|
|
"SUGGESTIONS$AUTO_MERGE_PRS",
|
|
"SUGGESTIONS$FIX_README",
|
|
"SUGGESTIONS$CLEAN_DEPENDENCIES",
|
|
"WORKSPACE$TERMINAL_TAB_LABEL",
|
|
"WORKSPACE$BROWSER_TAB_LABEL",
|
|
"WORKSPACE$JUPYTER_TAB_LABEL",
|
|
"WORKSPACE$CODE_EDITOR_TAB_LABEL",
|
|
"WORKSPACE$TITLE",
|
|
"PROJECT$NEW_PROJECT",
|
|
"TERMINAL$WAITING_FOR_CLIENT",
|
|
"STATUS$CONNECTED",
|
|
"STATUS$CONNECTED_TO_SERVER",
|
|
"TIME$MINUTES_AGO",
|
|
"TIME$HOURS_AGO",
|
|
"TIME$DAYS_AGO"
|
|
];
|
|
|
|
|
|
const missingTranslationsMap = new Map<string, string[]>();
|
|
translationKeys.forEach(key => {
|
|
const missing = checkTranslationExists(key);
|
|
if (missing.length > 0) {
|
|
missingTranslationsMap.set(key, missing);
|
|
}
|
|
});
|
|
|
|
|
|
if (missingTranslationsMap.size > 0) {
|
|
const errorMessage = Array.from(missingTranslationsMap.entries())
|
|
.map(([key, langs]) => `\n- "${key}" is missing translations for: ${langs.join(', ')}`)
|
|
.join('');
|
|
throw new Error(`Missing translations:${errorMessage}`);
|
|
}
|
|
});
|
|
|
|
test("translation file should not have duplicate keys", () => {
|
|
const duplicates = findDuplicateKeys(translations);
|
|
|
|
if (duplicates.length > 0) {
|
|
throw new Error(`Found duplicate translation keys: ${duplicates.join(', ')}`);
|
|
}
|
|
});
|
|
});
|
|
|