SurfSense/.cursor/skills/playwright-testing/multi-context-and-popups.md
2026-05-04 13:54:13 +05:30

526 lines
21 KiB
Markdown
Executable file

# Multi-Context, Popups, and New Windows
> **When to use**: Handling popup windows, new tabs, OAuth authorization flows, payment gateway redirects, multi-tab coordination, and any scenario where your application opens additional browser windows or tabs.
> **Prerequisites**: [core/assertions-and-waiting.md](assertions-and-waiting.md), [core/fixtures-and-hooks.md](fixtures-and-hooks.md)
## Quick Reference
```typescript
// Catch a popup triggered by a click
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Open preview' }).click();
const popup = await popupPromise;
await popup.waitForLoadState();
await expect(popup.getByRole('heading')).toContainText('Preview');
// List all open pages in a context
const allPages = context.pages();
console.log(`Open tabs: ${allPages.length}`);
```
**Key concept**: In Playwright, a "popup" is any new page opened by `window.open()`, `target="_blank"` links, or JavaScript-triggered new windows. They all fire the `popup` event on the originating page.
## Patterns
### Handling Basic Popups
**Use when**: A user action opens a new tab or window and you need to interact with it.
**Avoid when**: The "popup" is actually a modal dialog within the same page -- use `getByRole('dialog')` instead.
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('handle popup opened by target="_blank" link', async ({ page }) => {
await page.goto('/help');
// Set up the popup listener BEFORE the action that triggers it
const popupPromise = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Documentation' }).click();
const popup = await popupPromise;
// Wait for the popup to load
await popup.waitForLoadState();
// Interact with the popup
await expect(popup.getByRole('heading', { level: 1 })).toContainText('Documentation');
expect(popup.url()).toContain('/docs');
// Close the popup when done
await popup.close();
});
test('handle popup opened by window.open()', async ({ page }) => {
await page.goto('/reports');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Print report' }).click();
const popup = await popupPromise;
await popup.waitForLoadState();
await expect(popup.getByTestId('print-preview')).toBeVisible();
// The original page is still accessible
await expect(page.getByRole('heading', { name: 'Reports' })).toBeVisible();
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('handle popup opened by target="_blank" link', async ({ page }) => {
await page.goto('/help');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Documentation' }).click();
const popup = await popupPromise;
await popup.waitForLoadState();
await expect(popup.getByRole('heading', { level: 1 })).toContainText('Documentation');
await popup.close();
});
```
### OAuth Popup Flows
**Use when**: Your app opens a third-party OAuth window (Google, GitHub, Microsoft, etc.) for authentication.
**Avoid when**: You can bypass OAuth entirely by injecting auth tokens directly -- see [core/authentication.md](authentication.md).
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('Google OAuth popup flow', async ({ page }) => {
await page.goto('/login');
// Listen for the popup before clicking the OAuth button
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Sign in with Google' }).click();
const oauthPopup = await popupPromise;
// Wait for the OAuth provider page to load
await oauthPopup.waitForLoadState();
expect(oauthPopup.url()).toContain('accounts.google.com');
// Fill in credentials on the OAuth provider page
await oauthPopup.getByLabel('Email or phone').fill('testuser@gmail.com');
await oauthPopup.getByRole('button', { name: 'Next' }).click();
await oauthPopup.getByLabel('Enter your password').fill('test-password');
await oauthPopup.getByRole('button', { name: 'Next' }).click();
// The popup closes automatically after authorization
// Wait for the original page to receive the auth callback
await page.waitForURL('/dashboard');
await expect(page.getByRole('heading', { name: 'Dashboard' })).toBeVisible();
});
test('GitHub OAuth popup flow', async ({ page }) => {
await page.goto('/login');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Sign in with GitHub' }).click();
const popup = await popupPromise;
await popup.waitForLoadState();
expect(popup.url()).toContain('github.com');
await popup.getByLabel('Username or email address').fill('testuser');
await popup.getByLabel('Password').fill('test-password');
await popup.getByRole('button', { name: 'Sign in' }).click();
// Authorize the app if prompted
const authorizeButton = popup.getByRole('button', { name: 'Authorize' });
if (await authorizeButton.isVisible({ timeout: 3000 }).catch(() => false)) {
await authorizeButton.click();
}
// Popup closes, original page redirects
await page.waitForURL('/dashboard');
await expect(page.getByText(/Welcome/)).toBeVisible();
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('GitHub OAuth popup flow', async ({ page }) => {
await page.goto('/login');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Sign in with GitHub' }).click();
const popup = await popupPromise;
await popup.waitForLoadState();
await popup.getByLabel('Username or email address').fill('testuser');
await popup.getByLabel('Password').fill('test-password');
await popup.getByRole('button', { name: 'Sign in' }).click();
await page.waitForURL('/dashboard');
await expect(page.getByText(/Welcome/)).toBeVisible();
});
```
### Payment Gateway Popups
**Use when**: Checkout flows open a payment provider in a new window (PayPal, 3D Secure verification).
**Avoid when**: The payment gateway loads in an iframe on the same page -- use `frameLocator` instead.
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('PayPal popup checkout flow', async ({ page }) => {
await page.goto('/checkout');
await page.getByLabel('Email').fill('buyer@example.com');
await page.getByRole('button', { name: 'Proceed to payment' }).click();
// PayPal opens in a popup
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Pay with PayPal' }).click();
const paypalPopup = await popupPromise;
await paypalPopup.waitForLoadState();
expect(paypalPopup.url()).toContain('paypal.com');
// Complete PayPal flow
await paypalPopup.getByLabel('Email').fill('buyer@paypal-test.com');
await paypalPopup.getByRole('button', { name: 'Next' }).click();
await paypalPopup.getByLabel('Password').fill('test-password');
await paypalPopup.getByRole('button', { name: 'Log In' }).click();
await paypalPopup.getByRole('button', { name: 'Complete Purchase' }).click();
// Popup closes, return to original page
await page.waitForURL('/order/confirmation');
await expect(page.getByText('Payment successful')).toBeVisible();
});
test('3D Secure verification popup', async ({ page }) => {
await page.goto('/checkout');
await page.getByLabel('Card number').fill('4000000000003220'); // 3DS test card
await page.getByLabel('Expiry').fill('12/28');
await page.getByLabel('CVC').fill('123');
await page.getByRole('button', { name: 'Pay' }).click();
// 3DS challenge opens in popup or iframe -- handle both
const popupPromise = page.waitForEvent('popup', { timeout: 5000 }).catch(() => null);
const popup = await popupPromise;
if (popup) {
await popup.waitForLoadState();
await popup.getByRole('button', { name: 'Complete authentication' }).click();
} else {
// Fallback: 3DS in iframe
const frame = page.frameLocator('iframe[name*="challenge"]');
await frame.getByRole('button', { name: 'Complete authentication' }).click();
}
await expect(page.getByText('Payment successful')).toBeVisible();
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('PayPal popup checkout flow', async ({ page }) => {
await page.goto('/checkout');
await page.getByRole('button', { name: 'Proceed to payment' }).click();
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Pay with PayPal' }).click();
const paypalPopup = await popupPromise;
await paypalPopup.waitForLoadState();
await paypalPopup.getByLabel('Email').fill('buyer@paypal-test.com');
await paypalPopup.getByRole('button', { name: 'Next' }).click();
await paypalPopup.getByLabel('Password').fill('test-password');
await paypalPopup.getByRole('button', { name: 'Log In' }).click();
await paypalPopup.getByRole('button', { name: 'Complete Purchase' }).click();
await page.waitForURL('/order/confirmation');
await expect(page.getByText('Payment successful')).toBeVisible();
});
```
### Multi-Tab Coordination
**Use when**: Testing scenarios where multiple tabs share state -- real-time collaboration, shopping cart sync, or session management across tabs.
**Avoid when**: Each tab is independent and can be tested in separate test cases.
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('cart updates reflect across tabs', async ({ context }) => {
// Open two tabs in the same context (shared cookies/storage)
const page1 = await context.newPage();
const page2 = await context.newPage();
await page1.goto('/products');
await page2.goto('/cart');
// Add item in tab 1
await page1.getByRole('button', { name: 'Add to cart' }).first().click();
// Tab 2 should reflect the update (via WebSocket, polling, or storage event)
await expect(page2.getByTestId('cart-count')).toHaveText('1', { timeout: 5000 });
});
test('logout in one tab logs out all tabs', async ({ context }) => {
const page1 = await context.newPage();
const page2 = await context.newPage();
// Both tabs are on authenticated pages
await page1.goto('/dashboard');
await page2.goto('/settings');
// Log out from tab 1
await page1.getByRole('button', { name: 'Log out' }).click();
await page1.waitForURL('/login');
// Tab 2 should redirect to login on next action or automatically
await page2.reload();
expect(page2.url()).toContain('/login');
});
test('manage multiple tabs with context.pages()', async ({ context, page }) => {
await page.goto('/dashboard');
// Open several new tabs
const popupPromise1 = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Report A' }).click();
const tab1 = await popupPromise1;
const popupPromise2 = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Report B' }).click();
const tab2 = await popupPromise2;
// List all pages in this context
const allPages = context.pages();
expect(allPages).toHaveLength(3); // original + 2 popups
// Interact with specific tabs
await tab1.waitForLoadState();
await expect(tab1.getByRole('heading')).toContainText('Report A');
await tab2.waitForLoadState();
await expect(tab2.getByRole('heading')).toContainText('Report B');
// Close tabs when done
await tab2.close();
await tab1.close();
expect(context.pages()).toHaveLength(1);
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('cart updates reflect across tabs', async ({ context }) => {
const page1 = await context.newPage();
const page2 = await context.newPage();
await page1.goto('/products');
await page2.goto('/cart');
await page1.getByRole('button', { name: 'Add to cart' }).first().click();
await expect(page2.getByTestId('cart-count')).toHaveText('1', { timeout: 5000 });
});
test('manage tabs with context.pages()', async ({ context, page }) => {
await page.goto('/dashboard');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Report A' }).click();
const newTab = await popupPromise;
expect(context.pages()).toHaveLength(2);
await newTab.close();
expect(context.pages()).toHaveLength(1);
});
```
### Download Triggers in New Tabs
**Use when**: A download is triggered by opening a new tab (e.g., PDF generation that opens in a new window before downloading).
**Avoid when**: The download starts directly without opening a new tab -- use `page.waitForEvent('download')` directly.
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('PDF download from new tab', async ({ page }) => {
await page.goto('/invoices');
// Some apps open the PDF in a new tab, which then triggers a download
const popupPromise = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Download Invoice #123' }).click();
const popup = await popupPromise;
// The popup may directly trigger a download
const downloadPromise = popup.waitForEvent('download');
const download = await downloadPromise;
expect(download.suggestedFilename()).toContain('invoice');
const path = await download.path();
expect(path).toBeTruthy();
await popup.close();
});
test('export opens in new tab then auto-downloads', async ({ page }) => {
await page.goto('/reports');
// Handle both: popup that downloads AND popup that shows content
const popupPromise = page.waitForEvent('popup');
await page.getByRole('button', { name: 'Export CSV' }).click();
const popup = await popupPromise;
// Try to catch a download; if no download, the popup has the content
try {
const download = await popup.waitForEvent('download', { timeout: 5000 });
const filename = download.suggestedFilename();
expect(filename).toMatch(/\.csv$/);
await download.saveAs(`./downloads/${filename}`);
} catch {
// No download event — content displayed in the popup
const content = await popup.textContent('body');
expect(content).toContain('Report Data');
}
await popup.close();
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('PDF download from new tab', async ({ page }) => {
await page.goto('/invoices');
const popupPromise = page.waitForEvent('popup');
await page.getByRole('link', { name: 'Download Invoice #123' }).click();
const popup = await popupPromise;
const downloadPromise = popup.waitForEvent('download');
const download = await downloadPromise;
expect(download.suggestedFilename()).toContain('invoice');
await popup.close();
});
```
### Multi-Context for Isolated Sessions
**Use when**: Testing interactions between different users (e.g., admin and regular user, two chat participants).
**Avoid when**: You only need a single user perspective -- a single context is sufficient.
**TypeScript**
```typescript
import { test, expect } from '@playwright/test';
test('admin sees user changes in real-time', async ({ browser }) => {
// Create separate contexts for two users (separate sessions)
const adminContext = await browser.newContext();
const userContext = await browser.newContext();
const adminPage = await adminContext.newPage();
const userPage = await userContext.newPage();
// Admin logs in and watches the user list
await adminPage.goto('/admin/users');
// User signs up
await userPage.goto('/register');
await userPage.getByLabel('Name').fill('New User');
await userPage.getByLabel('Email').fill('newuser@example.com');
await userPage.getByLabel('Password').fill('password123');
await userPage.getByRole('button', { name: 'Register' }).click();
// Admin should see the new user appear
await expect(adminPage.getByText('newuser@example.com')).toBeVisible({ timeout: 10000 });
await adminContext.close();
await userContext.close();
});
```
**JavaScript**
```javascript
const { test, expect } = require('@playwright/test');
test('admin sees user changes in real-time', async ({ browser }) => {
const adminContext = await browser.newContext();
const userContext = await browser.newContext();
const adminPage = await adminContext.newPage();
const userPage = await userContext.newPage();
await adminPage.goto('/admin/users');
await userPage.goto('/register');
await userPage.getByLabel('Name').fill('New User');
await userPage.getByLabel('Email').fill('newuser@example.com');
await userPage.getByLabel('Password').fill('password123');
await userPage.getByRole('button', { name: 'Register' }).click();
await expect(adminPage.getByText('newuser@example.com')).toBeVisible({ timeout: 10000 });
await adminContext.close();
await userContext.close();
});
```
## Decision Guide
| Scenario | Approach | Why |
|---|---|---|
| `target="_blank"` link | `page.waitForEvent('popup')` | Playwright fires `popup` for all new windows/tabs |
| `window.open()` call | `page.waitForEvent('popup')` | Same mechanism regardless of how the window opens |
| OAuth login popup | `waitForEvent('popup')` + interact + wait for close | OAuth providers redirect back and close the popup |
| Payment popup (PayPal) | `waitForEvent('popup')` + complete flow | Same as OAuth but with payment-specific UI |
| Download in new tab | `popup.waitForEvent('download')` | Catch the download event on the popup page |
| Multiple tabs, same user | Open pages in the same `context` | Shared cookies, localStorage, session |
| Multiple users (separate sessions) | Create separate `browser.newContext()` per user | Isolated cookies, storage, auth state |
| Tab sync testing | Multiple `context.newPage()` + assert shared state | Tests real-time state synchronization |
| Popup that may or may not appear | `waitForEvent('popup', { timeout })` with try/catch | Graceful handling of conditional popups |
## Anti-Patterns
| Don't Do This | Problem | Do This Instead |
|---|---|---|
| Clicking then calling `waitForEvent('popup')` | Popup may open before listener is registered (race condition) | Set up `waitForEvent` BEFORE the click |
| Using `context.pages()[1]` to get a popup | Index order is not guaranteed; brittle | Use `waitForEvent('popup')` which returns the exact page |
| Forgetting `popup.waitForLoadState()` | Popup page may not be loaded when you interact with it | Always call `waitForLoadState()` after receiving the popup |
| Not closing popups after test | Leaked pages consume memory and may affect subsequent tests | Close popups with `popup.close()` in the test or use fixtures |
| Using separate `browser.newContext()` when tabs should share state | Separate contexts have separate cookies/sessions | Use `context.newPage()` for tabs in the same session |
| Using `context.newPage()` for isolated users | Pages in the same context share state | Use `browser.newContext()` for separate user sessions |
| `page.waitForTimeout()` after popup trigger | Popup timing is unpredictable | Use `page.waitForEvent('popup')` which resolves when the popup opens |
| Catching popup on the wrong page | Popup event fires on the page that triggered it | Listen for `popup` on the page where the click/action happens |
## Troubleshooting
| Symptom | Likely Cause | Fix |
|---|---|---|
| `waitForEvent('popup')` times out | The action did not open a new window; it navigated in the same tab | Check if the link has `target="_blank"` or if `window.open` is called |
| Popup opens but is blank | Popup is still loading; you interacted too early | Add `await popup.waitForLoadState()` before interacting |
| Popup URL is `about:blank` | Popup is created empty, then navigated by JavaScript | Wait for `popup.waitForURL()` with the expected URL pattern |
| OAuth popup is blocked by browser | Pop-up blocker is active | Playwright disables pop-up blocking by default; check if a browser arg re-enables it |
| Two popups open but only one is caught | `waitForEvent` resolves for the first event only | Use `page.on('popup', callback)` to catch all popups, or chain two `waitForEvent` calls |
| `context.pages()` returns unexpected count | Previous test left pages open | Close all extra pages in `afterEach` or use per-test contexts |
| Popup closes before interaction completes | The app or OAuth provider auto-closes after timeout | Increase test speed or remove `slowMo`; interact with the popup immediately |
| Cross-origin popup interaction fails | Popup navigated to a different origin | Playwright handles cross-origin popups; ensure you are not setting `--disable-web-security` |
## Related
- [core/authentication.md](authentication.md) -- bypassing OAuth popups with stored auth state
- [core/multi-user-and-collaboration.md](multi-user-and-collaboration.md) -- multi-user real-time collaboration testing
- [core/file-operations.md](file-operations.md) -- file download handling without popups
- [core/third-party-integrations.md](third-party-integrations.md) -- mocking OAuth providers to avoid real popups
- [core/fixtures-and-hooks.md](fixtures-and-hooks.md) -- fixtures for multi-context setups