@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,409 @@
|
|
|
1
|
+
# Mobile & Responsive Testing
|
|
2
|
+
|
|
3
|
+
## Table of Contents
|
|
4
|
+
|
|
5
|
+
1. [Device Emulation](#device-emulation)
|
|
6
|
+
2. [Touch Gestures](#touch-gestures)
|
|
7
|
+
3. [Viewport Testing](#viewport-testing)
|
|
8
|
+
4. [Mobile-Specific UI](#mobile-specific-ui)
|
|
9
|
+
5. [Responsive Breakpoints](#responsive-breakpoints)
|
|
10
|
+
|
|
11
|
+
## Device Emulation
|
|
12
|
+
|
|
13
|
+
### Use Built-in Devices
|
|
14
|
+
|
|
15
|
+
```typescript
|
|
16
|
+
import { test, devices } from "@playwright/test";
|
|
17
|
+
|
|
18
|
+
// Configure in playwright.config.ts
|
|
19
|
+
export default defineConfig({
|
|
20
|
+
projects: [
|
|
21
|
+
{ name: "Desktop Chrome", use: { ...devices["Desktop Chrome"] } },
|
|
22
|
+
{ name: "Mobile Safari", use: { ...devices["iPhone 14"] } },
|
|
23
|
+
{ name: "Mobile Chrome", use: { ...devices["Pixel 7"] } },
|
|
24
|
+
{ name: "Tablet", use: { ...devices["iPad Pro 11"] } },
|
|
25
|
+
],
|
|
26
|
+
});
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
### Custom Device Configuration
|
|
30
|
+
|
|
31
|
+
```typescript
|
|
32
|
+
test.use({
|
|
33
|
+
viewport: { width: 390, height: 844 },
|
|
34
|
+
deviceScaleFactor: 3,
|
|
35
|
+
isMobile: true,
|
|
36
|
+
hasTouch: true,
|
|
37
|
+
userAgent:
|
|
38
|
+
"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15",
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
test("custom mobile device", async ({ page }) => {
|
|
42
|
+
await page.goto("/");
|
|
43
|
+
// Test runs with custom device settings
|
|
44
|
+
});
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
### Test Across Multiple Devices
|
|
48
|
+
|
|
49
|
+
```typescript
|
|
50
|
+
const mobileDevices = ["iPhone 14", "Pixel 7", "Galaxy S21"];
|
|
51
|
+
|
|
52
|
+
for (const deviceName of mobileDevices) {
|
|
53
|
+
test(`checkout on ${deviceName}`, async ({ browser }) => {
|
|
54
|
+
const device = devices[deviceName];
|
|
55
|
+
const context = await browser.newContext({ ...device });
|
|
56
|
+
const page = await context.newPage();
|
|
57
|
+
|
|
58
|
+
await page.goto("/checkout");
|
|
59
|
+
await expect(page.getByRole("button", { name: "Pay" })).toBeVisible();
|
|
60
|
+
|
|
61
|
+
await context.close();
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
## Touch Gestures
|
|
67
|
+
|
|
68
|
+
### Tap
|
|
69
|
+
|
|
70
|
+
```typescript
|
|
71
|
+
test.use({ hasTouch: true });
|
|
72
|
+
|
|
73
|
+
test("tap to interact", async ({ page }) => {
|
|
74
|
+
await page.goto("/gallery");
|
|
75
|
+
|
|
76
|
+
// Tap is like click but for touch devices
|
|
77
|
+
await page.getByRole("img", { name: "Photo 1" }).tap();
|
|
78
|
+
|
|
79
|
+
await expect(page.getByRole("dialog")).toBeVisible();
|
|
80
|
+
});
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
### Swipe
|
|
84
|
+
|
|
85
|
+
```typescript
|
|
86
|
+
test("swipe carousel", async ({ page }) => {
|
|
87
|
+
await page.goto("/carousel");
|
|
88
|
+
|
|
89
|
+
const carousel = page.getByTestId("carousel");
|
|
90
|
+
const box = await carousel.boundingBox();
|
|
91
|
+
|
|
92
|
+
if (box) {
|
|
93
|
+
// Swipe left
|
|
94
|
+
await page.touchscreen.tap(box.x + box.width - 50, box.y + box.height / 2);
|
|
95
|
+
await page.mouse.move(box.x + 50, box.y + box.height / 2);
|
|
96
|
+
|
|
97
|
+
// Or use drag
|
|
98
|
+
await carousel.dragTo(carousel, {
|
|
99
|
+
sourcePosition: { x: box.width - 50, y: box.height / 2 },
|
|
100
|
+
targetPosition: { x: 50, y: box.height / 2 },
|
|
101
|
+
});
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
await expect(page.getByText("Slide 2")).toBeVisible();
|
|
105
|
+
});
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
### Swipe Fixture
|
|
109
|
+
|
|
110
|
+
```typescript
|
|
111
|
+
// fixtures/touch.fixture.ts
|
|
112
|
+
import { test as base, Page } from "@playwright/test";
|
|
113
|
+
|
|
114
|
+
type TouchFixtures = {
|
|
115
|
+
swipe: (
|
|
116
|
+
element: Locator,
|
|
117
|
+
direction: "left" | "right" | "up" | "down",
|
|
118
|
+
) => Promise<void>;
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
export const test = base.extend<TouchFixtures>({
|
|
122
|
+
swipe: async ({ page }, use) => {
|
|
123
|
+
await use(async (element, direction) => {
|
|
124
|
+
const box = await element.boundingBox();
|
|
125
|
+
if (!box) throw new Error("Element not visible");
|
|
126
|
+
|
|
127
|
+
const centerX = box.x + box.width / 2;
|
|
128
|
+
const centerY = box.y + box.height / 2;
|
|
129
|
+
const distance = 100;
|
|
130
|
+
|
|
131
|
+
const moves = {
|
|
132
|
+
left: {
|
|
133
|
+
startX: centerX + distance,
|
|
134
|
+
endX: centerX - distance,
|
|
135
|
+
y: centerY,
|
|
136
|
+
},
|
|
137
|
+
right: {
|
|
138
|
+
startX: centerX - distance,
|
|
139
|
+
endX: centerX + distance,
|
|
140
|
+
y: centerY,
|
|
141
|
+
},
|
|
142
|
+
up: {
|
|
143
|
+
startX: centerX,
|
|
144
|
+
endX: centerX,
|
|
145
|
+
startY: centerY + distance,
|
|
146
|
+
endY: centerY - distance,
|
|
147
|
+
},
|
|
148
|
+
down: {
|
|
149
|
+
startX: centerX,
|
|
150
|
+
endX: centerX,
|
|
151
|
+
startY: centerY - distance,
|
|
152
|
+
endY: centerY + distance,
|
|
153
|
+
},
|
|
154
|
+
};
|
|
155
|
+
|
|
156
|
+
const move = moves[direction];
|
|
157
|
+
await page.touchscreen.tap(move.startX, move.startY ?? move.y);
|
|
158
|
+
await page.mouse.move(move.endX, move.endY ?? move.y, { steps: 10 });
|
|
159
|
+
await page.mouse.up();
|
|
160
|
+
});
|
|
161
|
+
},
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
// Usage
|
|
165
|
+
test("swipe to delete", async ({ page, swipe }) => {
|
|
166
|
+
await page.goto("/inbox");
|
|
167
|
+
|
|
168
|
+
const message = page.getByTestId("message-1");
|
|
169
|
+
await swipe(message, "left");
|
|
170
|
+
|
|
171
|
+
await expect(page.getByRole("button", { name: "Delete" })).toBeVisible();
|
|
172
|
+
});
|
|
173
|
+
```
|
|
174
|
+
|
|
175
|
+
### Long Press
|
|
176
|
+
|
|
177
|
+
```typescript
|
|
178
|
+
test("long press for context menu", async ({ page }) => {
|
|
179
|
+
await page.goto("/files");
|
|
180
|
+
|
|
181
|
+
const file = page.getByText("document.pdf");
|
|
182
|
+
const box = await file.boundingBox();
|
|
183
|
+
|
|
184
|
+
if (box) {
|
|
185
|
+
// Touch down
|
|
186
|
+
await page.touchscreen.tap(box.x + box.width / 2, box.y + box.height / 2);
|
|
187
|
+
|
|
188
|
+
// Hold for 500ms
|
|
189
|
+
await page.waitForTimeout(500);
|
|
190
|
+
|
|
191
|
+
// Context menu should appear
|
|
192
|
+
await expect(page.getByRole("menu")).toBeVisible();
|
|
193
|
+
}
|
|
194
|
+
});
|
|
195
|
+
```
|
|
196
|
+
|
|
197
|
+
### Pinch Zoom
|
|
198
|
+
|
|
199
|
+
```typescript
|
|
200
|
+
test("pinch to zoom image", async ({ page }) => {
|
|
201
|
+
await page.goto("/map");
|
|
202
|
+
|
|
203
|
+
// Pinch zoom requires two touch points
|
|
204
|
+
// Playwright doesn't have native pinch support, so we simulate via evaluate
|
|
205
|
+
await page.evaluate(() => {
|
|
206
|
+
const element = document.querySelector("#map");
|
|
207
|
+
if (element) {
|
|
208
|
+
// Simulate wheel event as fallback for zoom
|
|
209
|
+
element.dispatchEvent(
|
|
210
|
+
new WheelEvent("wheel", {
|
|
211
|
+
deltaY: -100, // Negative = zoom in
|
|
212
|
+
ctrlKey: true, // Ctrl+wheel = pinch on many apps
|
|
213
|
+
}),
|
|
214
|
+
);
|
|
215
|
+
}
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
// Or trigger the app's zoom function directly
|
|
219
|
+
await page.evaluate(() => {
|
|
220
|
+
(window as any).mapInstance?.setZoom(15);
|
|
221
|
+
});
|
|
222
|
+
});
|
|
223
|
+
```
|
|
224
|
+
|
|
225
|
+
## Viewport Testing
|
|
226
|
+
|
|
227
|
+
### Test Different Sizes
|
|
228
|
+
|
|
229
|
+
```typescript
|
|
230
|
+
const viewports = [
|
|
231
|
+
{ name: "mobile", width: 375, height: 667 },
|
|
232
|
+
{ name: "tablet", width: 768, height: 1024 },
|
|
233
|
+
{ name: "desktop", width: 1920, height: 1080 },
|
|
234
|
+
];
|
|
235
|
+
|
|
236
|
+
for (const { name, width, height } of viewports) {
|
|
237
|
+
test(`navigation on ${name}`, async ({ page }) => {
|
|
238
|
+
await page.setViewportSize({ width, height });
|
|
239
|
+
await page.goto("/");
|
|
240
|
+
|
|
241
|
+
if (width < 768) {
|
|
242
|
+
// Mobile: should have hamburger menu
|
|
243
|
+
await expect(page.getByRole("button", { name: "Menu" })).toBeVisible();
|
|
244
|
+
} else {
|
|
245
|
+
// Desktop: should have visible nav links
|
|
246
|
+
await expect(page.getByRole("link", { name: "Products" })).toBeVisible();
|
|
247
|
+
}
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
```
|
|
251
|
+
|
|
252
|
+
### Dynamic Viewport Changes
|
|
253
|
+
|
|
254
|
+
```typescript
|
|
255
|
+
test("responsive layout change", async ({ page }) => {
|
|
256
|
+
await page.setViewportSize({ width: 1200, height: 800 });
|
|
257
|
+
await page.goto("/dashboard");
|
|
258
|
+
|
|
259
|
+
// Desktop: sidebar visible
|
|
260
|
+
await expect(page.getByRole("complementary")).toBeVisible();
|
|
261
|
+
|
|
262
|
+
// Resize to mobile
|
|
263
|
+
await page.setViewportSize({ width: 375, height: 667 });
|
|
264
|
+
|
|
265
|
+
// Mobile: sidebar hidden, hamburger visible
|
|
266
|
+
await expect(page.getByRole("complementary")).toBeHidden();
|
|
267
|
+
await expect(page.getByRole("button", { name: "Menu" })).toBeVisible();
|
|
268
|
+
});
|
|
269
|
+
```
|
|
270
|
+
|
|
271
|
+
## Mobile-Specific UI
|
|
272
|
+
|
|
273
|
+
### Hamburger Menu
|
|
274
|
+
|
|
275
|
+
```typescript
|
|
276
|
+
test("mobile navigation", async ({ page }) => {
|
|
277
|
+
await page.setViewportSize({ width: 375, height: 667 });
|
|
278
|
+
await page.goto("/");
|
|
279
|
+
|
|
280
|
+
// Open hamburger menu
|
|
281
|
+
await page.getByRole("button", { name: "Menu" }).click();
|
|
282
|
+
|
|
283
|
+
// Navigation drawer should appear
|
|
284
|
+
const nav = page.getByRole("navigation");
|
|
285
|
+
await expect(nav).toBeVisible();
|
|
286
|
+
|
|
287
|
+
// Navigate via mobile menu
|
|
288
|
+
await nav.getByRole("link", { name: "Products" }).click();
|
|
289
|
+
|
|
290
|
+
await expect(page).toHaveURL("/products");
|
|
291
|
+
// Menu should close after navigation
|
|
292
|
+
await expect(nav).toBeHidden();
|
|
293
|
+
});
|
|
294
|
+
```
|
|
295
|
+
|
|
296
|
+
### Bottom Sheet
|
|
297
|
+
|
|
298
|
+
```typescript
|
|
299
|
+
test("bottom sheet interaction", async ({ page }) => {
|
|
300
|
+
await page.setViewportSize({ width: 375, height: 667 });
|
|
301
|
+
await page.goto("/product/123");
|
|
302
|
+
|
|
303
|
+
await page.getByRole("button", { name: "Add to Cart" }).click();
|
|
304
|
+
|
|
305
|
+
// Bottom sheet appears
|
|
306
|
+
const sheet = page.getByRole("dialog");
|
|
307
|
+
await expect(sheet).toBeVisible();
|
|
308
|
+
|
|
309
|
+
// Select options
|
|
310
|
+
await sheet.getByRole("combobox", { name: "Size" }).selectOption("Large");
|
|
311
|
+
await sheet.getByRole("button", { name: "Confirm" }).click();
|
|
312
|
+
|
|
313
|
+
await expect(page.getByText("Added to cart")).toBeVisible();
|
|
314
|
+
});
|
|
315
|
+
```
|
|
316
|
+
|
|
317
|
+
### Pull to Refresh
|
|
318
|
+
|
|
319
|
+
```typescript
|
|
320
|
+
test("pull to refresh", async ({ page }) => {
|
|
321
|
+
await page.goto("/feed");
|
|
322
|
+
|
|
323
|
+
const feed = page.getByTestId("feed");
|
|
324
|
+
const initialFirstItem = await feed.locator("> *").first().textContent();
|
|
325
|
+
|
|
326
|
+
// Simulate pull down
|
|
327
|
+
const box = await feed.boundingBox();
|
|
328
|
+
if (box) {
|
|
329
|
+
await page.touchscreen.tap(box.x + box.width / 2, box.y + 50);
|
|
330
|
+
await page.mouse.move(box.x + box.width / 2, box.y + 200, { steps: 20 });
|
|
331
|
+
await page.mouse.up();
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
// Wait for refresh
|
|
335
|
+
await expect(page.getByTestId("loading")).toBeVisible();
|
|
336
|
+
await expect(page.getByTestId("loading")).toBeHidden();
|
|
337
|
+
|
|
338
|
+
// Content should be updated (in a real app)
|
|
339
|
+
});
|
|
340
|
+
```
|
|
341
|
+
|
|
342
|
+
## Responsive Breakpoints
|
|
343
|
+
|
|
344
|
+
### Test All Breakpoints
|
|
345
|
+
|
|
346
|
+
```typescript
|
|
347
|
+
const breakpoints = {
|
|
348
|
+
xs: 320,
|
|
349
|
+
sm: 640,
|
|
350
|
+
md: 768,
|
|
351
|
+
lg: 1024,
|
|
352
|
+
xl: 1280,
|
|
353
|
+
"2xl": 1536,
|
|
354
|
+
};
|
|
355
|
+
|
|
356
|
+
test.describe("responsive header", () => {
|
|
357
|
+
for (const [name, width] of Object.entries(breakpoints)) {
|
|
358
|
+
test(`header at ${name} (${width}px)`, async ({ page }) => {
|
|
359
|
+
await page.setViewportSize({ width, height: 800 });
|
|
360
|
+
await page.goto("/");
|
|
361
|
+
|
|
362
|
+
if (width < 768) {
|
|
363
|
+
await expect(page.getByTestId("mobile-menu-button")).toBeVisible();
|
|
364
|
+
await expect(page.getByTestId("desktop-nav")).toBeHidden();
|
|
365
|
+
} else {
|
|
366
|
+
await expect(page.getByTestId("mobile-menu-button")).toBeHidden();
|
|
367
|
+
await expect(page.getByTestId("desktop-nav")).toBeVisible();
|
|
368
|
+
}
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
});
|
|
372
|
+
```
|
|
373
|
+
|
|
374
|
+
### Visual Regression at Breakpoints
|
|
375
|
+
|
|
376
|
+
```typescript
|
|
377
|
+
test.describe("visual regression", () => {
|
|
378
|
+
const sizes = [
|
|
379
|
+
{ width: 375, height: 667, name: "mobile" },
|
|
380
|
+
{ width: 768, height: 1024, name: "tablet" },
|
|
381
|
+
{ width: 1440, height: 900, name: "desktop" },
|
|
382
|
+
];
|
|
383
|
+
|
|
384
|
+
for (const { width, height, name } of sizes) {
|
|
385
|
+
test(`homepage at ${name}`, async ({ page }) => {
|
|
386
|
+
await page.setViewportSize({ width, height });
|
|
387
|
+
await page.goto("/");
|
|
388
|
+
|
|
389
|
+
await expect(page).toHaveScreenshot(`homepage-${name}.png`);
|
|
390
|
+
});
|
|
391
|
+
}
|
|
392
|
+
});
|
|
393
|
+
```
|
|
394
|
+
|
|
395
|
+
## Anti-Patterns to Avoid
|
|
396
|
+
|
|
397
|
+
| Anti-Pattern | Problem | Solution |
|
|
398
|
+
| --------------------------- | ------------------------- | -------------------------------- |
|
|
399
|
+
| Only testing one viewport | Misses responsive bugs | Test multiple breakpoints |
|
|
400
|
+
| Ignoring touch events | Features broken on mobile | Test tap, swipe, long press |
|
|
401
|
+
| Hardcoded viewport in tests | Can't test multiple sizes | Use `page.setViewportSize()` |
|
|
402
|
+
| Not testing orientation | Landscape bugs missed | Test both portrait and landscape |
|
|
403
|
+
|
|
404
|
+
## Related References
|
|
405
|
+
|
|
406
|
+
- **Visual Testing**: See [test-suite-structure.md](../core/test-suite-structure.md) for screenshot testing
|
|
407
|
+
- **Locators**: See [locators.md](../core/locators.md) for mobile-friendly selectors
|
|
408
|
+
- **Browser APIs**: See [browser-apis.md](../browser-apis/browser-apis.md) for permissions (camera, geolocation, notifications)
|
|
409
|
+
- **Canvas/Touch**: See [canvas-webgl.md](../testing-patterns/canvas-webgl.md) for touch gestures on canvas elements
|
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
# Multi-Tab, Window & Popup Testing
|
|
2
|
+
|
|
3
|
+
This file covers **single-user scenarios** with multiple browser tabs, windows, and popups. For **multi-user collaboration testing** (multiple users interacting simultaneously), see [multi-user.md](multi-user.md).
|
|
4
|
+
|
|
5
|
+
## Table of Contents
|
|
6
|
+
|
|
7
|
+
1. [Popup Handling](#popup-handling)
|
|
8
|
+
2. [New Tab Navigation](#new-tab-navigation)
|
|
9
|
+
3. [OAuth Flows](#oauth-flows)
|
|
10
|
+
4. [Multiple Windows](#multiple-windows)
|
|
11
|
+
5. [Tab Coordination](#tab-coordination)
|
|
12
|
+
|
|
13
|
+
## Popup Handling
|
|
14
|
+
|
|
15
|
+
### Basic Popup
|
|
16
|
+
|
|
17
|
+
```typescript
|
|
18
|
+
test("handle popup window", async ({ page }) => {
|
|
19
|
+
await page.goto("/");
|
|
20
|
+
|
|
21
|
+
// Start waiting for popup before triggering it
|
|
22
|
+
const popupPromise = page.waitForEvent("popup");
|
|
23
|
+
await page.getByRole("button", { name: "Open Support Chat" }).click();
|
|
24
|
+
const popup = await popupPromise;
|
|
25
|
+
|
|
26
|
+
// Wait for popup to load
|
|
27
|
+
await popup.waitForLoadState();
|
|
28
|
+
|
|
29
|
+
// Interact with popup
|
|
30
|
+
await popup.getByLabel("Message").fill("Need help");
|
|
31
|
+
await popup.getByRole("button", { name: "Send" }).click();
|
|
32
|
+
|
|
33
|
+
await expect(popup.getByText("Message sent")).toBeVisible();
|
|
34
|
+
|
|
35
|
+
// Close popup
|
|
36
|
+
await popup.close();
|
|
37
|
+
});
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
### Popup with Authentication
|
|
41
|
+
|
|
42
|
+
```typescript
|
|
43
|
+
test("popup login flow", async ({ page }) => {
|
|
44
|
+
await page.goto("/dashboard");
|
|
45
|
+
|
|
46
|
+
const popupPromise = page.waitForEvent("popup");
|
|
47
|
+
await page.getByRole("button", { name: "Connect Account" }).click();
|
|
48
|
+
const popup = await popupPromise;
|
|
49
|
+
|
|
50
|
+
await popup.waitForLoadState();
|
|
51
|
+
|
|
52
|
+
// Complete login in popup
|
|
53
|
+
await popup.getByLabel("Email").fill("user@example.com");
|
|
54
|
+
await popup.getByLabel("Password").fill("password123");
|
|
55
|
+
await popup.getByRole("button", { name: "Log In" }).click();
|
|
56
|
+
|
|
57
|
+
// Popup should close automatically after auth
|
|
58
|
+
await popup.waitForEvent("close");
|
|
59
|
+
|
|
60
|
+
// Main page should reflect connected state
|
|
61
|
+
await expect(page.getByText("Account connected")).toBeVisible();
|
|
62
|
+
});
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
### Handle Blocked Popups
|
|
66
|
+
|
|
67
|
+
```typescript
|
|
68
|
+
test("handle popup blocker", async ({ page }) => {
|
|
69
|
+
await page.goto("/share");
|
|
70
|
+
|
|
71
|
+
// Listen for console messages about blocked popup
|
|
72
|
+
page.on("console", (msg) => {
|
|
73
|
+
if (msg.text().includes("popup blocked")) {
|
|
74
|
+
console.log("Popup was blocked");
|
|
75
|
+
}
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
const popupPromise = page.waitForEvent("popup").catch(() => null);
|
|
79
|
+
await page.getByRole("button", { name: "Share to Twitter" }).click();
|
|
80
|
+
const popup = await popupPromise;
|
|
81
|
+
|
|
82
|
+
if (!popup) {
|
|
83
|
+
// Popup blocked - app should show fallback
|
|
84
|
+
await expect(page.getByText("Copy share link instead")).toBeVisible();
|
|
85
|
+
}
|
|
86
|
+
});
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
## New Tab Navigation
|
|
90
|
+
|
|
91
|
+
### Link Opens in New Tab
|
|
92
|
+
|
|
93
|
+
```typescript
|
|
94
|
+
test("external link opens in new tab", async ({ page, context }) => {
|
|
95
|
+
await page.goto("/resources");
|
|
96
|
+
|
|
97
|
+
// Wait for new page in context
|
|
98
|
+
const pagePromise = context.waitForEvent("page");
|
|
99
|
+
await page.getByRole("link", { name: "Documentation" }).click();
|
|
100
|
+
const newPage = await pagePromise;
|
|
101
|
+
|
|
102
|
+
await newPage.waitForLoadState();
|
|
103
|
+
|
|
104
|
+
expect(newPage.url()).toContain("docs.example.com");
|
|
105
|
+
await expect(newPage.getByRole("heading", { level: 1 })).toBeVisible();
|
|
106
|
+
|
|
107
|
+
// Original page still there
|
|
108
|
+
expect(page.url()).toContain("/resources");
|
|
109
|
+
|
|
110
|
+
await newPage.close();
|
|
111
|
+
});
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
### Intercept New Tab
|
|
115
|
+
|
|
116
|
+
```typescript
|
|
117
|
+
test("prevent new tab for testing", async ({ page }) => {
|
|
118
|
+
await page.goto("/links");
|
|
119
|
+
|
|
120
|
+
// Remove target="_blank" to keep navigation in same tab
|
|
121
|
+
await page.evaluate(() => {
|
|
122
|
+
document.querySelectorAll('a[target="_blank"]').forEach((a) => {
|
|
123
|
+
a.removeAttribute("target");
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
// Now link opens in same tab
|
|
128
|
+
await page.getByRole("link", { name: "External Site" }).click();
|
|
129
|
+
|
|
130
|
+
// Can test the destination page
|
|
131
|
+
await expect(page).toHaveURL(/external-site\.com/);
|
|
132
|
+
});
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
## OAuth Flows
|
|
136
|
+
|
|
137
|
+
### Google OAuth Popup
|
|
138
|
+
|
|
139
|
+
```typescript
|
|
140
|
+
test("Google OAuth login", async ({ page }) => {
|
|
141
|
+
await page.goto("/login");
|
|
142
|
+
|
|
143
|
+
const popupPromise = page.waitForEvent("popup");
|
|
144
|
+
await page.getByRole("button", { name: "Sign in with Google" }).click();
|
|
145
|
+
const popup = await popupPromise;
|
|
146
|
+
|
|
147
|
+
await popup.waitForLoadState();
|
|
148
|
+
|
|
149
|
+
// Handle Google's OAuth flow
|
|
150
|
+
await popup.getByLabel("Email or phone").fill("test@gmail.com");
|
|
151
|
+
await popup.getByRole("button", { name: "Next" }).click();
|
|
152
|
+
|
|
153
|
+
await popup.getByLabel("Enter your password").fill("password");
|
|
154
|
+
await popup.getByRole("button", { name: "Next" }).click();
|
|
155
|
+
|
|
156
|
+
// Wait for redirect back and popup close
|
|
157
|
+
await popup.waitForEvent("close");
|
|
158
|
+
|
|
159
|
+
// Verify logged in on main page
|
|
160
|
+
await expect(page.getByText("Welcome, Test User")).toBeVisible();
|
|
161
|
+
});
|
|
162
|
+
```
|
|
163
|
+
|
|
164
|
+
### Mock OAuth (Recommended)
|
|
165
|
+
|
|
166
|
+
```typescript
|
|
167
|
+
test("mock OAuth flow", async ({ page, context }) => {
|
|
168
|
+
// Mock the OAuth callback instead of real flow
|
|
169
|
+
await page.route("**/auth/callback**", async (route) => {
|
|
170
|
+
// Simulate successful OAuth
|
|
171
|
+
const url = new URL(route.request().url());
|
|
172
|
+
url.searchParams.set("code", "mock-auth-code");
|
|
173
|
+
await route.fulfill({
|
|
174
|
+
status: 302,
|
|
175
|
+
headers: { Location: "/dashboard" },
|
|
176
|
+
});
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
// Mock token exchange
|
|
180
|
+
await page.route("**/api/auth/token", (route) =>
|
|
181
|
+
route.fulfill({
|
|
182
|
+
json: {
|
|
183
|
+
access_token: "mock-token",
|
|
184
|
+
user: { name: "Test User", email: "test@example.com" },
|
|
185
|
+
},
|
|
186
|
+
}),
|
|
187
|
+
);
|
|
188
|
+
|
|
189
|
+
await page.goto("/login");
|
|
190
|
+
await page.getByRole("button", { name: "Sign in with Google" }).click();
|
|
191
|
+
|
|
192
|
+
// Should redirect to dashboard without actual OAuth
|
|
193
|
+
await expect(page).toHaveURL("/dashboard");
|
|
194
|
+
await expect(page.getByText("Welcome, Test User")).toBeVisible();
|
|
195
|
+
});
|
|
196
|
+
```
|
|
197
|
+
|
|
198
|
+
### OAuth Fixture
|
|
199
|
+
|
|
200
|
+
> **For comprehensive OAuth mocking patterns** (fixtures, multiple providers, SAML SSO), see [third-party.md](third-party.md#oauthsso-mocking). This section focuses on popup window handling mechanics for OAuth flows.
|
|
201
|
+
|
|
202
|
+
## Multiple Windows
|
|
203
|
+
|
|
204
|
+
### Test Across Multiple Windows
|
|
205
|
+
|
|
206
|
+
```typescript
|
|
207
|
+
test("sync between windows", async ({ context }) => {
|
|
208
|
+
// Open two pages
|
|
209
|
+
const page1 = await context.newPage();
|
|
210
|
+
const page2 = await context.newPage();
|
|
211
|
+
|
|
212
|
+
await page1.goto("/dashboard");
|
|
213
|
+
await page2.goto("/dashboard");
|
|
214
|
+
|
|
215
|
+
// Make change in first window
|
|
216
|
+
await page1.getByRole("button", { name: "Add Item" }).click();
|
|
217
|
+
await page1.getByLabel("Name").fill("New Item");
|
|
218
|
+
await page1.getByRole("button", { name: "Save" }).click();
|
|
219
|
+
|
|
220
|
+
// Should sync to second window (if app supports real-time sync)
|
|
221
|
+
await expect(page2.getByText("New Item")).toBeVisible({ timeout: 10000 });
|
|
222
|
+
});
|
|
223
|
+
```
|
|
224
|
+
|
|
225
|
+
### Different Users in Different Windows
|
|
226
|
+
|
|
227
|
+
> **For multi-user collaboration patterns** (admin/user interactions, real-time collaboration, role-based testing, concurrent actions), see [multi-user.md](multi-user.md). This file focuses on single-user scenarios with multiple tabs/windows/popups.
|
|
228
|
+
|
|
229
|
+
## Tab Coordination
|
|
230
|
+
|
|
231
|
+
### Switch Between Tabs
|
|
232
|
+
|
|
233
|
+
```typescript
|
|
234
|
+
test("manage multiple tabs", async ({ context }) => {
|
|
235
|
+
const page1 = await context.newPage();
|
|
236
|
+
await page1.goto("/editor");
|
|
237
|
+
|
|
238
|
+
const page2 = await context.newPage();
|
|
239
|
+
await page2.goto("/preview");
|
|
240
|
+
|
|
241
|
+
// Edit in first tab
|
|
242
|
+
await page1.bringToFront();
|
|
243
|
+
await page1.getByLabel("Content").fill("Hello World");
|
|
244
|
+
|
|
245
|
+
// Check preview in second tab
|
|
246
|
+
await page2.bringToFront();
|
|
247
|
+
await page2.reload(); // If preview needs refresh
|
|
248
|
+
await expect(page2.getByText("Hello World")).toBeVisible();
|
|
249
|
+
});
|
|
250
|
+
```
|
|
251
|
+
|
|
252
|
+
### Close All Tabs Except One
|
|
253
|
+
|
|
254
|
+
```typescript
|
|
255
|
+
test("cleanup tabs after test", async ({ context }) => {
|
|
256
|
+
const mainPage = await context.newPage();
|
|
257
|
+
await mainPage.goto("/");
|
|
258
|
+
|
|
259
|
+
// Open several popups during test
|
|
260
|
+
for (let i = 0; i < 3; i++) {
|
|
261
|
+
const popup = await context.newPage();
|
|
262
|
+
await popup.goto(`/popup/${i}`);
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
// Close all except main page
|
|
266
|
+
for (const page of context.pages()) {
|
|
267
|
+
if (page !== mainPage) {
|
|
268
|
+
await page.close();
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
expect(context.pages()).toHaveLength(1);
|
|
273
|
+
});
|
|
274
|
+
```
|
|
275
|
+
|
|
276
|
+
## Anti-Patterns to Avoid
|
|
277
|
+
|
|
278
|
+
| Anti-Pattern | Problem | Solution |
|
|
279
|
+
| ----------------------- | ------------------------------ | ------------------------------------------ |
|
|
280
|
+
| Not waiting for popup | Race condition | Use `waitForEvent("popup")` before trigger |
|
|
281
|
+
| Testing real OAuth | Slow, flaky, needs credentials | Mock OAuth endpoints |
|
|
282
|
+
| Assuming popup opens | May be blocked | Handle both open and blocked cases |
|
|
283
|
+
| Not closing extra pages | Resource leak | Close pages in cleanup |
|
|
284
|
+
|
|
285
|
+
## Related References
|
|
286
|
+
|
|
287
|
+
- **Authentication**: See [fixtures-hooks.md](../core/fixtures-hooks.md) for auth patterns
|
|
288
|
+
- **Network**: See [network-advanced.md](network-advanced.md) for mocking OAuth
|