@arcadialdev/arcality 2.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.agents/skills/e2e-testing-expert/SKILL.md +28 -0
- package/.agents/skills/frontend-design/LICENSE.txt +177 -0
- package/.agents/skills/frontend-design/SKILL.md +42 -0
- package/.agents/skills/nodejs-backend-patterns/SKILL.md +639 -0
- package/.agents/skills/nodejs-backend-patterns/references/advanced-patterns.md +430 -0
- package/.agents/skills/playwright-best-practices/LICENSE.md +7 -0
- package/.agents/skills/playwright-best-practices/README.md +147 -0
- package/.agents/skills/playwright-best-practices/SKILL.md +303 -0
- package/.agents/skills/playwright-best-practices/advanced/authentication-flows.md +360 -0
- package/.agents/skills/playwright-best-practices/advanced/authentication.md +871 -0
- package/.agents/skills/playwright-best-practices/advanced/clock-mocking.md +364 -0
- package/.agents/skills/playwright-best-practices/advanced/mobile-testing.md +409 -0
- package/.agents/skills/playwright-best-practices/advanced/multi-context.md +288 -0
- package/.agents/skills/playwright-best-practices/advanced/multi-user.md +393 -0
- package/.agents/skills/playwright-best-practices/advanced/network-advanced.md +452 -0
- package/.agents/skills/playwright-best-practices/advanced/third-party.md +464 -0
- package/.agents/skills/playwright-best-practices/architecture/pom-vs-fixtures.md +363 -0
- package/.agents/skills/playwright-best-practices/architecture/test-architecture.md +369 -0
- package/.agents/skills/playwright-best-practices/architecture/when-to-mock.md +383 -0
- package/.agents/skills/playwright-best-practices/browser-apis/browser-apis.md +391 -0
- package/.agents/skills/playwright-best-practices/browser-apis/iframes.md +403 -0
- package/.agents/skills/playwright-best-practices/browser-apis/service-workers.md +504 -0
- package/.agents/skills/playwright-best-practices/browser-apis/websockets.md +403 -0
- package/.agents/skills/playwright-best-practices/core/annotations.md +424 -0
- package/.agents/skills/playwright-best-practices/core/assertions-waiting.md +361 -0
- package/.agents/skills/playwright-best-practices/core/configuration.md +452 -0
- package/.agents/skills/playwright-best-practices/core/fixtures-hooks.md +417 -0
- package/.agents/skills/playwright-best-practices/core/global-setup.md +434 -0
- package/.agents/skills/playwright-best-practices/core/locators.md +242 -0
- package/.agents/skills/playwright-best-practices/core/page-object-model.md +315 -0
- package/.agents/skills/playwright-best-practices/core/projects-dependencies.md +453 -0
- package/.agents/skills/playwright-best-practices/core/test-data.md +492 -0
- package/.agents/skills/playwright-best-practices/core/test-suite-structure.md +361 -0
- package/.agents/skills/playwright-best-practices/core/test-tags.md +298 -0
- package/.agents/skills/playwright-best-practices/debugging/console-errors.md +420 -0
- package/.agents/skills/playwright-best-practices/debugging/debugging.md +504 -0
- package/.agents/skills/playwright-best-practices/debugging/error-testing.md +360 -0
- package/.agents/skills/playwright-best-practices/debugging/flaky-tests.md +496 -0
- package/.agents/skills/playwright-best-practices/frameworks/angular.md +530 -0
- package/.agents/skills/playwright-best-practices/frameworks/nextjs.md +469 -0
- package/.agents/skills/playwright-best-practices/frameworks/react.md +531 -0
- package/.agents/skills/playwright-best-practices/frameworks/vue.md +574 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/ci-cd.md +468 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/docker.md +283 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/github-actions.md +546 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/gitlab.md +397 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/other-providers.md +521 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/parallel-sharding.md +371 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/performance.md +453 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/reporting.md +424 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/test-coverage.md +497 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/accessibility.md +359 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/api-testing.md +719 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/browser-extensions.md +506 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/canvas-webgl.md +493 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/component-testing.md +500 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/drag-drop.md +576 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/electron.md +509 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/file-operations.md +377 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/file-upload-download.md +562 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/forms-validation.md +561 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/graphql-testing.md +331 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/i18n.md +508 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/performance-testing.md +476 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/security-testing.md +430 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/visual-regression.md +634 -0
- package/.env.example +21 -0
- package/README.md +30 -0
- package/bin/arcality.mjs +86 -0
- package/package.json +66 -0
- package/playwright.config.ts +12 -0
- package/scripts/cleanup-qmsdev.mjs +63 -0
- package/scripts/discover-view.mjs +52 -0
- package/scripts/extract-view.mjs +64 -0
- package/scripts/gen-and-run.mjs +838 -0
- package/scripts/init.mjs +290 -0
- package/scripts/migrate-to-central-out.mjs +157 -0
- package/scripts/postinstall.mjs +63 -0
- package/scripts/rebrand-report.mjs +241 -0
- package/scripts/setup.mjs +166 -0
- package/src/KnowledgeService.ts +239 -0
- package/src/arcalityClient.mjs +266 -0
- package/src/configLoader.mjs +179 -0
- package/src/configManager.mjs +172 -0
- package/src/consoleBanner.ts +32 -0
- package/src/envSetup.ts +205 -0
- package/src/index.ts +25 -0
- package/src/projectInspector.ts +42 -0
- package/src/services/collectiveMemoryService.ts +178 -0
- package/src/testRunner.ts +201 -0
- package/tests/_helpers/ArcalityReporter.ts +490 -0
- package/tests/_helpers/agentic-runner.spec.ts +741 -0
- package/tests/_helpers/ai-agent-helper.ts +1573 -0
- package/tests/_helpers/discover-view.spec.ts +238 -0
- package/tests/_helpers/extract-view.spec.ts +118 -0
- package/tests/_helpers/qa-tools.ts +333 -0
- package/tests/_helpers/smart-action.spec.ts +1458 -0
|
@@ -0,0 +1,508 @@
|
|
|
1
|
+
# Internationalization (i18n) Testing
|
|
2
|
+
|
|
3
|
+
## Table of Contents
|
|
4
|
+
|
|
5
|
+
1. [Locale Configuration](#locale-configuration)
|
|
6
|
+
2. [Testing Multiple Locales](#testing-multiple-locales)
|
|
7
|
+
3. [RTL Layout Testing](#rtl-layout-testing)
|
|
8
|
+
4. [Date, Time & Number Formats](#date-time--number-formats)
|
|
9
|
+
5. [Translation Verification](#translation-verification)
|
|
10
|
+
6. [Visual Regression for i18n](#visual-regression-for-i18n)
|
|
11
|
+
|
|
12
|
+
## Locale Configuration
|
|
13
|
+
|
|
14
|
+
### Setting Browser Locale
|
|
15
|
+
|
|
16
|
+
```typescript
|
|
17
|
+
// playwright.config.ts
|
|
18
|
+
import { defineConfig, devices } from "@playwright/test";
|
|
19
|
+
|
|
20
|
+
export default defineConfig({
|
|
21
|
+
projects: [
|
|
22
|
+
{
|
|
23
|
+
name: "english",
|
|
24
|
+
use: {
|
|
25
|
+
...devices["Desktop Chrome"],
|
|
26
|
+
locale: "en-US",
|
|
27
|
+
timezoneId: "America/New_York",
|
|
28
|
+
},
|
|
29
|
+
},
|
|
30
|
+
{
|
|
31
|
+
name: "german",
|
|
32
|
+
use: {
|
|
33
|
+
...devices["Desktop Chrome"],
|
|
34
|
+
locale: "de-DE",
|
|
35
|
+
timezoneId: "Europe/Berlin",
|
|
36
|
+
},
|
|
37
|
+
},
|
|
38
|
+
{
|
|
39
|
+
name: "japanese",
|
|
40
|
+
use: {
|
|
41
|
+
...devices["Desktop Chrome"],
|
|
42
|
+
locale: "ja-JP",
|
|
43
|
+
timezoneId: "Asia/Tokyo",
|
|
44
|
+
},
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
name: "arabic",
|
|
48
|
+
use: {
|
|
49
|
+
...devices["Desktop Chrome"],
|
|
50
|
+
locale: "ar-SA",
|
|
51
|
+
timezoneId: "Asia/Riyadh",
|
|
52
|
+
},
|
|
53
|
+
},
|
|
54
|
+
],
|
|
55
|
+
});
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
### Per-Test Locale Override
|
|
59
|
+
|
|
60
|
+
```typescript
|
|
61
|
+
test("test in French locale", async ({ browser }) => {
|
|
62
|
+
const context = await browser.newContext({
|
|
63
|
+
locale: "fr-FR",
|
|
64
|
+
timezoneId: "Europe/Paris",
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
const page = await context.newPage();
|
|
68
|
+
await page.goto("/");
|
|
69
|
+
|
|
70
|
+
// Verify French content
|
|
71
|
+
await expect(page.getByRole("button", { name: "Connexion" })).toBeVisible();
|
|
72
|
+
|
|
73
|
+
await context.close();
|
|
74
|
+
});
|
|
75
|
+
```
|
|
76
|
+
|
|
77
|
+
### Accept-Language Header
|
|
78
|
+
|
|
79
|
+
```typescript
|
|
80
|
+
test("server-side locale detection", async ({ browser }) => {
|
|
81
|
+
const context = await browser.newContext({
|
|
82
|
+
locale: "es-ES",
|
|
83
|
+
extraHTTPHeaders: {
|
|
84
|
+
"Accept-Language": "es-ES,es;q=0.9,en;q=0.8",
|
|
85
|
+
},
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
const page = await context.newPage();
|
|
89
|
+
await page.goto("/");
|
|
90
|
+
|
|
91
|
+
// Server should respond with Spanish content
|
|
92
|
+
await expect(page.locator("html")).toHaveAttribute("lang", "es");
|
|
93
|
+
});
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
## Testing Multiple Locales
|
|
97
|
+
|
|
98
|
+
### Parameterized Locale Tests
|
|
99
|
+
|
|
100
|
+
```typescript
|
|
101
|
+
const locales = [
|
|
102
|
+
{ locale: "en-US", greeting: "Hello", button: "Sign In" },
|
|
103
|
+
{ locale: "de-DE", greeting: "Hallo", button: "Anmelden" },
|
|
104
|
+
{ locale: "fr-FR", greeting: "Bonjour", button: "Se connecter" },
|
|
105
|
+
{ locale: "ja-JP", greeting: "こんにちは", button: "ログイン" },
|
|
106
|
+
];
|
|
107
|
+
|
|
108
|
+
for (const { locale, greeting, button } of locales) {
|
|
109
|
+
test(`login page in ${locale}`, async ({ browser }) => {
|
|
110
|
+
const context = await browser.newContext({ locale });
|
|
111
|
+
const page = await context.newPage();
|
|
112
|
+
|
|
113
|
+
await page.goto("/login");
|
|
114
|
+
|
|
115
|
+
await expect(page.getByText(greeting)).toBeVisible();
|
|
116
|
+
await expect(page.getByRole("button", { name: button })).toBeVisible();
|
|
117
|
+
|
|
118
|
+
await context.close();
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
```
|
|
122
|
+
|
|
123
|
+
### Locale Fixture
|
|
124
|
+
|
|
125
|
+
```typescript
|
|
126
|
+
// fixtures/i18n.ts
|
|
127
|
+
import { test as base } from "@playwright/test";
|
|
128
|
+
|
|
129
|
+
type LocaleFixtures = {
|
|
130
|
+
localePage: (locale: string) => Promise<Page>;
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
export const test = base.extend<LocaleFixtures>({
|
|
134
|
+
localePage: async ({ browser }, use) => {
|
|
135
|
+
const pages: Page[] = [];
|
|
136
|
+
|
|
137
|
+
const createLocalePage = async (locale: string) => {
|
|
138
|
+
const context = await browser.newContext({ locale });
|
|
139
|
+
const page = await context.newPage();
|
|
140
|
+
pages.push(page);
|
|
141
|
+
return page;
|
|
142
|
+
};
|
|
143
|
+
|
|
144
|
+
await use(createLocalePage);
|
|
145
|
+
|
|
146
|
+
// Cleanup
|
|
147
|
+
for (const page of pages) {
|
|
148
|
+
await page.context().close();
|
|
149
|
+
}
|
|
150
|
+
},
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
// Usage
|
|
154
|
+
test("compare locales", async ({ localePage }) => {
|
|
155
|
+
const enPage = await localePage("en-US");
|
|
156
|
+
const dePage = await localePage("de-DE");
|
|
157
|
+
|
|
158
|
+
await enPage.goto("/pricing");
|
|
159
|
+
await dePage.goto("/pricing");
|
|
160
|
+
|
|
161
|
+
const enPrice = await enPage.getByTestId("price").textContent();
|
|
162
|
+
const dePrice = await dePage.getByTestId("price").textContent();
|
|
163
|
+
|
|
164
|
+
expect(enPrice).toContain("$");
|
|
165
|
+
expect(dePrice).toContain("€");
|
|
166
|
+
});
|
|
167
|
+
```
|
|
168
|
+
|
|
169
|
+
### Testing Locale Switching
|
|
170
|
+
|
|
171
|
+
```typescript
|
|
172
|
+
test("user can switch locale", async ({ page }) => {
|
|
173
|
+
await page.goto("/");
|
|
174
|
+
|
|
175
|
+
// Initial locale (from browser)
|
|
176
|
+
await expect(page.locator("html")).toHaveAttribute("lang", "en");
|
|
177
|
+
|
|
178
|
+
// Switch to German
|
|
179
|
+
await page.getByRole("button", { name: "Language" }).click();
|
|
180
|
+
await page.getByRole("menuitem", { name: "Deutsch" }).click();
|
|
181
|
+
|
|
182
|
+
// Verify switch
|
|
183
|
+
await expect(page.locator("html")).toHaveAttribute("lang", "de");
|
|
184
|
+
await expect(page.getByRole("heading", { level: 1 })).toContainText(
|
|
185
|
+
/Willkommen/,
|
|
186
|
+
);
|
|
187
|
+
|
|
188
|
+
// Verify persistence (reload)
|
|
189
|
+
await page.reload();
|
|
190
|
+
await expect(page.locator("html")).toHaveAttribute("lang", "de");
|
|
191
|
+
});
|
|
192
|
+
```
|
|
193
|
+
|
|
194
|
+
## RTL Layout Testing
|
|
195
|
+
|
|
196
|
+
### Setting Up RTL Tests
|
|
197
|
+
|
|
198
|
+
```typescript
|
|
199
|
+
// playwright.config.ts
|
|
200
|
+
export default defineConfig({
|
|
201
|
+
projects: [
|
|
202
|
+
{
|
|
203
|
+
name: "rtl-arabic",
|
|
204
|
+
use: {
|
|
205
|
+
locale: "ar-SA",
|
|
206
|
+
// RTL is usually set by the app based on locale
|
|
207
|
+
},
|
|
208
|
+
},
|
|
209
|
+
{
|
|
210
|
+
name: "rtl-hebrew",
|
|
211
|
+
use: {
|
|
212
|
+
locale: "he-IL",
|
|
213
|
+
},
|
|
214
|
+
},
|
|
215
|
+
],
|
|
216
|
+
});
|
|
217
|
+
```
|
|
218
|
+
|
|
219
|
+
### Verifying RTL Direction
|
|
220
|
+
|
|
221
|
+
```typescript
|
|
222
|
+
test("RTL layout is applied", async ({ page }) => {
|
|
223
|
+
await page.goto("/");
|
|
224
|
+
|
|
225
|
+
// Check document direction
|
|
226
|
+
await expect(page.locator("html")).toHaveAttribute("dir", "rtl");
|
|
227
|
+
|
|
228
|
+
// Or check computed style
|
|
229
|
+
const direction = await page.evaluate(() => {
|
|
230
|
+
return window.getComputedStyle(document.body).direction;
|
|
231
|
+
});
|
|
232
|
+
expect(direction).toBe("rtl");
|
|
233
|
+
});
|
|
234
|
+
```
|
|
235
|
+
|
|
236
|
+
### RTL-Specific Element Positioning
|
|
237
|
+
|
|
238
|
+
```typescript
|
|
239
|
+
test("sidebar is on the right in RTL", async ({ page }) => {
|
|
240
|
+
await page.goto("/dashboard");
|
|
241
|
+
|
|
242
|
+
const sidebar = page.getByTestId("sidebar");
|
|
243
|
+
const main = page.getByTestId("main-content");
|
|
244
|
+
|
|
245
|
+
const sidebarBox = await sidebar.boundingBox();
|
|
246
|
+
const mainBox = await main.boundingBox();
|
|
247
|
+
|
|
248
|
+
// In RTL, sidebar should be to the right of main content
|
|
249
|
+
expect(sidebarBox!.x).toBeGreaterThan(mainBox!.x);
|
|
250
|
+
});
|
|
251
|
+
```
|
|
252
|
+
|
|
253
|
+
### RTL Visual Regression
|
|
254
|
+
|
|
255
|
+
```typescript
|
|
256
|
+
test("RTL layout matches snapshot", async ({ page }) => {
|
|
257
|
+
await page.goto("/");
|
|
258
|
+
|
|
259
|
+
// Screenshot for RTL comparison
|
|
260
|
+
await expect(page).toHaveScreenshot("homepage-rtl.png", {
|
|
261
|
+
// Separate snapshots per locale/direction
|
|
262
|
+
fullPage: true,
|
|
263
|
+
});
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
// LTR comparison
|
|
267
|
+
test("LTR layout matches snapshot", async ({ browser }) => {
|
|
268
|
+
const context = await browser.newContext({ locale: "en-US" });
|
|
269
|
+
const page = await context.newPage();
|
|
270
|
+
|
|
271
|
+
await page.goto("/");
|
|
272
|
+
await expect(page).toHaveScreenshot("homepage-ltr.png", { fullPage: true });
|
|
273
|
+
});
|
|
274
|
+
```
|
|
275
|
+
|
|
276
|
+
### Testing Bidirectional Text
|
|
277
|
+
|
|
278
|
+
```typescript
|
|
279
|
+
test("bidirectional text renders correctly", async ({ page }) => {
|
|
280
|
+
await page.goto("/profile");
|
|
281
|
+
|
|
282
|
+
// Mixed LTR/RTL content
|
|
283
|
+
const nameField = page.getByTestId("full-name");
|
|
284
|
+
|
|
285
|
+
// Arabic name with English email
|
|
286
|
+
await expect(nameField).toContainText("محمد (mohammed@example.com)");
|
|
287
|
+
|
|
288
|
+
// Verify text doesn't overlap or break
|
|
289
|
+
const box = await nameField.boundingBox();
|
|
290
|
+
expect(box!.width).toBeGreaterThan(100); // Content not collapsed
|
|
291
|
+
});
|
|
292
|
+
```
|
|
293
|
+
|
|
294
|
+
## Date, Time & Number Formats
|
|
295
|
+
|
|
296
|
+
### Testing Date Formats
|
|
297
|
+
|
|
298
|
+
```typescript
|
|
299
|
+
test("dates are formatted per locale", async ({ browser }) => {
|
|
300
|
+
const testDate = new Date("2024-03-15");
|
|
301
|
+
|
|
302
|
+
const formats = [
|
|
303
|
+
{ locale: "en-US", expected: "March 15, 2024" },
|
|
304
|
+
{ locale: "en-GB", expected: "15 March 2024" },
|
|
305
|
+
{ locale: "de-DE", expected: "15. März 2024" },
|
|
306
|
+
{ locale: "ja-JP", expected: "2024年3月15日" },
|
|
307
|
+
];
|
|
308
|
+
|
|
309
|
+
for (const { locale, expected } of formats) {
|
|
310
|
+
const context = await browser.newContext({ locale });
|
|
311
|
+
const page = await context.newPage();
|
|
312
|
+
|
|
313
|
+
await page.goto(`/event?date=${testDate.toISOString()}`);
|
|
314
|
+
|
|
315
|
+
const dateDisplay = page.getByTestId("event-date");
|
|
316
|
+
await expect(dateDisplay).toContainText(expected);
|
|
317
|
+
|
|
318
|
+
await context.close();
|
|
319
|
+
}
|
|
320
|
+
});
|
|
321
|
+
```
|
|
322
|
+
|
|
323
|
+
### Testing Number Formats
|
|
324
|
+
|
|
325
|
+
```typescript
|
|
326
|
+
test("numbers are formatted per locale", async ({ browser }) => {
|
|
327
|
+
const testNumber = 1234567.89;
|
|
328
|
+
|
|
329
|
+
const formats = [
|
|
330
|
+
{ locale: "en-US", expected: "1,234,567.89" },
|
|
331
|
+
{ locale: "de-DE", expected: "1.234.567,89" },
|
|
332
|
+
{ locale: "fr-FR", expected: "1 234 567,89" },
|
|
333
|
+
];
|
|
334
|
+
|
|
335
|
+
for (const { locale, expected } of formats) {
|
|
336
|
+
const context = await browser.newContext({ locale });
|
|
337
|
+
const page = await context.newPage();
|
|
338
|
+
|
|
339
|
+
await page.goto(`/stats?value=${testNumber}`);
|
|
340
|
+
|
|
341
|
+
await expect(page.getByTestId("formatted-number")).toHaveText(expected);
|
|
342
|
+
|
|
343
|
+
await context.close();
|
|
344
|
+
}
|
|
345
|
+
});
|
|
346
|
+
```
|
|
347
|
+
|
|
348
|
+
### Testing Currency Formats
|
|
349
|
+
|
|
350
|
+
```typescript
|
|
351
|
+
test("currency displays correctly", async ({ browser }) => {
|
|
352
|
+
const price = 99.99;
|
|
353
|
+
|
|
354
|
+
const currencies = [
|
|
355
|
+
{ locale: "en-US", currency: "USD", expected: "$99.99" },
|
|
356
|
+
{ locale: "de-DE", currency: "EUR", expected: "99,99 €" },
|
|
357
|
+
{ locale: "ja-JP", currency: "JPY", expected: "¥100" }, // JPY has no decimals
|
|
358
|
+
{ locale: "en-GB", currency: "GBP", expected: "£99.99" },
|
|
359
|
+
];
|
|
360
|
+
|
|
361
|
+
for (const { locale, currency, expected } of currencies) {
|
|
362
|
+
const context = await browser.newContext({ locale });
|
|
363
|
+
const page = await context.newPage();
|
|
364
|
+
|
|
365
|
+
await page.goto(`/product?price=${price}¤cy=${currency}`);
|
|
366
|
+
|
|
367
|
+
await expect(page.getByTestId("price")).toContainText(expected);
|
|
368
|
+
|
|
369
|
+
await context.close();
|
|
370
|
+
}
|
|
371
|
+
});
|
|
372
|
+
```
|
|
373
|
+
|
|
374
|
+
## Translation Verification
|
|
375
|
+
|
|
376
|
+
### Checking for Missing Translations
|
|
377
|
+
|
|
378
|
+
```typescript
|
|
379
|
+
test("no missing translations", async ({ page }) => {
|
|
380
|
+
await page.goto("/");
|
|
381
|
+
|
|
382
|
+
// Common patterns for missing translations
|
|
383
|
+
const missingPatterns = [
|
|
384
|
+
/\{\{.*\}\}/, // Handlebars-style
|
|
385
|
+
/\$\{.*\}/, // Template literal style
|
|
386
|
+
/t\(["'][\w.]+["']\)/, // i18n key exposed
|
|
387
|
+
/MISSING_TRANSLATION/, // Common placeholder
|
|
388
|
+
/\[UNTRANSLATED\]/, // Another placeholder
|
|
389
|
+
];
|
|
390
|
+
|
|
391
|
+
const bodyText = await page.locator("body").textContent();
|
|
392
|
+
|
|
393
|
+
for (const pattern of missingPatterns) {
|
|
394
|
+
expect(bodyText).not.toMatch(pattern);
|
|
395
|
+
}
|
|
396
|
+
});
|
|
397
|
+
```
|
|
398
|
+
|
|
399
|
+
### Detecting Text Overflow
|
|
400
|
+
|
|
401
|
+
```typescript
|
|
402
|
+
test("translations fit UI containers", async ({ browser }) => {
|
|
403
|
+
const locales = ["en-US", "de-DE", "fr-FR", "es-ES"];
|
|
404
|
+
const issues: string[] = [];
|
|
405
|
+
|
|
406
|
+
for (const locale of locales) {
|
|
407
|
+
const context = await browser.newContext({ locale });
|
|
408
|
+
const page = await context.newPage();
|
|
409
|
+
await page.goto("/");
|
|
410
|
+
|
|
411
|
+
const overflowing = await page.evaluate(() => {
|
|
412
|
+
const elements = document.querySelectorAll("button, .label, h1, h2, h3");
|
|
413
|
+
return Array.from(elements)
|
|
414
|
+
.filter(
|
|
415
|
+
(el) =>
|
|
416
|
+
(el as HTMLElement).scrollWidth > (el as HTMLElement).clientWidth,
|
|
417
|
+
)
|
|
418
|
+
.map((el) => `${el.tagName}: "${el.textContent?.substring(0, 20)}..."`);
|
|
419
|
+
});
|
|
420
|
+
|
|
421
|
+
if (overflowing.length > 0)
|
|
422
|
+
issues.push(`${locale}: ${overflowing.join(", ")}`);
|
|
423
|
+
await context.close();
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
expect(issues).toEqual([]);
|
|
427
|
+
});
|
|
428
|
+
```
|
|
429
|
+
|
|
430
|
+
## Visual Regression for i18n
|
|
431
|
+
|
|
432
|
+
### Locale-Specific Snapshots
|
|
433
|
+
|
|
434
|
+
```typescript
|
|
435
|
+
// playwright.config.ts
|
|
436
|
+
export default defineConfig({
|
|
437
|
+
snapshotPathTemplate:
|
|
438
|
+
"{testDir}/__snapshots__/{projectName}/{testFilePath}/{arg}{ext}",
|
|
439
|
+
|
|
440
|
+
projects: [
|
|
441
|
+
{ name: "en-US", use: { locale: "en-US" } },
|
|
442
|
+
{ name: "de-DE", use: { locale: "de-DE" } },
|
|
443
|
+
{ name: "ja-JP", use: { locale: "ja-JP" } },
|
|
444
|
+
{ name: "ar-SA", use: { locale: "ar-SA" } },
|
|
445
|
+
],
|
|
446
|
+
});
|
|
447
|
+
```
|
|
448
|
+
|
|
449
|
+
```typescript
|
|
450
|
+
// test file
|
|
451
|
+
test("homepage visual", async ({ page }) => {
|
|
452
|
+
await page.goto("/");
|
|
453
|
+
|
|
454
|
+
// Snapshot auto-saved to {projectName}/homepage.png
|
|
455
|
+
await expect(page).toHaveScreenshot("homepage.png");
|
|
456
|
+
});
|
|
457
|
+
```
|
|
458
|
+
|
|
459
|
+
### Critical Element Screenshots
|
|
460
|
+
|
|
461
|
+
```typescript
|
|
462
|
+
test("navigation in all locales", async ({ page }) => {
|
|
463
|
+
await page.goto("/");
|
|
464
|
+
|
|
465
|
+
// Just the nav - catches overflow, truncation
|
|
466
|
+
const nav = page.getByRole("navigation");
|
|
467
|
+
await expect(nav).toHaveScreenshot("navigation.png");
|
|
468
|
+
});
|
|
469
|
+
|
|
470
|
+
test("buttons dont truncate", async ({ page }) => {
|
|
471
|
+
await page.goto("/checkout");
|
|
472
|
+
|
|
473
|
+
const ctaButton = page.getByRole("button", {
|
|
474
|
+
name: /checkout|kaufen|acheter/i,
|
|
475
|
+
});
|
|
476
|
+
await expect(ctaButton).toHaveScreenshot("checkout-button.png");
|
|
477
|
+
});
|
|
478
|
+
```
|
|
479
|
+
|
|
480
|
+
### Font Loading for i18n
|
|
481
|
+
|
|
482
|
+
```typescript
|
|
483
|
+
test("wait for fonts before screenshot", async ({ page }) => {
|
|
484
|
+
await page.goto("/");
|
|
485
|
+
|
|
486
|
+
// Wait for fonts (important for CJK, Arabic)
|
|
487
|
+
await page.evaluate(() => document.fonts.ready);
|
|
488
|
+
await page.waitForFunction(() =>
|
|
489
|
+
document.fonts.check("16px 'Noto Sans Arabic'"),
|
|
490
|
+
);
|
|
491
|
+
|
|
492
|
+
await expect(page).toHaveScreenshot("with-fonts.png");
|
|
493
|
+
});
|
|
494
|
+
```
|
|
495
|
+
|
|
496
|
+
## Anti-Patterns to Avoid
|
|
497
|
+
|
|
498
|
+
| Anti-Pattern | Problem | Solution |
|
|
499
|
+
| ------------------------- | ------------------------------- | ------------------------------- |
|
|
500
|
+
| Hardcoded text assertions | Breaks in other locales | Use test IDs or parameterize |
|
|
501
|
+
| Single locale testing | Misses i18n bugs | Test multiple locales |
|
|
502
|
+
| Ignoring RTL | Layout broken for RTL users | Dedicated RTL project |
|
|
503
|
+
| No font wait | Screenshots with fallback fonts | Wait for `document.fonts.ready` |
|
|
504
|
+
|
|
505
|
+
## Related References
|
|
506
|
+
|
|
507
|
+
- **Clock Mocking**: See [clock-mocking.md](../advanced/clock-mocking.md) for timezone testing
|
|
508
|
+
- **Mobile Testing**: See [mobile-testing.md](../advanced/mobile-testing.md) for device-specific locales
|