@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.
Files changed (97) hide show
  1. package/.agents/skills/e2e-testing-expert/SKILL.md +28 -0
  2. package/.agents/skills/frontend-design/LICENSE.txt +177 -0
  3. package/.agents/skills/frontend-design/SKILL.md +42 -0
  4. package/.agents/skills/nodejs-backend-patterns/SKILL.md +639 -0
  5. package/.agents/skills/nodejs-backend-patterns/references/advanced-patterns.md +430 -0
  6. package/.agents/skills/playwright-best-practices/LICENSE.md +7 -0
  7. package/.agents/skills/playwright-best-practices/README.md +147 -0
  8. package/.agents/skills/playwright-best-practices/SKILL.md +303 -0
  9. package/.agents/skills/playwright-best-practices/advanced/authentication-flows.md +360 -0
  10. package/.agents/skills/playwright-best-practices/advanced/authentication.md +871 -0
  11. package/.agents/skills/playwright-best-practices/advanced/clock-mocking.md +364 -0
  12. package/.agents/skills/playwright-best-practices/advanced/mobile-testing.md +409 -0
  13. package/.agents/skills/playwright-best-practices/advanced/multi-context.md +288 -0
  14. package/.agents/skills/playwright-best-practices/advanced/multi-user.md +393 -0
  15. package/.agents/skills/playwright-best-practices/advanced/network-advanced.md +452 -0
  16. package/.agents/skills/playwright-best-practices/advanced/third-party.md +464 -0
  17. package/.agents/skills/playwright-best-practices/architecture/pom-vs-fixtures.md +363 -0
  18. package/.agents/skills/playwright-best-practices/architecture/test-architecture.md +369 -0
  19. package/.agents/skills/playwright-best-practices/architecture/when-to-mock.md +383 -0
  20. package/.agents/skills/playwright-best-practices/browser-apis/browser-apis.md +391 -0
  21. package/.agents/skills/playwright-best-practices/browser-apis/iframes.md +403 -0
  22. package/.agents/skills/playwright-best-practices/browser-apis/service-workers.md +504 -0
  23. package/.agents/skills/playwright-best-practices/browser-apis/websockets.md +403 -0
  24. package/.agents/skills/playwright-best-practices/core/annotations.md +424 -0
  25. package/.agents/skills/playwright-best-practices/core/assertions-waiting.md +361 -0
  26. package/.agents/skills/playwright-best-practices/core/configuration.md +452 -0
  27. package/.agents/skills/playwright-best-practices/core/fixtures-hooks.md +417 -0
  28. package/.agents/skills/playwright-best-practices/core/global-setup.md +434 -0
  29. package/.agents/skills/playwright-best-practices/core/locators.md +242 -0
  30. package/.agents/skills/playwright-best-practices/core/page-object-model.md +315 -0
  31. package/.agents/skills/playwright-best-practices/core/projects-dependencies.md +453 -0
  32. package/.agents/skills/playwright-best-practices/core/test-data.md +492 -0
  33. package/.agents/skills/playwright-best-practices/core/test-suite-structure.md +361 -0
  34. package/.agents/skills/playwright-best-practices/core/test-tags.md +298 -0
  35. package/.agents/skills/playwright-best-practices/debugging/console-errors.md +420 -0
  36. package/.agents/skills/playwright-best-practices/debugging/debugging.md +504 -0
  37. package/.agents/skills/playwright-best-practices/debugging/error-testing.md +360 -0
  38. package/.agents/skills/playwright-best-practices/debugging/flaky-tests.md +496 -0
  39. package/.agents/skills/playwright-best-practices/frameworks/angular.md +530 -0
  40. package/.agents/skills/playwright-best-practices/frameworks/nextjs.md +469 -0
  41. package/.agents/skills/playwright-best-practices/frameworks/react.md +531 -0
  42. package/.agents/skills/playwright-best-practices/frameworks/vue.md +574 -0
  43. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/ci-cd.md +468 -0
  44. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/docker.md +283 -0
  45. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/github-actions.md +546 -0
  46. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/gitlab.md +397 -0
  47. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/other-providers.md +521 -0
  48. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/parallel-sharding.md +371 -0
  49. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/performance.md +453 -0
  50. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/reporting.md +424 -0
  51. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/test-coverage.md +497 -0
  52. package/.agents/skills/playwright-best-practices/testing-patterns/accessibility.md +359 -0
  53. package/.agents/skills/playwright-best-practices/testing-patterns/api-testing.md +719 -0
  54. package/.agents/skills/playwright-best-practices/testing-patterns/browser-extensions.md +506 -0
  55. package/.agents/skills/playwright-best-practices/testing-patterns/canvas-webgl.md +493 -0
  56. package/.agents/skills/playwright-best-practices/testing-patterns/component-testing.md +500 -0
  57. package/.agents/skills/playwright-best-practices/testing-patterns/drag-drop.md +576 -0
  58. package/.agents/skills/playwright-best-practices/testing-patterns/electron.md +509 -0
  59. package/.agents/skills/playwright-best-practices/testing-patterns/file-operations.md +377 -0
  60. package/.agents/skills/playwright-best-practices/testing-patterns/file-upload-download.md +562 -0
  61. package/.agents/skills/playwright-best-practices/testing-patterns/forms-validation.md +561 -0
  62. package/.agents/skills/playwright-best-practices/testing-patterns/graphql-testing.md +331 -0
  63. package/.agents/skills/playwright-best-practices/testing-patterns/i18n.md +508 -0
  64. package/.agents/skills/playwright-best-practices/testing-patterns/performance-testing.md +476 -0
  65. package/.agents/skills/playwright-best-practices/testing-patterns/security-testing.md +430 -0
  66. package/.agents/skills/playwright-best-practices/testing-patterns/visual-regression.md +634 -0
  67. package/.env.example +21 -0
  68. package/README.md +30 -0
  69. package/bin/arcality.mjs +86 -0
  70. package/package.json +66 -0
  71. package/playwright.config.ts +12 -0
  72. package/scripts/cleanup-qmsdev.mjs +63 -0
  73. package/scripts/discover-view.mjs +52 -0
  74. package/scripts/extract-view.mjs +64 -0
  75. package/scripts/gen-and-run.mjs +838 -0
  76. package/scripts/init.mjs +290 -0
  77. package/scripts/migrate-to-central-out.mjs +157 -0
  78. package/scripts/postinstall.mjs +63 -0
  79. package/scripts/rebrand-report.mjs +241 -0
  80. package/scripts/setup.mjs +166 -0
  81. package/src/KnowledgeService.ts +239 -0
  82. package/src/arcalityClient.mjs +266 -0
  83. package/src/configLoader.mjs +179 -0
  84. package/src/configManager.mjs +172 -0
  85. package/src/consoleBanner.ts +32 -0
  86. package/src/envSetup.ts +205 -0
  87. package/src/index.ts +25 -0
  88. package/src/projectInspector.ts +42 -0
  89. package/src/services/collectiveMemoryService.ts +178 -0
  90. package/src/testRunner.ts +201 -0
  91. package/tests/_helpers/ArcalityReporter.ts +490 -0
  92. package/tests/_helpers/agentic-runner.spec.ts +741 -0
  93. package/tests/_helpers/ai-agent-helper.ts +1573 -0
  94. package/tests/_helpers/discover-view.spec.ts +238 -0
  95. package/tests/_helpers/extract-view.spec.ts +118 -0
  96. package/tests/_helpers/qa-tools.ts +333 -0
  97. 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