跳到主要内容

实用的 API

以下的 API 在真实的用户交互中没有一对一的对应关系。

因此,他们的行为是对 "感知的" 用户交互如何转化为 DOM 上的实际事件的一种解释。

clear()

clear(element: Element): Promise<void>

此 API 可用于快速清除可编辑元素的内容。

  1. 聚焦元素
  2. 根据浏览器菜单选择所有内容
  3. 根据浏览器菜单删除内容
test('clear', async () => {
render(<textarea defaultValue="Hello, World!" />)

await userEvent.clear(screen.getByRole('textbox'))

expect(screen.getByRole('textbox')).toHaveValue('')
})

如果元素无法聚焦或无法选择内容,则 Promise 被拒绝。

selectOptions(), deselectOptions()

selectOptions(
element: Element,
values: HTMLElement | HTMLElement[] | string[] | string,
): Promise<void>
deselectOptions(
element: Element,
values: HTMLElement | HTMLElement[] | string[] | string,
): Promise<void>

选择/取消选择 HTMLSelectElement列表框 .

values 参数可以按其值指代一个选项,HTML 内容或只是提供元素。它也接受一个这样的 数组。

选择多个选项和(或)取消选择 HTMLSelectElement 的选项只有在指定了 multiple 的情况下才有可能。

test('selectOptions', async () => {
render(
<select multiple>
<option value="1">A</option>
<option value="2">B</option>
<option value="3">C</option>
</select>,
)

await userEvent.selectOptions(screen.getByRole('listbox'), ['1', 'C'])

expect(screen.getByRole('option', {name: 'A'}).selected).toBe(true)
expect(screen.getByRole('option', {name: 'B'}).selected).toBe(false)
expect(screen.getByRole('option', {name: 'C'}).selected).toBe(true)
})
test('deselectOptions', async () => {
render(
<select multiple>
<option value="1">A</option>
<option value="2" selected>
B
</option>
<option value="3">C</option>
</select>,
)

await userEvent.deselectOptions(screen.getByRole('listbox'), '2')

expect(screen.getByText('B').selected).toBe(false)
})

请注意,这个 API 会触发指针事件,因此要受到 pointerEventsCheck 的约束。

type()

type(
element: Element,
text: KeyboardInput,
options?: {
skipClick?: boolean
skipAutoClose?: boolean
initialSelectionStart?: number
initialSelectionEnd?: number
}
): Promise<void>

在一个输入元素中打字。

如果你想只是模拟按键盘上的按钮,你应该使用 keyboard()。 如果你只是想方便地在一个输入字段或文本区中插入一些文本,你可以使用 type()

  1. 除非 skipClicktrue,否则点击该元素。
  2. 如果 initialSelectionStart 被设置,则在该元素上设置选中。如果 initialSelectionEnd 没有被设置,这将导致一个塌陷的选中。
  3. keyboard() 键入给定的 text
  4. 除非 skipAutoClosetrue,否则释放所有按下的键。
test('type into an input field', async () => {
render(<input defaultValue="Hello," />)
const input = screen.getByRole('textbox')

await userEvent.type(input, ' World!')

expect(input).toHaveValue('Hello, World!')
})

upload()

upload(
element: HTMLElement,
fileOrFiles: File | File[],
): Promise<void>

改变一个文件输入,就像用户点击它并在弹出的文件上传对话框中选择文件一样。

不符合 accept 属性的文件将被自动丢弃,除非 applyAccept 被设置为 false。

test('upload file', async () => {
render(
<div>
<label htmlFor="file-uploader">Upload file:</label>
<input id="file-uploader" type="file" />
</div>,
)
const file = new File(['hello'], 'hello.png', {type: 'image/png'})
const input = screen.getByLabelText(/upload file/i)

await userEvent.upload(input, file)

expect(input.files[0]).toBe(file)
expect(input.files.item(0)).toBe(file)
expect(input.files).toHaveLength(1)
})

test('upload multiple files', async () => {
render(
<div>
<label htmlFor="file-uploader">Upload file:</label>
<input id="file-uploader" type="file" multiple />
</div>,
)
const files = [
new File(['hello'], 'hello.png', {type: 'image/png'}),
new File(['there'], 'there.png', {type: 'image/png'}),
]
const input = screen.getByLabelText(/upload file/i)

await userEvent.upload(input, files)

expect(input.files).toHaveLength(2)
expect(input.files[0]).toBe(files[0])
expect(input.files[1]).toBe(files[1])
})