From be32317638c0920e263c5c0b411c4aae12997f20 Mon Sep 17 00:00:00 2001 From: Vladimir Date: Sun, 30 Jun 2024 19:12:32 +0200 Subject: [PATCH] feat: add an option to return base64 from page.screenshot (#5993) --- docs/guide/browser.md | 10 ++++-- eslint.config.js | 1 + packages/browser/context.d.ts | 14 ++++++-- .../browser/src/node/commands/screenshot.ts | 36 ++++++++++++------- test/browser/test/dom.test.ts | 36 ++++++++++++++++--- 5 files changed, 74 insertions(+), 23 deletions(-) diff --git a/docs/guide/browser.md b/docs/guide/browser.md index 7cf1838181c9..7dc57bf9c04c 100644 --- a/docs/guide/browser.md +++ b/docs/guide/browser.md @@ -465,12 +465,16 @@ export const page: { /** * Change the size of iframe's viewport. */ - viewport: (width: number | string, height: number | string) => Promise + viewport(width: number | string, height: number | string): Promise /** * Make a screenshot of the test iframe or a specific element. - * @returns Path to the screenshot file. + * @returns Path to the screenshot file or path and base64. */ - screenshot: (options?: ScreenshotOptions) => Promise + screenshot(options: Omit & { base64: true }): Promise<{ + path: string + base64: string + }> + screenshot(options?: ScreenshotOptions): Promise } export const cdp: () => CDPSession diff --git a/eslint.config.js b/eslint.config.js index 1034d607f7e5..cba65838bba2 100644 --- a/eslint.config.js +++ b/eslint.config.js @@ -100,6 +100,7 @@ export default antfu( 'import/newline-after-import': 'off', 'import/first': 'off', 'unused-imports/no-unused-imports': 'off', + 'ts/method-signature-style': 'off', }, }, { diff --git a/packages/browser/context.d.ts b/packages/browser/context.d.ts index 66b6d0cfdcec..b6da5aa83c3d 100644 --- a/packages/browser/context.d.ts +++ b/packages/browser/context.d.ts @@ -29,6 +29,10 @@ export interface ScreenshotOptions { * Path relative to the `screenshotDirectory` in the test config. */ path?: string + /** + * Will also return the base64 encoded screenshot alongside the path. + */ + base64?: boolean } export interface BrowserCommands { @@ -245,12 +249,16 @@ export interface BrowserPage { /** * Change the size of iframe's viewport. */ - viewport: (width: number, height: number) => Promise + viewport(width: number, height: number): Promise /** * Make a screenshot of the test iframe or a specific element. - * @returns Path to the screenshot file. + * @returns Path to the screenshot file or path and base64. */ - screenshot: (options?: ScreenshotOptions) => Promise + screenshot(options: Omit & { base64: true }): Promise<{ + path: string + base64: string + }> + screenshot(options?: ScreenshotOptions): Promise } export const page: BrowserPage diff --git a/packages/browser/src/node/commands/screenshot.ts b/packages/browser/src/node/commands/screenshot.ts index a68bd6cfae5a..211e12f13f92 100644 --- a/packages/browser/src/node/commands/screenshot.ts +++ b/packages/browser/src/node/commands/screenshot.ts @@ -29,28 +29,29 @@ export const screenshot: BrowserCommand<[string, ScreenshotOptions]> = async ( if (options.element) { const { element: elementXpath, ...config } = options const element = context.iframe.locator(`xpath=${elementXpath}`) - await element.screenshot({ ...config, path: savePath }) + const buffer = await element.screenshot({ ...config, path: savePath }) + return returnResult(options, path, buffer) } - else { - await context.iframe.locator('body').screenshot({ - ...options, - path: savePath, - }) - } - return path + + const buffer = await context.iframe.locator('body').screenshot({ + ...options, + path: savePath, + }) + return returnResult(options, path, buffer) } if (context.provider instanceof WebdriverBrowserProvider) { const page = context.provider.browser! if (!options.element) { const body = await page.$('body') - await body.saveScreenshot(savePath) - return path + const buffer = await body.saveScreenshot(savePath) + return returnResult(options, path, buffer) } + const xpath = `//${options.element}` const element = await page.$(xpath) - await element.saveScreenshot(savePath) - return path + const buffer = await element.saveScreenshot(savePath) + return returnResult(options, path, buffer) } throw new Error( @@ -75,3 +76,14 @@ function resolveScreenshotPath( } return resolve(dir, '__screenshots__', base, name) } + +function returnResult( + options: ScreenshotOptions, + path: string, + buffer: Buffer, +) { + if (options.base64) { + return { path, base64: buffer.toString('base64') } + } + return path +} diff --git a/test/browser/test/dom.test.ts b/test/browser/test/dom.test.ts index d955883974bb..c906600ed28c 100644 --- a/test/browser/test/dom.test.ts +++ b/test/browser/test/dom.test.ts @@ -1,16 +1,19 @@ -import { describe, expect, test } from 'vitest' +import { beforeEach, describe, expect, test } from 'vitest' import { page } from '@vitest/browser/context' import { createNode } from '#src/createNode' import '../src/button.css' describe('dom related activity', () => { - test('renders div', async () => { + beforeEach(() => { document.body.style.background = '#f3f3f3' - const wrapper = document.createElement('div') - wrapper.className = 'wrapper' - document.body.appendChild(wrapper) + document.body.replaceChildren() + }) + + test('renders div', async () => { + const wrapper = createWrapper() const div = createNode() wrapper.appendChild(div) + await expect.element(div).toHaveTextContent('Hello World!') const screenshotPath = await page.screenshot({ element: wrapper, @@ -19,4 +22,27 @@ describe('dom related activity', () => { /__screenshots__\/dom.test.ts\/dom-related-activity-renders-div-1.png/, ) }) + + test('resolves base64 screenshot', async () => { + const wrapper = createWrapper() + const div = createNode() + wrapper.appendChild(div) + + const { path, base64 } = await page.screenshot({ + element: wrapper, + base64: true, + }) + expect(path).toMatch( + /__screenshots__\/dom.test.ts\/dom-related-activity-resolves-base64-screenshot-1.png/, + ) + expect(base64).toBeTypeOf('string') + }) }) + +function createWrapper() { + document.body.style.background = '#f3f3f3' + const wrapper = document.createElement('div') + wrapper.className = 'wrapper' + document.body.appendChild(wrapper) + return wrapper +}