Dialog

A popup that opens on top of the entire page.

View as Markdown

Anatomy

Import the component and assemble its parts:

Anatomy

API reference

Root

Groups all parts of the dialog. Doesn’t render its own HTML element.

defaultOpen

boolean

(default:

false

)
Description

Whether the dialog is initially open.

To render a controlled dialog, use the open prop instead.

Type

boolean

Default

false

open

boolean

Name
Description

Whether the dialog is currently open.

Type

boolean

onOpenChange

function

Description

Event handler called when the dialog is opened or closed.

Type

((open: boolean, event: Event | undefined, reason: Dialog.Root.OpenChangeReason | undefined) => void)

actionsRef

RefObject<Dialog.Root.Actions>

Description

A ref to imperative actions.

  • unmount: When specified, the dialog will not be unmounted when closed. Instead, the unmount function must be called to unmount the dialog manually. Useful when the dialog's animation is controlled by an external library.
Type

RefObject<Dialog.Root.Actions>

dismissible

boolean

(default:

true

)
Description

Determines whether the dialog should close on outside clicks.

Type

boolean

Default

true

modal

boolean | 'trap-focus'

(default:

true

)
Name
Description

Determines if the dialog enters a modal state when open.

  • true: user interaction is limited to just the dialog: focus is trapped, document page scroll is locked, and pointer interactions on outside elements are disabled.
  • false: user interaction with the rest of the document is allowed.
  • 'trap-focus': focus is trapped inside the dialog, but document page scroll is not locked and pointer interactions outside of it remain enabled.
Type

boolean | 'trap-focus'

Default

true

onOpenChangeComplete

function

Description

Event handler called after any animations complete when the dialog is opened or closed.

Type

((open: boolean) => void)

children

ReactNode

Type

ReactNode

Trigger

A button that opens the dialog. Renders a <button> element.

nativeButton

boolean

(default:

true

)
Description

Whether the component renders a native <button> element when replacing it via the render prop. Set to false if the rendered element is not a button (e.g. <div>).

Type

boolean

Default

true

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Trigger.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Trigger.State) => ReactElement)

data-popup-open

Present when the corresponding dialog is open.

data-disabled

Present when the trigger is disabled.

Portal

A portal element that moves the popup to a different part of the DOM. By default, the portal element is appended to <body>.

container

Union

Description

A parent element to render the portal element into.

Type

| HTMLElement
| ShadowRoot
| RefObject<HTMLElement
| ShadowRoot
| null>
| null

children

ReactNode

Type

ReactNode

keepMounted

boolean

(default:

false

)
Description

Whether to keep the portal mounted in the DOM while the popup is hidden.

Type

boolean

Default

false

Backdrop

An overlay displayed beneath the popup. Renders a <div> element.

forceRender

boolean

(default:

false

)
Description

Whether the backdrop is forced to render even when nested.

Type

boolean

Default

false

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Backdrop.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Backdrop.State) => ReactElement)

data-open

Present when the dialog is open.

data-closed

Present when the dialog is closed.

data-starting-style

Present when the dialog is animating in.

data-ending-style

Present when the dialog is animating out.

A container for the dialog contents. Renders a <div> element.

initialFocus

Union

Description

Determines the element to focus when the dialog is opened. By default, the first focusable element is focused.

Type

| RefObject<HTMLElement
| null>
| ((interactionType: InteractionType) => RefObject<HTMLElement | null>)

finalFocus

RefObject<HTMLElement | null>

Description

Determines the element to focus when the dialog is closed. By default, focus returns to the trigger.

Type

RefObject<HTMLElement | null>

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Popup.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Popup.State) => ReactElement)

data-open

Present when the dialog is open.

data-closed

Present when the dialog is closed.

data-nested

Present when the dialog is nested within another dialog.

data-nested-dialog-open

Present when the dialog has other open dialogs nested within it.

data-starting-style

Present when the dialog is animating in.

data-ending-style

Present when the dialog is animating out.

--nested-dialogs

Indicates how many dialogs are nested within.

Title

A heading that labels the dialog. Renders an <h2> element.

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Title.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Title.State) => ReactElement)

Description

A paragraph with additional information about the dialog. Renders a <p> element.

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Description.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Description.State) => ReactElement)

Close

A button that closes the dialog. Renders a <button> element.

nativeButton

boolean

(default:

true

)
Description

Whether the component renders a native <button> element when replacing it via the render prop. Set to false if the rendered element is not a button (e.g. <div>).

Type

boolean

Default

true

className

string | function

Description

CSS class applied to the element, or a function that returns a class based on the component’s state.

Type

| string
| ((state: Dialog.Close.State) => string)

render

ReactElement | function

Name
Description

Allows you to replace the component’s HTML element with a different tag, or compose it with another component.

Accepts a ReactElement or a function that returns the element to render.

Type

| ReactElement
| ((props: HTMLProps, state: Dialog.Close.State) => ReactElement)

data-disabled

Present when the button is disabled.

Examples

State

By default, Dialog is an uncontrolled component that manages its own state.

Uncontrolled dialog

Use open and onOpenChange props if you need to access or control the state of the dialog. For example, you can control the dialog state in order to open it imperatively from another place in your app.

Controlled dialog

It’s also common to use onOpenChange if your app needs to do something when the dialog is closed or opened. This is recommended over React.useEffect when reacting to state changes.

Running code when dialog state changes

Open from a menu

In order to open a dialog using a menu, control the dialog state and open it imperatively using the onClick handler on the menu item.

Make sure to also use the dialog’s finalFocus prop to return focus back to the menu trigger.

Connecting a dialog to a menu

Nested dialogs

You can nest dialogs within one another normally.

Use the [data-nested-dialog-open] selector and the var(--nested-dialogs) CSS variable to customize the styling of the parent dialog. Backdrops of the child dialogs won’t be rendered so that you can present the parent dialog in a clean way behind the one on top of it.

Close confirmation

This example shows a nested confirmation dialog that opens if the text entered in the parent dialog is going to be discarded.

To implement this, both dialogs should be controlled. The confirmation dialog may be opened when onOpenChange callback of the parent dialog receives a request to close. This way, the confirmation is automatically shown when the user clicks the backdrop, presses the Esc key, or clicks a close button.