Interactivity API
Vitest 使用 Chrome DevTools Protocol 或 webdriver API 实现了 @testing-library/user-event
应用程序接口的子集,而不是伪造事件,这使得浏览器行为更加可靠和一致。
import { userEvent } from '@vitest/browser/context'
await userEvent.click(document.querySelector('.button'))
几乎所有的 userEvent
方法都继承了其提供者的选项。要在我们的 IDE 中查看所有可用选项,请根据我们的提供者,将 webdriver
或 playwright
类型添加到我们的设置文件或配置文件中(取决于我们的 tsconfig.json
中 included
部分包含的内容):
/// <reference types="@vitest/browser/providers/playwright" />
/// <reference types="@vitest/browser/providers/webdriverio" />
userEvent.setup
function setup(): UserEvent
创建一个新的用户事件实例。如果需要保持键盘状态,以便正确按下和释放按钮,这将非常有用。
WARNING
与 @testing-library/user-event
不同,来自 @vitest/browser/context
的默认 userEvent
实例只创建一次,而不是每次调用其方法时都创建一次!您可以从本代码段中看到其工作方式的不同之处:
import { userEvent as vitestUserEvent } from '@vitest/browser/context'
import { userEvent as originalUserEvent } from '@testing-library/user-event'
await vitestUserEvent.keyboard('{Shift}') // press shift without releasing
await vitestUserEvent.keyboard('{/Shift}') // releases shift
await originalUserEvent.keyboard('{Shift}') // press shift without releasing
await originalUserEvent.keyboard('{/Shift}') // DID NOT release shift because the state is different
这种行为更有用,因为我们并没有模拟键盘,而是实际按下了 Shift 键,所以保留原来的行为会在字段中键入时造成意想不到的问题。
userEvent.click
function click(
element: Element | Locator,
options?: UserEventClickOptions,
): Promise<void>
点击元素。继承 provider 的选项。有关此方法如何工作的详细说明,请参阅 provider 的文档。
import { page, userEvent } from '@vitest/browser/context'
test('clicks on an element', async () => {
const logo = page.getByRole('img', { name: /logo/ })
await userEvent.click(logo)
// or you can access it directly on the locator
await logo.click()
})
References:
userEvent.dblClick
function dblClick(
element: Element | Locator,
options?: UserEventDoubleClickOptions,
): Promise<void>
触发元素的双击事件
请参阅你的 provider 的文档以获取有关此方法如何工作的详细说明。
import { page, userEvent } from '@vitest/browser/context'
test('triggers a double click on an element', async () => {
const logo = page.getByRole('img', { name: /logo/ })
await userEvent.dblClick(logo)
// or you can access it directly on the locator
await logo.dblClick()
})
References:
userEvent.tripleClick
function tripleClick(
element: Element | Locator,
options?: UserEventTripleClickOptions,
): Promise<void>
在元素上触发三连击事件。由于浏览器 API 中没有 tripleclick
,此方法会连续触发三次点击事件,因此你必须检查 点击事件的 detail 属性 来过滤事件:evt.detail === 3
。
请参阅你的提供商文档以获取有关此方法工作原理的详细说明。
import { page, userEvent } from '@vitest/browser/context'
test('triggers a triple click on an element', async () => {
const logo = page.getByRole('img', { name: /logo/ })
let tripleClickFired = false
logo.addEventListener('click', (evt) => {
if (evt.detail === 3) {
tripleClickFired = true
}
})
await userEvent.tripleClick(logo)
// or you can access it directly on the locator
await logo.tripleClick()
expect(tripleClickFired).toBe(true)
})
相关链接:
- Playwright
locator.click
API:通过click
方法并设置clickCount: 3
来实现。 - WebdriverIO
browser.action
API:通过动作 API 实现,包含move
操作加上连续的三个down + up + pause
事件。 - testing-library
tripleClick
API:通过tripleClick
方法实现。
userEvent.fill
function fill(
element: Element | Locator,
text: string,
): Promise<void>
为 input/textarea/conteneditable
字段设置值。这将在设置新值前移除输入中的任何现有文本。
import { page, userEvent } from '@vitest/browser/context'
test('update input', async () => {
const input = page.getByRole('input')
await userEvent.fill(input, 'foo') // input.value == foo
await userEvent.fill(input, '{{a[[') // input.value == {{a[[
await userEvent.fill(input, '{Shift}') // input.value == {Shift}
// or you can access it directly on the locator
await input.fill('foo') // input.value == foo
})
该方法聚焦元素、填充元素并在填充后触发一个 input
事件。您可以使用空字符串来清除字段。
TIP
该 API 比使用 userEvent.type
或 userEvent.keyboard
更快,但不支持 user-event keyboard
syntax (例如,{Shift}{selectall}
)。
在不需要输入特殊字符或对按键事件进行细粒度控制的情况下,我们建议使用此 API 而不是 userEvent.type
。
相关链接:
userEvent.keyboard
function keyboard(text: string): Promise<void>
通过 userEvent.keyboard
可以触发键盘输入。如果任何输入有焦点,它就会在该输入中键入字符。否则,它将触发当前焦点元素(如果没有焦点元素,则为 document.body
)上的键盘事件。
此 API 支持 user-event keyboard
语法。
import { userEvent } from '@vitest/browser/context'
test('trigger keystrokes', async () => {
await userEvent.keyboard('foo') // translates to: f, o, o
await userEvent.keyboard('{{a[[') // translates to: {, a, [
await userEvent.keyboard('{Shift}{f}{o}{o}') // translates to: Shift, f, o, o
await userEvent.keyboard('{a>5}') // press a without releasing it and trigger 5 keydown
await userEvent.keyboard('{a>5/}') // press a for 5 keydown and then release it
})
相关链接:
userEvent.tab
function tab(options?: UserEventTabOptions): Promise<void>
发送一个 Tab
键事件。这是userEvent.keyboard('{tab}')
的简写。
import { page, userEvent } from '@vitest/browser/context'
test('tab works', async () => {
const [input1, input2] = page.getByRole('input').elements()
expect(input1).toHaveFocus()
await userEvent.tab()
expect(input2).toHaveFocus()
await userEvent.tab({ shift: true })
expect(input1).toHaveFocus()
})
相关链接:
userEvent.type
function type(
element: Element | Locator,
text: string,
options?: UserEventTypeOptions,
): Promise<void>
WARNING
如果不依赖 special characters(例如,{shift}
或 {selectall}
),建议使用 userEvent.fill
。
type
方法在 keyboard
API 的基础上实现了 @testing-library/user-event
的 type
工具。
该函数允许您在 input/textarea/conteneditable 中键入字符。它支持 user-event keyboard
syntax。
如果只需按下字符而无需输入,请使用 userEvent.keyboard
API。
import { page, userEvent } from '@vitest/browser/context'
test('update input', async () => {
const input = page.getByRole('input')
await userEvent.type(input, 'foo') // input.value == foo
await userEvent.type(input, '{{a[[') // input.value == foo{a[
await userEvent.type(input, '{Shift}') // input.value == foo{a[
})
INFO
Vitest 没有像 input.type
那样在定位器上公开 .type
方法,因为它的存在只是为了与 userEvent
库兼容。请考虑使用 .fill
,因为它更快。
相关链接:
userEvent.clear
function clear(element: Element | Locator): Promise<void>
此方法会清除输入元素的内容。
import { page, userEvent } from '@vitest/browser/context'
test('clears input', async () => {
const input = page.getByRole('input')
await userEvent.fill(input, 'foo')
expect(input).toHaveValue('foo')
await userEvent.clear(input)
// or you can access it directly on the locator
await input.clear()
expect(input).toHaveValue('')
})
相关链接:
userEvent.selectOptions
function selectOptions(
element: Element | Locator,
values:
| HTMLElement
| HTMLElement[]
| Locator
| Locator[]
| string
| string[],
options?: UserEventSelectOptions,
): Promise<void>
The userEvent.selectOptions
allows selecting a value in a <select>
element.
WARNING
如果 select 元素没有 multiple
属性,Vitest 将只选择数组中的第一个元素。
与 @testing-library
不同,Vitest 目前不支持 listbox,但我们计划在将来添加对它的支持。
import { page, userEvent } from '@vitest/browser/context'
test('clears input', async () => {
const select = page.getByRole('select')
await userEvent.selectOptions(select, 'Option 1')
// or you can access it directly on the locator
await select.selectOptions('Option 1')
expect(select).toHaveValue('option-1')
await userEvent.selectOptions(select, 'option-1')
expect(select).toHaveValue('option-1')
await userEvent.selectOptions(select, [
page.getByRole('option', { name: 'Option 1' }),
page.getByRole('option', { name: 'Option 2' }),
])
expect(select).toHaveValue(['option-1', 'option-2'])
})
WARNING
webdriverio
provider 不支持选择多个元素,因为它不提供选择多个元素的 API。
相关链接:
- Playwright
locator.selectOption
API - WebdriverIO
element.selectByIndex
API - testing-library
selectOptions
API
userEvent.hover
function hover(
element: Element | Locator,
options?: UserEventHoverOptions,
): Promise<void>
该方法将光标位置移动到所选元素上。有关此方法如何工作的详细说明,请参阅 provider 的文档。
WARNING
如果使用的是 webdriverio
provider,光标默认会移动到元素的中心。
如果使用的是 playwright
provider,光标会移动到元素的某个可见点。
import { page, userEvent } from '@vitest/browser/context'
test('hovers logo element', async () => {
const logo = page.getByRole('img', { name: /logo/ })
await userEvent.hover(logo)
// or you can access it directly on the locator
await logo.hover()
})
相关链接:
userEvent.unhover
function unhover(
element: Element | Locator,
options?: UserEventHoverOptions,
): Promise<void>
其作用与 userEvent.hover
相同,但会将光标移至 document.body
元素。
WARNING
默认情况下,光标位置位于 body 元素的 "某个" 可见位置(在 playwright
provider中)或中心位置(在 webdriverio
provider中),因此如果当前悬停的元素已经位于相同位置,本方法将不起作用。
import { page, userEvent } from '@vitest/browser/context'
test('unhover logo element', async () => {
const logo = page.getByRole('img', { name: /logo/ })
await userEvent.unhover(logo)
// or you can access it directly on the locator
await logo.unhover()
})
相关链接:
userEvent.upload
function upload(
element: Element | Locator,
files: string[] | string | File[] | File,
): Promise<void>
更改文件输入元素,使其包含指定文件。
import { page, userEvent } from '@vitest/browser/context'
test('can upload a file', async () => {
const input = page.getByRole('button', { name: /Upload files/ })
const file = new File(['file'], 'file.png', { type: 'image/png' })
await userEvent.upload(input, file)
// or you can access it directly on the locator
await input.upload(file)
// you can also use file paths relative to the test file
await userEvent.upload(input, '../fixtures/file.png')
})
WARNING
webdriverio
provider 仅在 chrome
和 edge
浏览器中支持该命令。目前也只支持字符串类型。
相关链接:
userEvent.dragAndDrop
function dragAndDrop(
source: Element | Locator,
target: Element | Locator,
options?: UserEventDragAndDropOptions,
): Promise<void>
将源元素拖到目标元素的顶部。不要忘记,源元素的draggable
属性必须设置为 true
。
import { page, userEvent } from '@vitest/browser/context'
test('drag and drop works', async () => {
const source = page.getByRole('img', { name: /logo/ })
const target = page.getByTestId('logo-target')
await userEvent.dragAndDrop(source, target)
// or you can access it directly on the locator
await source.dropTo(target)
await expect.element(target).toHaveTextContent('Logo is processed')
})
WARNING
preview
provider不支持此 API。
相关链接:
userEvent.copy
function copy(): Promise<void>
将选中的文本复制到剪贴板。
import { page, userEvent } from '@vitest/browser/context'
test('copy and paste', async () => {
// write to 'source'
await userEvent.click(page.getByPlaceholder('source'))
await userEvent.keyboard('hello')
// select and copy 'source'
await userEvent.dblClick(page.getByPlaceholder('source'))
await userEvent.copy()
// paste to 'target'
await userEvent.click(page.getByPlaceholder('target'))
await userEvent.paste()
await expect.element(page.getByPlaceholder('source')).toHaveTextContent('hello')
await expect.element(page.getByPlaceholder('target')).toHaveTextContent('hello')
})
相关链接:
userEvent.cut
function cut(): Promise<void>
将选中的文本剪切到剪贴板。
import { page, userEvent } from '@vitest/browser/context'
test('copy and paste', async () => {
// write to 'source'
await userEvent.click(page.getByPlaceholder('source'))
await userEvent.keyboard('hello')
// select and cut 'source'
await userEvent.dblClick(page.getByPlaceholder('source'))
await userEvent.cut()
// paste to 'target'
await userEvent.click(page.getByPlaceholder('target'))
await userEvent.paste()
await expect.element(page.getByPlaceholder('source')).toHaveTextContent('')
await expect.element(page.getByPlaceholder('target')).toHaveTextContent('hello')
})
相关链接:
userEvent.paste
function paste(): Promise<void>
将文本从剪贴板粘贴。请参阅 userEvent.copy
和 userEvent.cut
以获取使用示例。
相关链接: