mirror of
https://github.com/vitest-dev/vitest.git
synced 2025-12-08 18:26:03 +00:00
600 lines
19 KiB
Markdown
600 lines
19 KiB
Markdown
---
|
|
title: Interactivity API | Browser Mode
|
|
---
|
|
|
|
# Interactivity API
|
|
|
|
Vitest implements a subset of [`@testing-library/user-event`](https://testing-library.com/docs/user-event/intro) APIs using [Chrome DevTools Protocol](https://chromedevtools.github.io/devtools-protocol/) or [webdriver](https://www.w3.org/TR/webdriver/) instead of faking events which makes the browser behaviour more reliable and consistent with how users interact with a page.
|
|
|
|
```ts
|
|
import { userEvent } from '@vitest/browser/context'
|
|
|
|
await userEvent.click(document.querySelector('.button'))
|
|
```
|
|
|
|
Almost every `userEvent` method inherits its provider options. To see all available options in your IDE, add `webdriver` or `playwright` types (depending on your provider) to your [setup file](/config/#setupfile) or a [config file](/config/) (depending on what is in `included` in your `tsconfig.json`):
|
|
|
|
::: code-group
|
|
```ts [playwright]
|
|
/// <reference types="@vitest/browser/providers/playwright" />
|
|
```
|
|
```ts [webdriverio]
|
|
/// <reference types="@vitest/browser/providers/webdriverio" />
|
|
```
|
|
:::
|
|
|
|
## userEvent.setup
|
|
|
|
```ts
|
|
function setup(): UserEvent
|
|
```
|
|
|
|
Creates a new user event instance. This is useful if you need to keep the state of keyboard to press and release buttons correctly.
|
|
|
|
::: warning
|
|
Unlike `@testing-library/user-event`, the default `userEvent` instance from `@vitest/browser/context` is created once, not every time its methods are called! You can see the difference in how it works in this snippet:
|
|
|
|
```ts
|
|
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
|
|
```
|
|
|
|
This behaviour is more useful because we do not emulate the keyboard, we actually press the Shift, so keeping the original behaviour would cause unexpected issues when typing in the field.
|
|
:::
|
|
|
|
## userEvent.click
|
|
|
|
```ts
|
|
function click(
|
|
element: Element | Locator,
|
|
options?: UserEventClickOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
Click on an element. Inherits provider's options. Please refer to your provider's documentation for detailed explanation about how this method works.
|
|
|
|
```ts
|
|
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:
|
|
|
|
- [Playwright `locator.click` API](https://playwright.dev/docs/api/class-locator#locator-click)
|
|
- [WebdriverIO `element.click` API](https://webdriver.io/docs/api/element/click/)
|
|
- [testing-library `click` API](https://testing-library.com/docs/user-event/convenience/#click)
|
|
|
|
## userEvent.dblClick
|
|
|
|
```ts
|
|
function dblClick(
|
|
element: Element | Locator,
|
|
options?: UserEventDoubleClickOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
Triggers a double click event on an element.
|
|
|
|
Please refer to your provider's documentation for detailed explanation about how this method works.
|
|
|
|
```ts
|
|
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:
|
|
|
|
- [Playwright `locator.dblclick` API](https://playwright.dev/docs/api/class-locator#locator-dblclick)
|
|
- [WebdriverIO `element.doubleClick` API](https://webdriver.io/docs/api/element/doubleClick/)
|
|
- [testing-library `dblClick` API](https://testing-library.com/docs/user-event/convenience/#dblClick)
|
|
|
|
## userEvent.tripleClick
|
|
|
|
```ts
|
|
function tripleClick(
|
|
element: Element | Locator,
|
|
options?: UserEventTripleClickOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
Triggers a triple click event on an element. Since there is no `tripleclick` in browser api, this method will fire three click events in a row, and so you must check [click event detail](https://developer.mozilla.org/en-US/docs/Web/API/Element/click_event#usage_notes) to filter the event: `evt.detail === 3`.
|
|
|
|
Please refer to your provider's documentation for detailed explanation about how this method works.
|
|
|
|
```ts
|
|
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)
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.click` API](https://playwright.dev/docs/api/class-locator#locator-click): implemented via `click` with `clickCount: 3` .
|
|
- [WebdriverIO `browser.action` API](https://webdriver.io/docs/api/browser/action/): implemented via actions api with `move` plus three `down + up + pause` events in a row
|
|
- [testing-library `tripleClick` API](https://testing-library.com/docs/user-event/convenience/#tripleClick)
|
|
|
|
## userEvent.fill
|
|
|
|
```ts
|
|
function fill(
|
|
element: Element | Locator,
|
|
text: string,
|
|
): Promise<void>
|
|
```
|
|
|
|
Set a value to the `input`/`textarea`/`contenteditable` field. This will remove any existing text in the input before setting the new value.
|
|
|
|
```ts
|
|
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
|
|
})
|
|
```
|
|
|
|
This methods focuses the element, fills it and triggers an `input` event after filling. You can use an empty string to clear the field.
|
|
|
|
::: tip
|
|
This API is faster than using [`userEvent.type`](#userevent-type) or [`userEvent.keyboard`](#userevent-keyboard), but it **doesn't support** [user-event `keyboard` syntax](https://testing-library.com/docs/user-event/keyboard) (e.g., `{Shift}{selectall}`).
|
|
|
|
We recommend using this API over [`userEvent.type`](#userevent-type) in situations when you don't need to enter special characters or have granular control over keypress events.
|
|
:::
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.fill` API](https://playwright.dev/docs/api/class-locator#locator-fill)
|
|
- [WebdriverIO `element.setValue` API](https://webdriver.io/docs/api/element/setValue)
|
|
- [testing-library `type` API](https://testing-library.com/docs/user-event/utility/#type)
|
|
|
|
## userEvent.keyboard
|
|
|
|
```ts
|
|
function keyboard(text: string): Promise<void>
|
|
```
|
|
|
|
The `userEvent.keyboard` allows you to trigger keyboard strokes. If any input has a focus, it will type characters into that input. Otherwise, it will trigger keyboard events on the currently focused element (`document.body` if there are no focused elements).
|
|
|
|
This API supports [user-event `keyboard` syntax](https://testing-library.com/docs/user-event/keyboard).
|
|
|
|
```ts
|
|
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
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `Keyboard` API](https://playwright.dev/docs/api/class-keyboard)
|
|
- [WebdriverIO `action('key')` API](https://webdriver.io/docs/api/browser/action#key-input-source)
|
|
- [testing-library `type` API](https://testing-library.com/docs/user-event/utility/#type)
|
|
|
|
## userEvent.tab
|
|
|
|
```ts
|
|
function tab(options?: UserEventTabOptions): Promise<void>
|
|
```
|
|
|
|
Sends a `Tab` key event. This is a shorthand for `userEvent.keyboard('{tab}')`.
|
|
|
|
```ts
|
|
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()
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `Keyboard` API](https://playwright.dev/docs/api/class-keyboard)
|
|
- [WebdriverIO `action('key')` API](https://webdriver.io/docs/api/browser/action#key-input-source)
|
|
- [testing-library `tab` API](https://testing-library.com/docs/user-event/convenience/#tab)
|
|
|
|
## userEvent.type
|
|
|
|
```ts
|
|
function type(
|
|
element: Element | Locator,
|
|
text: string,
|
|
options?: UserEventTypeOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
::: warning
|
|
If you don't rely on [special characters](https://testing-library.com/docs/user-event/keyboard) (e.g., `{shift}` or `{selectall}`), it is recommended to use [`userEvent.fill`](#userevent-fill) instead for better performance.
|
|
:::
|
|
|
|
The `type` method implements `@testing-library/user-event`'s [`type`](https://testing-library.com/docs/user-event/utility/#type) utility built on top of [`keyboard`](https://testing-library.com/docs/user-event/keyboard) API.
|
|
|
|
This function allows you to type characters into an `input`/`textarea`/`contenteditable` element. It supports [user-event `keyboard` syntax](https://testing-library.com/docs/user-event/keyboard).
|
|
|
|
If you just need to press characters without an input, use [`userEvent.keyboard`](#userevent-keyboard) API.
|
|
|
|
```ts
|
|
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 doesn't expose `.type` method on the locator like `input.type` because it exists only for compatibility with the `userEvent` library. Consider using `.fill` instead as it is faster.
|
|
:::
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.press` API](https://playwright.dev/docs/api/class-locator#locator-press)
|
|
- [WebdriverIO `action('key')` API](https://webdriver.io/docs/api/browser/action#key-input-source)
|
|
- [testing-library `type` API](https://testing-library.com/docs/user-event/utility/#type)
|
|
|
|
## userEvent.clear
|
|
|
|
```ts
|
|
function clear(element: Element | Locator, options?: UserEventClearOptions): Promise<void>
|
|
```
|
|
|
|
This method clears the input element content.
|
|
|
|
```ts
|
|
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('')
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.clear` API](https://playwright.dev/docs/api/class-locator#locator-clear)
|
|
- [WebdriverIO `element.clearValue` API](https://webdriver.io/docs/api/element/clearValue)
|
|
- [testing-library `clear` API](https://testing-library.com/docs/user-event/utility/#clear)
|
|
|
|
## userEvent.selectOptions
|
|
|
|
```ts
|
|
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
|
|
If select element doesn't have [`multiple`](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/select#attr-multiple) attribute, Vitest will select only the first element in the array.
|
|
|
|
Unlike `@testing-library`, Vitest doesn't support [listbox](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/listbox_role) at the moment, but we plan to add support for it in the future.
|
|
:::
|
|
|
|
```ts
|
|
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 doesn't support selecting multiple elements because it doesn't provide API to do so.
|
|
:::
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.selectOption` API](https://playwright.dev/docs/api/class-locator#locator-select-option)
|
|
- [WebdriverIO `element.selectByIndex` API](https://webdriver.io/docs/api/element/selectByIndex)
|
|
- [testing-library `selectOptions` API](https://testing-library.com/docs/user-event/utility/#-selectoptions-deselectoptions)
|
|
|
|
## userEvent.hover
|
|
|
|
```ts
|
|
function hover(
|
|
element: Element | Locator,
|
|
options?: UserEventHoverOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
This method moves the cursor position to the selected element. Please refer to your provider's documentation for detailed explanation about how this method works.
|
|
|
|
::: warning
|
|
If you are using `webdriverio` provider, the cursor will move to the center of the element by default.
|
|
|
|
If you are using `playwright` provider, the cursor moves to "some" visible point of the element.
|
|
:::
|
|
|
|
```ts
|
|
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()
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.hover` API](https://playwright.dev/docs/api/class-locator#locator-hover)
|
|
- [WebdriverIO `element.moveTo` API](https://webdriver.io/docs/api/element/moveTo/)
|
|
- [testing-library `hover` API](https://testing-library.com/docs/user-event/convenience/#hover)
|
|
|
|
## userEvent.unhover
|
|
|
|
```ts
|
|
function unhover(
|
|
element: Element | Locator,
|
|
options?: UserEventHoverOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
This works the same as [`userEvent.hover`](#userevent-hover), but moves the cursor to the `document.body` element instead.
|
|
|
|
::: warning
|
|
By default, the cursor position is in "some" visible place (in `playwright` provider) or in the center (in `webdriverio` provider) of the body element, so if the currently hovered element is already in the same position, this method will have no effect.
|
|
:::
|
|
|
|
```ts
|
|
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()
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.hover` API](https://playwright.dev/docs/api/class-locator#locator-hover)
|
|
- [WebdriverIO `element.moveTo` API](https://webdriver.io/docs/api/element/moveTo/)
|
|
- [testing-library `hover` API](https://testing-library.com/docs/user-event/convenience/#hover)
|
|
|
|
## userEvent.upload
|
|
|
|
```ts
|
|
function upload(
|
|
element: Element | Locator,
|
|
files: string[] | string | File[] | File,
|
|
options?: UserEventUploadOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
Change a file input element to have the specified files.
|
|
|
|
```ts
|
|
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 root of the project
|
|
await userEvent.upload(input, './fixtures/file.png')
|
|
})
|
|
```
|
|
|
|
::: warning
|
|
`webdriverio` provider supports this command only in `chrome` and `edge` browsers. It also only supports string types at the moment.
|
|
:::
|
|
|
|
References:
|
|
|
|
- [Playwright `locator.setInputFiles` API](https://playwright.dev/docs/api/class-locator#locator-set-input-files)
|
|
- [WebdriverIO `browser.uploadFile` API](https://webdriver.io/docs/api/browser/uploadFile)
|
|
- [testing-library `upload` API](https://testing-library.com/docs/user-event/utility/#upload)
|
|
|
|
## userEvent.dragAndDrop
|
|
|
|
```ts
|
|
function dragAndDrop(
|
|
source: Element | Locator,
|
|
target: Element | Locator,
|
|
options?: UserEventDragAndDropOptions,
|
|
): Promise<void>
|
|
```
|
|
|
|
Drags the source element on top of the target element. Don't forget that the `source` element has to have the `draggable` attribute set to `true`.
|
|
|
|
```ts
|
|
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
|
|
This API is not supported by the default `preview` provider.
|
|
:::
|
|
|
|
References:
|
|
|
|
- [Playwright `frame.dragAndDrop` API](https://playwright.dev/docs/api/class-frame#frame-drag-and-drop)
|
|
- [WebdriverIO `element.dragAndDrop` API](https://webdriver.io/docs/api/element/dragAndDrop/)
|
|
|
|
## userEvent.copy
|
|
|
|
```ts
|
|
function copy(): Promise<void>
|
|
```
|
|
|
|
Copy the selected text to the clipboard.
|
|
|
|
```js
|
|
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')
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [testing-library `copy` API](https://testing-library.com/docs/user-event/convenience/#copy)
|
|
|
|
## userEvent.cut
|
|
|
|
```ts
|
|
function cut(): Promise<void>
|
|
```
|
|
|
|
Cut the selected text to the clipboard.
|
|
|
|
```js
|
|
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')
|
|
})
|
|
```
|
|
|
|
References:
|
|
|
|
- [testing-library `cut` API](https://testing-library.com/docs/user-event/clipboard#cut)
|
|
|
|
## userEvent.paste
|
|
|
|
```ts
|
|
function paste(): Promise<void>
|
|
```
|
|
|
|
Paste the text from the clipboard. See [`userEvent.copy`](#userevent-copy) and [`userEvent.cut`](#userevent-cut) for usage examples.
|
|
|
|
References:
|
|
|
|
- [testing-library `paste` API](https://testing-library.com/docs/user-event/clipboard#paste)
|