playwright-bdd-step-definitions
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
ChinesePlaywright BDD Step Definitions
Playwright BDD 步骤定义
Expert knowledge of creating step definitions for Playwright BDD, including step functions, parameter types, fixtures, and Page Object Model integration.
精通Playwright BDD的步骤定义创建,包括步骤函数、参数类型、fixture以及Page Object Model的集成。
Overview
概述
Step definitions connect Gherkin steps in feature files to executable code. Playwright BDD uses to generate type-safe step definition functions that integrate with Playwright's fixtures and assertions.
createBdd()步骤定义用于将功能文件中的Gherkin步骤与可执行代码关联起来。Playwright BDD使用生成类型安全的步骤定义函数,这些函数可与Playwright的fixture和断言集成。
createBdd()Basic Step Definitions
基础步骤定义
Creating Step Functions
创建步骤函数
typescript
// steps/common.steps.ts
import { createBdd } from 'playwright-bdd';
const { Given, When, Then } = createBdd();
Given('I am on the home page', async ({ page }) => {
await page.goto('/');
});
When('I click the login button', async ({ page }) => {
await page.getByRole('button', { name: 'Login' }).click();
});
Then('I should see the dashboard', async ({ page }) => {
await expect(page.getByRole('heading', { name: 'Dashboard' })).toBeVisible();
});typescript
// steps/common.steps.ts
import { createBdd } from 'playwright-bdd';
const { Given, When, Then } = createBdd();
Given('I am on the home page', async ({ page }) => {
await page.goto('/');
});
When('I click the login button', async ({ page }) => {
await page.getByRole('button', { name: 'Login' }).click();
});
Then('I should see the dashboard', async ({ page }) => {
await expect(page.getByRole('heading', { name: 'Dashboard' })).toBeVisible();
});Step with Playwright Fixtures
结合Playwright Fixture的步骤
All Playwright fixtures are available in step definitions:
typescript
import { createBdd } from 'playwright-bdd';
const { Given, When, Then } = createBdd();
Given('I am logged in as {string}', async ({ page, context }, username: string) => {
// Access page and context fixtures
await page.goto('/login');
await page.getByLabel('Username').fill(username);
await page.getByRole('button', { name: 'Login' }).click();
});
When('I take a screenshot', async ({ page }) => {
await page.screenshot({ path: 'screenshot.png' });
});
Then('the browser has {int} cookies', async ({ context }, count: number) => {
const cookies = await context.cookies();
expect(cookies).toHaveLength(count);
});所有Playwright fixture都可在步骤定义中使用:
typescript
import { createBdd } from 'playwright-bdd';
const { Given, When, Then } = createBdd();
Given('I am logged in as {string}', async ({ page, context }, username: string) => {
// 访问page和context fixture
await page.goto('/login');
await page.getByLabel('Username').fill(username);
await page.getByRole('button', { name: 'Login' }).click();
});
When('I take a screenshot', async ({ page }) => {
await page.screenshot({ path: 'screenshot.png' });
});
Then('the browser has {int} cookies', async ({ context }, count: number) => {
const cookies = await context.cookies();
expect(cookies).toHaveLength(count);
});Parameter Types
参数类型
Built-in Parameters
内置参数
typescript
// String parameter: {string}
Given('the user name is {string}', async ({}, name: string) => {
console.log(name); // "John"
});
// Integer parameter: {int}
When('I wait {int} seconds', async ({}, seconds: number) => {
await page.waitForTimeout(seconds * 1000);
});
// Float parameter: {float}
Then('the price is {float}', async ({}, price: number) => {
console.log(price); // 19.99
});
// Word parameter: {word}
Given('I am on the {word} page', async ({ page }, pageName: string) => {
await page.goto(`/${pageName}`);
});typescript
// 字符串参数: {string}
Given('the user name is {string}', async ({}, name: string) => {
console.log(name); // "John"
});
// 整数参数: {int}
When('I wait {int} seconds', async ({}, seconds: number) => {
await page.waitForTimeout(seconds * 1000);
});
// 浮点数参数: {float}
Then('the price is {float}', async ({}, price: number) => {
console.log(price); // 19.99
});
// 单词参数: {word}
Given('I am on the {word} page', async ({ page }, pageName: string) => {
await page.goto(`/${pageName}`);
});Anonymous Parameters
匿名参数
Use regex capture groups for flexible matching:
typescript
// Match any text in quotes
Given(/^I enter "(.*)" in the search box$/, async ({ page }, query: string) => {
await page.getByRole('searchbox').fill(query);
});
// Match numbers
When(/^I add (\d+) items to cart$/, async ({ page }, count: string) => {
const quantity = parseInt(count, 10);
for (let i = 0; i < quantity; i++) {
await page.getByRole('button', { name: 'Add to Cart' }).click();
}
});使用正则捕获组实现灵活匹配:
typescript
// 匹配引号中的任意文本
Given(/^I enter "(.*)" in the search box$/, async ({ page }, query: string) => {
await page.getByRole('searchbox').fill(query);
});
// 匹配数字
When(/^I add (\d+) items to cart$/, async ({ page }, count: string) => {
const quantity = parseInt(count, 10);
for (let i = 0; i < quantity; i++) {
await page.getByRole('button', { name: 'Add to Cart' }).click();
}
});Custom Parameter Types
自定义参数类型
Define reusable parameter types:
typescript
// steps/parameters.ts
import { defineParameterType } from 'playwright-bdd';
defineParameterType({
name: 'color',
regexp: /red|green|blue/,
transformer: (s) => s,
});
defineParameterType({
name: 'boolean',
regexp: /true|false/,
transformer: (s) => s === 'true',
});
defineParameterType({
name: 'date',
regexp: /\d{4}-\d{2}-\d{2}/,
transformer: (s) => new Date(s),
});Using custom parameters:
typescript
// steps/ui.steps.ts
import { createBdd } from 'playwright-bdd';
import './parameters'; // Import parameter definitions
const { Given, When, Then } = createBdd();
When('I select the {color} theme', async ({ page }, color: string) => {
await page.getByRole('button', { name: color }).click();
});
Then('dark mode is {boolean}', async ({ page }, enabled: boolean) => {
if (enabled) {
await expect(page.locator('body')).toHaveClass(/dark/);
}
});定义可复用的参数类型:
typescript
// steps/parameters.ts
import { defineParameterType } from 'playwright-bdd';
defineParameterType({
name: 'color',
regexp: /red|green|blue/,
transformer: (s) => s,
});
defineParameterType({
name: 'boolean',
regexp: /true|false/,
transformer: (s) => s === 'true',
});
defineParameterType({
name: 'date',
regexp: /\d{4}-\d{2}-\d{2}/,
transformer: (s) => new Date(s),
});使用自定义参数:
typescript
// steps/ui.steps.ts
import { createBdd } from 'playwright-bdd';
import './parameters'; // 导入参数定义
const { Given, When, Then } = createBdd();
When('I select the {color} theme', async ({ page }, color: string) => {
await page.getByRole('button', { name: color }).click();
});
Then('dark mode is {boolean}', async ({ page }, enabled: boolean) => {
if (enabled) {
await expect(page.locator('body')).toHaveClass(/dark/);
}
});Custom Fixtures
自定义Fixture
Creating Custom Fixtures
创建自定义Fixture
typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
// Define fixture types
type TestFixtures = {
todoPage: TodoPage;
apiClient: ApiClient;
};
// Extend base test with fixtures
export const test = base.extend<TestFixtures>({
todoPage: async ({ page }, use) => {
const todoPage = new TodoPage(page);
await use(todoPage);
},
apiClient: async ({ request }, use) => {
const client = new ApiClient(request);
await use(client);
},
});
// Create BDD functions with custom test
export const { Given, When, Then } = createBdd(test);typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
// 定义Fixture类型
type TestFixtures = {
todoPage: TodoPage;
apiClient: ApiClient;
};
// 扩展基础测试以添加fixture
export const test = base.extend<TestFixtures>({
todoPage: async ({ page }, use) => {
const todoPage = new TodoPage(page);
await use(todoPage);
},
apiClient: async ({ request }, use) => {
const client = new ApiClient(request);
await use(client);
},
});
// 使用自定义测试创建BDD函数
export const { Given, When, Then } = createBdd(test);Using Custom Fixtures in Steps
在步骤中使用自定义Fixture
typescript
// steps/todo.steps.ts
import { Given, When, Then } from './fixtures';
Given('I have an empty todo list', async ({ todoPage }) => {
await todoPage.goto();
await todoPage.clearAll();
});
When('I add a todo {string}', async ({ todoPage }, text: string) => {
await todoPage.addTodo(text);
});
Then('I should see {int} todos', async ({ todoPage }, count: number) => {
await todoPage.expectTodoCount(count);
});typescript
// steps/todo.steps.ts
import { Given, When, Then } from './fixtures';
Given('I have an empty todo list', async ({ todoPage }) => {
await todoPage.goto();
await todoPage.clearAll();
});
When('I add a todo {string}', async ({ todoPage }, text: string) => {
await todoPage.addTodo(text);
});
Then('I should see {int} todos', async ({ todoPage }, count: number) => {
await todoPage.expectTodoCount(count);
});Fixture with Setup and Teardown
带前置和后置操作的Fixture
typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
export const test = base.extend<{
authenticatedPage: Page;
}>({
authenticatedPage: async ({ page, context }, use) => {
// Setup: Login before test
await page.goto('/login');
await page.getByLabel('Email').fill('test@example.com');
await page.getByLabel('Password').fill('password');
await page.getByRole('button', { name: 'Login' }).click();
await page.waitForURL('/dashboard');
// Use the authenticated page
await use(page);
// Teardown: Logout after test
await page.goto('/logout');
},
});
export const { Given, When, Then } = createBdd(test);typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
export const test = base.extend<{
authenticatedPage: Page;
}>({
authenticatedPage: async ({ page, context }, use) => {
// 前置操作:测试前登录
await page.goto('/login');
await page.getByLabel('Email').fill('test@example.com');
await page.getByLabel('Password').fill('password');
await page.getByRole('button', { name: 'Login' }).click();
await page.waitForURL('/dashboard');
// 使用已认证的页面
await use(page);
// 后置操作:测试后登出
await page.goto('/logout');
},
});
export const { Given, When, Then } = createBdd(test);Page Object Model
Page Object Model
Page Object Definition
Page Object定义
typescript
// pages/TodoPage.ts
import { Page, Locator, expect } from '@playwright/test';
export class TodoPage {
readonly page: Page;
readonly input: Locator;
readonly list: Locator;
readonly items: Locator;
constructor(page: Page) {
this.page = page;
this.input = page.getByPlaceholder('What needs to be done?');
this.list = page.getByRole('list');
this.items = page.getByTestId('todo-item');
}
async goto() {
await this.page.goto('/todos');
}
async addTodo(text: string) {
await this.input.fill(text);
await this.input.press('Enter');
}
async removeTodo(text: string) {
const item = this.items.filter({ hasText: text });
await item.hover();
await item.getByRole('button', { name: 'Delete' }).click();
}
async toggleTodo(text: string) {
const item = this.items.filter({ hasText: text });
await item.getByRole('checkbox').click();
}
async expectTodoCount(count: number) {
await expect(this.items).toHaveCount(count);
}
async expectTodoVisible(text: string) {
await expect(this.items.filter({ hasText: text })).toBeVisible();
}
async clearAll() {
const count = await this.items.count();
for (let i = count - 1; i >= 0; i--) {
await this.items.nth(i).hover();
await this.items.nth(i).getByRole('button', { name: 'Delete' }).click();
}
}
}typescript
// pages/TodoPage.ts
import { Page, Locator, expect } from '@playwright/test';
export class TodoPage {
readonly page: Page;
readonly input: Locator;
readonly list: Locator;
readonly items: Locator;
constructor(page: Page) {
this.page = page;
this.input = page.getByPlaceholder('What needs to be done?');
this.list = page.getByRole('list');
this.items = page.getByTestId('todo-item');
}
async goto() {
await this.page.goto('/todos');
}
async addTodo(text: string) {
await this.input.fill(text);
await this.input.press('Enter');
}
async removeTodo(text: string) {
const item = this.items.filter({ hasText: text });
await item.hover();
await item.getByRole('button', { name: 'Delete' }).click();
}
async toggleTodo(text: string) {
const item = this.items.filter({ hasText: text });
await item.getByRole('checkbox').click();
}
async expectTodoCount(count: number) {
await expect(this.items).toHaveCount(count);
}
async expectTodoVisible(text: string) {
await expect(this.items.filter({ hasText: text })).toBeVisible();
}
async clearAll() {
const count = await this.items.count();
for (let i = count - 1; i >= 0; i--) {
await this.items.nth(i).hover();
await this.items.nth(i).getByRole('button', { name: 'Delete' }).click();
}
}
}Integrating Page Objects with Steps
将Page Object与步骤集成
typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
import { TodoPage } from '../pages/TodoPage';
import { LoginPage } from '../pages/LoginPage';
export const test = base.extend<{
todoPage: TodoPage;
loginPage: LoginPage;
}>({
todoPage: async ({ page }, use) => {
await use(new TodoPage(page));
},
loginPage: async ({ page }, use) => {
await use(new LoginPage(page));
},
});
export const { Given, When, Then } = createBdd(test);typescript
// steps/todo.steps.ts
import { Given, When, Then } from './fixtures';
Given('I am on the todo page', async ({ todoPage }) => {
await todoPage.goto();
});
When('I add {string} to my todos', async ({ todoPage }, text: string) => {
await todoPage.addTodo(text);
});
When('I complete the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.toggleTodo(text);
});
When('I remove the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.removeTodo(text);
});
Then('I should see the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.expectTodoVisible(text);
});
Then('there should be {int} todos', async ({ todoPage }, count: number) => {
await todoPage.expectTodoCount(count);
});typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
import { TodoPage } from '../pages/TodoPage';
import { LoginPage } from '../pages/LoginPage';
export const test = base.extend<{
todoPage: TodoPage;
loginPage: LoginPage;
}>({
todoPage: async ({ page }, use) => {
await use(new TodoPage(page));
},
loginPage: async ({ page }, use) => {
await use(new LoginPage(page));
},
});
export const { Given, When, Then } = createBdd(test);typescript
// steps/todo.steps.ts
import { Given, When, Then } from './fixtures';
Given('I am on the todo page', async ({ todoPage }) => {
await todoPage.goto();
});
When('I add {string} to my todos', async ({ todoPage }, text: string) => {
await todoPage.addTodo(text);
});
When('I complete the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.toggleTodo(text);
});
When('I remove the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.removeTodo(text);
});
Then('I should see the todo {string}', async ({ todoPage }, text: string) => {
await todoPage.expectTodoVisible(text);
});
Then('there should be {int} todos', async ({ todoPage }, count: number) => {
await todoPage.expectTodoCount(count);
});Decorator-Based Steps
基于装饰器的步骤
Using Decorators with Classes
在类中使用装饰器
typescript
// steps/TodoSteps.ts
import { Fixture, Given, When, Then } from 'playwright-bdd/decorators';
import { test } from './fixtures';
export
@Fixture<typeof test>('todoPage')
class TodoSteps {
@Given('I am on the todo page')
async gotoTodoPage() {
await this.todoPage.goto();
}
@When('I add {string} to my todos')
async addTodo(text: string) {
await this.todoPage.addTodo(text);
}
@Then('I should see {int} todos')
async checkTodoCount(count: number) {
await this.todoPage.expectTodoCount(count);
}
}typescript
// steps/TodoSteps.ts
import { Fixture, Given, When, Then } from 'playwright-bdd/decorators';
import { test } from './fixtures';
export
@Fixture<typeof test>('todoPage')
class TodoSteps {
@Given('I am on the todo page')
async gotoTodoPage() {
await this.todoPage.goto();
}
@When('I add {string} to my todos')
async addTodo(text: string) {
await this.todoPage.addTodo(text);
}
@Then('I should see {int} todos')
async checkTodoCount(count: number) {
await this.todoPage.expectTodoCount(count);
}
}Multiple Fixtures in Decorators
装饰器中的多Fixture
typescript
// steps/AuthenticatedSteps.ts
import { Fixture, Given, When, Then } from 'playwright-bdd/decorators';
import { test } from './fixtures';
export
@Fixture<typeof test>('loginPage')
@Fixture<typeof test>('todoPage')
class AuthenticatedSteps {
@Given('I am logged in')
async login() {
await this.loginPage.login('user@example.com', 'password');
}
@When('I visit my todos')
async visitTodos() {
await this.todoPage.goto();
}
}typescript
// steps/AuthenticatedSteps.ts
import { Fixture, Given, When, Then } from 'playwright-bdd/decorators';
import { test } from './fixtures';
export
@Fixture<typeof test>('loginPage')
@Fixture<typeof test>('todoPage')
class AuthenticatedSteps {
@Given('I am logged in')
async login() {
await this.loginPage.login('user@example.com', 'password');
}
@When('I visit my todos')
async visitTodos() {
await this.todoPage.goto();
}
}Data Tables
数据表格
Simple Tables
简单表格
gherkin
When I add the following todos:
| Buy milk |
| Buy bread |
| Buy eggs |typescript
import { DataTable } from '@cucumber/cucumber';
When('I add the following todos:', async ({ todoPage }, table: DataTable) => {
const todos = table.raw().flat();
for (const todo of todos) {
await todoPage.addTodo(todo);
}
});gherkin
When I add the following todos:
| Buy milk |
| Buy bread |
| Buy eggs |typescript
import { DataTable } from '@cucumber/cucumber';
When('I add the following todos:', async ({ todoPage }, table: DataTable) => {
const todos = table.raw().flat();
for (const todo of todos) {
await todoPage.addTodo(todo);
}
});Tables with Headers
带表头的表格
gherkin
When I create users:
| name | email | role |
| Alice | alice@test.com | admin |
| Bob | bob@test.com | user |typescript
When('I create users:', async ({ page }, table: DataTable) => {
const users = table.hashes(); // [{ name: 'Alice', email: '...', role: 'admin' }, ...]
for (const user of users) {
await page.getByLabel('Name').fill(user.name);
await page.getByLabel('Email').fill(user.email);
await page.getByLabel('Role').selectOption(user.role);
await page.getByRole('button', { name: 'Create' }).click();
}
});gherkin
When I create users:
| name | email | role |
| Alice | alice@test.com | admin |
| Bob | bob@test.com | user |typescript
When('I create users:', async ({ page }, table: DataTable) => {
const users = table.hashes(); // [{ name: 'Alice', email: '...', role: 'admin' }, ...]
for (const user of users) {
await page.getByLabel('Name').fill(user.name);
await page.getByLabel('Email').fill(user.email);
await page.getByLabel('Role').selectOption(user.role);
await page.getByRole('button', { name: 'Create' }).click();
}
});Vertical Tables
垂直表格
gherkin
When I fill the form:
| Name | John Doe |
| Email | john@example.com|
| Password | secret123 |typescript
When('I fill the form:', async ({ page }, table: DataTable) => {
const data = table.rowsHash(); // { Name: 'John Doe', Email: '...', Password: '...' }
await page.getByLabel('Name').fill(data.Name);
await page.getByLabel('Email').fill(data.Email);
await page.getByLabel('Password').fill(data.Password);
});gherkin
When I fill the form:
| Name | John Doe |
| Email | john@example.com|
| Password | secret123 |typescript
When('I fill the form:', async ({ page }, table: DataTable) => {
const data = table.rowsHash(); // { Name: 'John Doe', Email: '...', Password: '...' }
await page.getByLabel('Name').fill(data.Name);
await page.getByLabel('Email').fill(data.Email);
await page.getByLabel('Password').fill(data.Password);
});Doc Strings
文档字符串
Multi-line Text Input
多行文本输入
gherkin
When I write the following review:
"""
This product is amazing!
I highly recommend it to everyone.
Five stars!
"""typescript
When('I write the following review:', async ({ page }, docString: string) => {
await page.getByRole('textbox', { name: 'Review' }).fill(docString);
});gherkin
When I write the following review:
"""
This product is amazing!
I highly recommend it to everyone.
Five stars!
"""typescript
When('I write the following review:', async ({ page }, docString: string) => {
await page.getByRole('textbox', { name: 'Review' }).fill(docString);
});JSON Data
JSON数据
gherkin
When I send the API request:
"""json
{
"name": "Test Product",
"price": 29.99,
"category": "electronics"
}
"""typescript
When('I send the API request:', async ({ request }, docString: string) => {
const data = JSON.parse(docString);
await request.post('/api/products', { data });
});gherkin
When I send the API request:
"""json
{
"name": "Test Product",
"price": 29.99,
"category": "electronics"
}
"""typescript
When('I send the API request:', async ({ request }, docString: string) => {
const data = JSON.parse(docString);
await request.post('/api/products', { data });
});Sharing State Between Steps
在步骤之间共享状态
Using World/Context
使用World/上下文
typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
type World = {
currentUser?: { id: string; name: string };
createdItems: string[];
};
export const test = base.extend<{ world: World }>({
world: async ({}, use) => {
await use({
createdItems: [],
});
},
});
export const { Given, When, Then } = createBdd(test);typescript
// steps/user.steps.ts
import { Given, When, Then } from './fixtures';
Given('I am logged in as {string}', async ({ page, world }, name: string) => {
world.currentUser = { id: '123', name };
await page.goto('/login');
// ... login steps
});
When('I create an item {string}', async ({ world }, item: string) => {
world.createdItems.push(item);
// ... create item
});
Then('I should see my items', async ({ page, world }) => {
for (const item of world.createdItems) {
await expect(page.getByText(item)).toBeVisible();
}
});typescript
// steps/fixtures.ts
import { test as base, createBdd } from 'playwright-bdd';
type World = {
currentUser?: { id: string; name: string };
createdItems: string[];
};
export const test = base.extend<{ world: World }>({
world: async ({}, use) => {
await use({
createdItems: [],
});
},
});
export const { Given, When, Then } = createBdd(test);typescript
// steps/user.steps.ts
import { Given, When, Then } from './fixtures';
Given('I am logged in as {string}', async ({ page, world }, name: string) => {
world.currentUser = { id: '123', name };
await page.goto('/login');
// ... 登录步骤
});
When('I create an item {string}', async ({ world }, item: string) => {
world.createdItems.push(item);
// ... 创建项目
});
Then('I should see my items', async ({ page, world }) => {
for (const item of world.createdItems) {
await expect(page.getByText(item)).toBeVisible();
}
});Error Handling
错误处理
Pending Steps
待实现步骤
typescript
Given('a feature not yet implemented', async ({}) => {
throw new Error('Step not implemented');
});typescript
Given('a feature not yet implemented', async ({}) => {
throw new Error('Step not implemented');
});Skip Steps Conditionally
条件跳过步骤
typescript
Given('I am on a supported browser', async ({ browserName }) => {
if (browserName === 'webkit') {
test.skip(true, 'Feature not supported on WebKit');
}
// Continue with test
});typescript
Given('I am on a supported browser', async ({ browserName }) => {
if (browserName === 'webkit') {
test.skip(true, 'Feature not supported on WebKit');
}
// 继续测试
});Best Practices
最佳实践
Step Reusability
步骤可复用性
Write generic steps that work across multiple scenarios:
typescript
// Generic navigation step
Given('I am on the {string} page', async ({ page }, pageName: string) => {
const routes: Record<string, string> = {
home: '/',
login: '/login',
dashboard: '/dashboard',
settings: '/settings',
};
await page.goto(routes[pageName] || `/${pageName}`);
});
// Generic form fill step
When('I fill in {string} with {string}', async ({ page }, label: string, value: string) => {
await page.getByLabel(label).fill(value);
});
// Generic click step
When('I click {string}', async ({ page }, text: string) => {
await page.getByRole('button', { name: text }).or(
page.getByRole('link', { name: text })
).click();
});编写可在多个场景中使用的通用步骤:
typescript
// 通用导航步骤
Given('I am on the {string} page', async ({ page }, pageName: string) => {
const routes: Record<string, string> = {
home: '/',
login: '/login',
dashboard: '/dashboard',
settings: '/settings',
};
await page.goto(routes[pageName] || `/${pageName}`);
});
// 通用表单填充步骤
When('I fill in {string} with {string}', async ({ page }, label: string, value: string) => {
await page.getByLabel(label).fill(value);
});
// 通用点击步骤
When('I click {string}', async ({ page }, text: string) => {
await page.getByRole('button', { name: text }).or(
page.getByRole('link', { name: text })
).click();
});Step Organization
步骤组织
Organize steps by domain:
steps/
├── fixtures.ts # Shared fixtures
├── parameters.ts # Custom parameter types
├── common/
│ ├── navigation.steps.ts
│ └── forms.steps.ts
├── auth/
│ ├── login.steps.ts
│ └── logout.steps.ts
└── products/
├── catalog.steps.ts
└── cart.steps.ts按领域组织步骤:
steps/
├── fixtures.ts # 共享Fixture
├── parameters.ts # 自定义参数类型
├── common/
│ ├── navigation.steps.ts
│ └── forms.steps.ts
├── auth/
│ ├── login.steps.ts
│ └── logout.steps.ts
└── products/
├── catalog.steps.ts
└── cart.steps.tsAssertions
断言
Use Playwright's expect for assertions:
typescript
import { expect } from '@playwright/test';
Then('I should see {string}', async ({ page }, text: string) => {
await expect(page.getByText(text)).toBeVisible();
});
Then('the page title should be {string}', async ({ page }, title: string) => {
await expect(page).toHaveTitle(title);
});
Then('the URL should contain {string}', async ({ page }, path: string) => {
await expect(page).toHaveURL(new RegExp(path));
});使用Playwright的expect进行断言:
typescript
import { expect } from '@playwright/test';
Then('I should see {string}', async ({ page }, text: string) => {
await expect(page.getByText(text)).toBeVisible();
});
Then('the page title should be {string}', async ({ page }, title: string) => {
await expect(page).toHaveTitle(title);
});
Then('the URL should contain {string}', async ({ page }, path: string) => {
await expect(page).toHaveURL(new RegExp(path));
});When to Use This Skill
何时使用此技能
- Creating new step definitions
- Implementing Page Object Model with BDD
- Setting up custom fixtures
- Using data tables and doc strings
- Sharing state between steps
- Writing reusable generic steps
- Converting Cucumber steps to Playwright BDD
- Troubleshooting step matching issues
- 创建新的步骤定义
- 结合BDD实现Page Object Model
- 设置自定义Fixture
- 使用数据表格和文档字符串
- 在步骤之间共享状态
- 编写可复用的通用步骤
- 将Cucumber步骤转换为Playwright BDD
- 排查步骤匹配问题