跳至主要内容

评估 JavaScript

简介

Playwright 脚本在您的 Playwright 环境中运行。您的页面脚本在浏览器页面环境中运行。这些环境不会交叉,它们在不同的虚拟机、不同的进程中运行,甚至可能在不同的计算机上运行。

page.evaluate() API 可以在网页的上下文中运行 JavaScript 函数,并将结果带回 Playwright 环境。浏览器全局变量(如 windowdocument)可以在 evaluate 中使用。

const href = await page.evaluate(() => document.location.href);

如果结果是 Promise 或函数是异步的,evaluate 将自动等待其解析

const status = await page.evaluate(async () => {
const response = await fetch(location.href);
return response.status;
});

不同的环境

评估的脚本在浏览器环境中运行,而您的测试在测试环境中运行。这意味着您不能在页面中使用测试中的变量,反之亦然。相反,您应该将它们显式地作为参数传递。

以下代码段是错误的,因为它直接使用了变量

const data = 'some data';
const result = await page.evaluate(() => {
// WRONG: there is no "data" in the web page.
window.myApp.use(data);
});

以下代码段是正确的,因为它将值显式地作为参数传递

const data = 'some data';
// Pass |data| as a parameter.
const result = await page.evaluate(data => {
window.myApp.use(data);
}, data);

评估参数

Playwright 评估方法(如 page.evaluate())采用单个可选参数。此参数可以是 可序列化 值和 JSHandle 实例的混合。句柄会自动转换为它们表示的值。

// A primitive value.
await page.evaluate(num => num, 42);

// An array.
await page.evaluate(array => array.length, [1, 2, 3]);

// An object.
await page.evaluate(object => object.foo, { foo: 'bar' });

// A single handle.
const button = await page.evaluateHandle('window.button');
await page.evaluate(button => button.textContent, button);

// Alternative notation using JSHandle.evaluate.
await button.evaluate((button, from) => button.textContent.substring(from), 5);

// Object with multiple handles.
const button1 = await page.evaluateHandle('window.button1');
const button2 = await page.evaluateHandle('window.button2');
await page.evaluate(
o => o.button1.textContent + o.button2.textContent,
{ button1, button2 });

// Object destructuring works. Note that property names must match
// between the destructured object and the argument.
// Also note the required parenthesis.
await page.evaluate(
({ button1, button2 }) => button1.textContent + button2.textContent,
{ button1, button2 });

// Array works as well. Arbitrary names can be used for destructuring.
// Note the required parenthesis.
await page.evaluate(
([b1, b2]) => b1.textContent + b2.textContent,
[button1, button2]);

// Any mix of serializables and handles works.
await page.evaluate(
x => x.button1.textContent + x.list[0].textContent + String(x.foo),
{ button1, list: [button2], foo: null });

初始化脚本

有时在页面开始加载之前在页面中评估某些内容会很方便。例如,您可能希望设置一些模拟或测试数据。

在这种情况下,请使用 page.addInitScript()browserContext.addInitScript()。在下面的示例中,我们将用常量值替换 Math.random()

首先,创建一个包含模拟的 preload.js 文件。

// preload.js
Math.random = () => 42;

接下来,将初始化脚本添加到页面。

import { test, expect } from '@playwright/test';
import path from 'path';

test.beforeEach(async ({ page }) => {
// Add script for every test in the beforeEach hook.
// Make sure to correctly resolve the script path.
await page.addInitScript({ path: path.resolve(__dirname, '../mocks/preload.js') });
});

或者,您可以传递函数而不是创建预加载脚本文件。这对于简短或一次性脚本更方便。您也可以通过这种方式传递参数。

import { test, expect } from '@playwright/test';

// Add script for every test in the beforeEach hook.
test.beforeEach(async ({ page }) => {
const value = 42;
await page.addInitScript(value => {
Math.random = () => value;
}, value);
});