page.screenshot() does the job in two lines for a local capture or a test assertion. The friction arrives later: full-page shots that get clipped by sticky headers, Chromium installation issues in CI, and the long tail of keeping a browser fleet running in production. This guide covers the screenshot methods that work, the ones with hidden edge cases, and the point where offloading to an API makes more sense.
The basic Playwright screenshot
Launch a browser, open a page, capture:
import { chromium } from 'playwright';
const browser = await chromium.launch();
const page = await browser.newPage();
await page.goto('https://example.com', { waitUntil: 'networkidle' });
await page.screenshot({ path: 'example.png' });
await browser.close();
The waitUntil: 'networkidle' option is worth stating explicitly. The default load event fires as soon as the HTML and blocking scripts are done, but client-rendered content, web fonts, and lazy-loaded images often arrive later. networkidle waits until there are no network connections open for at least 500ms, which is a much better proxy for "the page actually looks right."
Full-page screenshots
By default page.screenshot() captures the viewport only. To capture the full scrolling page, set fullPage: true:
await page.screenshot({ path: 'full.png', fullPage: true });
Playwright expands the viewport to the document's full scroll height before capturing. This works reliably on most static pages.
Two things break it:
-
Sticky and fixed-position elements. A header with
position: fixedrenders in its normal viewport position for every "slice" Playwright captures, so it can appear repeated or floating over content in the final stitched image. There is no clean built-in workaround short of hiding the element with apage.addStyleTagbefore capturing. -
Lazy-loaded content. If a page only renders sections when they scroll into view,
fullPage: truecan come back short. Scroll the page yourself before capturing to trigger those renders:
await page.evaluate(async () => {
await new Promise<void>((resolve) => {
let scrolled = 0;
const step = 500;
const timer = setInterval(() => {
window.scrollBy(0, step);
scrolled += step;
if (scrolled >= document.body.scrollHeight) {
clearInterval(timer);
window.scrollTo(0, 0);
resolve();
}
}, 100);
});
});
await page.screenshot({ path: 'full.png', fullPage: true });
Screenshotting a specific element
Playwright's locator API makes element-level captures cleaner than Puppeteer's ElementHandle approach. Call .screenshot() on any locator and Playwright crops to that element's bounding box:
// Capture a single component by CSS selector
await page.locator('#pricing-card').screenshot({ path: 'card.png' });
// Or by role
await page.getByRole('dialog').screenshot({ path: 'modal.png' });
If the element is below the fold, Playwright scrolls it into view before capturing. If the selector matches nothing, the call throws, so wrap it in a waitFor when the element might not be immediately present:
const card = page.locator('#pricing-card');
await card.waitFor({ state: 'visible' });
await card.screenshot({ path: 'card.png' });
Format and quality options
Playwright defaults to PNG. For production use cases where file size matters, switch to JPEG or WebP:
await page.screenshot({
path: 'page.webp',
type: 'webp',
quality: 85, // 0–100, only for jpeg and webp
});
PNG is lossless and best for visual regression tests where pixel-level accuracy matters. JPEG and WebP are better for images that will be served over HTTP, since they are significantly smaller at comparable quality.
Screenshots in CI with GitHub Actions
Installing Playwright in CI requires browser binaries and their system dependencies. Use npx playwright install --with-deps (not just npx playwright install) to pull in the OS libraries Chromium needs:
- name: Install Playwright browsers
run: npx playwright install --with-deps chromium
- name: Run tests
run: npx playwright test
- name: Upload test artifacts
if: always()
uses: actions/upload-artifact@v4
with:
name: playwright-report
path: test-results/
retention-days: 7
The if: always() on the upload step is important: test artifacts are most useful when tests fail, and a plain if: success() would skip the upload precisely when you need the diffs.
Playwright's --with-deps flag is the most common CI pitfall. Without it, Chromium launches fine locally (because those libraries are already installed) but fails in a minimal CI container with a cryptic error while loading shared libraries message.
Visual regression with screenshots
Playwright's built-in snapshot assertion compares a screenshot against a stored baseline:
await expect(page).toHaveScreenshot('homepage.png');
On the first run, Playwright writes the baseline to __snapshots__/. On subsequent runs it compares pixel-by-pixel and fails if the diff exceeds the configured threshold. Pass { maxDiffPixelRatio: 0.01 } to tolerate minor anti-aliasing differences:
await expect(page.locator('.hero-section')).toHaveScreenshot('hero.png', {
maxDiffPixelRatio: 0.01,
});
For a longer look at visual regression workflows, the visual regression testing guide covers when snapshot tests make sense and how to keep them from becoming brittle.
Where running Playwright yourself gets costly
Playwright is the right tool for test suites that already drive a browser. When screenshots are a standalone production feature, the overhead adds up:
- Chromium in production. Deploying a headless browser to a serverless function means bundling system libraries, increasing cold-start times, and hitting the payload size limits of most platforms.
- Concurrency. One browser instance handles one capture at a time well. Handling bursts means a pool, a queue, and back-pressure logic.
- Security surface. A browser that can render arbitrary user-submitted URLs needs SSRF protection (block private IP ranges), sandboxing, and regular security patches.
This is the point where a dedicated screenshot API is faster to operate: the browser fleet, SSRF guards, and scaling are someone else's problem.
The same capture as an API call
Here is the full-page capture above as a single request to Grabbit. No browser to provision or patch:
curl https://api.grabbit.live/v1/grabs \
-H "Authorization: Bearer sk_live_..." \
-H "Content-Type: application/json" \
-d '{
"url": "https://example.com",
"width": 1280,
"height": 720,
"full_page": true,
"format": "webp"
}'
The response includes a hosted image_url you can use directly:
{
"id": "grb_01jx...",
"status": "done",
"image_url": "https://cdn.grabbit.live/grabs/grb_01jx....webp",
"width": 1280,
"format": "webp",
"bytes": 52340,
"execution_ms": 1240
}
The Playwright options you reach for most translate directly: fullPage: true becomes "full_page": true, the element locator pattern becomes a "selector" field, and manual wait time becomes "delay_ms" (0 to 10000). Width accepts 320 to 1920, height 240 to 1080, and "format" is "png", "jpeg", or "webp".
# capture one component after waiting for it to settle
-d '{
"url": "https://example.com/dashboard",
"selector": "#chart-container",
"delay_ms": 800,
"format": "png",
"width": 1280,
"height": 720
}'
Which to use
Use Playwright directly when screenshots are part of a test suite that already controls a browser, or when you need precise assertions against local builds that are not reachable from the internet. Use an API when screenshots are a production feature and you would rather ship in an afternoon than build a browser fleet.
For a head-to-head look at Puppeteer's API versus Playwright's for captures, see the Puppeteer screenshot guide. If you are picking a hosted screenshot service, the screenshot API comparison covers the trade-offs honestly.
Originally published on the Grabbit blog.
Top comments (0)