Assertion
expect.element is the API for Locator assertions in Browser Mode. It accepts a Locator and returns a chainable assertion object. Unlike expect(value) which primarily compares JS values, expect.element targets the real element state on the page, suitable for verifying visibility, text, form values, count, and other user-observable results.
With the Playwright provider, all expect.element matchers automatically retry until the assertion passes or the timeout is reached. You do not need to write manual waitFor or polling logic — just assert the expected state and the framework handles the waiting.
When you import @rstest/browser in a Browser Mode test (e.g., importing page), expect automatically gains the element capability. This allows queries, interactions, and assertions to be organized around the same Locator, focusing failure messages on element state.
The following minimal example demonstrates common usage: first assert visibility, then verify a checked state change, and finally validate an element attribute.
Type signature
Auto-retry and timeout
Assertion behavior in expect.element depends on the provider. With the Playwright provider, expect.element matchers are web-first assertions: they continuously retry within the timeout duration until the assertion passes or times out. You do not need to write manual waitFor or polling logic.
Timeout priority
The assertion timeout is determined by the following priority:
- Per-assertion timeout: The
timeoutparameter passed directly to the matcher, highest priority - Global
testTimeoutconfiguration: FromtestTimeoutinrstest.config.ts(default 5000ms) - RPC fallback timeout: 30000ms (serves only as a communication-layer safety net, usually not reached)
Failure output
When an assertion times out, the error message includes: the expected element state, the actual state, and the timeout duration. For example:
Use cases
Auto-retry is especially useful for handling async rendering scenarios — such as waiting for a loading indicator to disappear or UI changes after data finishes loading:
Input constraints
locatormust be aLocatorreturned by@rstest/browser. This allows the runtime to recognize and replay the complete query chain; hand-crafted objects or locators from other libraries cannot be parsed by this mechanism.expect.elementis not available outside of Browser Mode. It relies on Browser Mode's browser runtime and communication channel to execute element assertions; pure Node test environments only support regularexpect(value)assertions.
Element assertions
The matchers listed below represent the currently supported subset. Matchers not listed here are not yet available.
not
- Type:
BrowserElementExpect
Negates the subsequent assertion.
All the following matchers support an optional options?: { timeout?: number } parameter (unless the type signature states otherwise).
toBeVisible
- Type:
(options?: { timeout?: number }) => Promise<void>
The Locator resolves to a mounted and visible element within the timeout.
Visibility is determined by: the element has a non-empty bounding box and visibility is not hidden. For example, display: none or zero dimensions are not considered visible; opacity: 0 is still considered visible.
toBeHidden
- Type:
(options?: { timeout?: number }) => Promise<void>
The Locator meets any of the following conditions within the timeout: does not match any DOM node, or matches a non-visible node.
Think of it as the opposite of toBeVisible; for example, setting only opacity: 0 typically will not cause toBeHidden to pass.
toBeEnabled
- Type:
(options?: { timeout?: number }) => Promise<void>
The element is not in a disabled state.
Disabled determination: a native form control with the disabled attribute, within a disabled fieldset, or within an aria-disabled=true semantic context may all be considered disabled.
toBeDisabled
- Type:
(options?: { timeout?: number }) => Promise<void>
The element is determined to be disabled.
The determination rules are the same as toBeEnabled, just with the opposite result. Recommended for scenarios like button submission and preventing duplicate clicks.
toBeChecked
- Type:
(options?: { timeout?: number }) => Promise<void>
The checked state of a checkbox or radio is true.
Commonly used to verify user check actions or whether a default checked state has taken effect.
toBeUnchecked
- Type:
(options?: { timeout?: number }) => Promise<void>
The checked state of a checkbox or radio is false.
Suitable for use with check / uncheck to verify state changes before and after interaction.
toBeAttached
- Type:
(options?: { timeout?: number }) => Promise<void>
The node pointed to by the Locator is connected to a Document or ShadowRoot (equivalent to Node.isConnected === true).
This assertion only checks "whether it is in the DOM tree" and does not require the element to be visible.
toBeDetached
- Type:
(options?: { timeout?: number }) => Promise<void>
The Locator no longer points to a connected DOM node.
Common for verification after conditional rendering, async unmounting, or delete operations.
toBeEditable
- Type:
(options?: { timeout?: number }) => Promise<void>
The element is both enabled and not readonly.
Readonly determination includes both the native readonly attribute and aria-readonly=true semantics.
toBeFocused
- Type:
(options?: { timeout?: number }) => Promise<void>
The element is the current document's focus target (active element).
Suitable for verifying keyboard navigation, auto-focus, or form focus-switching behavior.
toBeEmpty
- Type:
(options?: { timeout?: number }) => Promise<void>
An editable element's content is empty, or a regular DOM node has no text content.
It checks "whether the content is empty", not "whether the element exists" or "whether it is visible".
toBeInViewport
- Type:
(options?: { timeout?: number; ratio?: number }) => Promise<void>
The element intersects with the viewport (based on Intersection Observer semantics).
ratio represents the minimum intersection ratio; for example, ratio: 0.5 means at least half of the area is within the viewport.
toHaveText
- Type:
(text: string | RegExp, options?: { timeout?: number }) => Promise<void>
The element's text fully matches the expected value (supports string / RegExp).
Text computation includes nested child element content. When the expected value is a string, whitespace and line breaks are normalized before matching.
toContainText
- Type:
(text: string | RegExp, options?: { timeout?: number }) => Promise<void>
The element's text contains the expected substring, or matches the given regex.
The difference from toHaveText is: toContainText does substring matching, while toHaveText does full matching.
toHaveValue
- Type:
(value: string | RegExp, options?: { timeout?: number }) => Promise<void>
The form control's current value matches the expected value (supports string / RegExp).
Applicable to input, textarea, select, and other elements with retrievable values.
toHaveId
- Type:
(value: string | RegExp, options?: { timeout?: number }) => Promise<void>
The element's id matches the expected value (supports string / RegExp).
Suitable for validating dynamically generated IDs or fixed IDs injected after component mounting.
toHaveClass
- Type:
(value: string | RegExp, options?: { timeout?: number }) => Promise<void>
The element's class attribute matches the expected value (supports string / RegExp).
When passing a string, it matches against the entire class string. If you only care about a specific class, consider using a more targeted regex.
toHaveAttribute
- Type:
(name: string, options?: { timeout?: number }) => Promise<void>(name: string, value: string | RegExp, options?: { timeout?: number }) => Promise<void>
When only name is passed, it asserts the attribute exists. When name + value is passed, it asserts the attribute value matches (value supports string / RegExp).
Common use cases include validating structural attributes like type, disabled, aria-*, etc.
toHaveCount
- Type:
(count: number, options?: { timeout?: number }) => Promise<void>
The number of elements resolved by the Locator exactly matches count.
Suitable for list rendering, filter results, pagination item counts, and similar scenarios.
toHaveCSS
- Type:
(name: string, value: string | RegExp, options?: { timeout?: number }) => Promise<void>
The specified CSS property value in the element's computed style matches the expected value.
name must be a non-empty string; value supports string / RegExp.
toHaveJSProperty
- Type:
(name: string, value: unknown, options?: { timeout?: number }) => Promise<void>
The JS property on the element matches the expected value.
name must be a non-empty string; value must be JSON-serializable (assertion parameters are transmitted through the Browser Mode channel).