Dialog (Compound Component)
The dialog component informs users about a task and can contain critical information, require decisions, or involve multiple tasks. The Dialog component can be used to create a modal, alert, or other custom dialog and can be fully customized to fit your needs.
<Dialog.Root> <Dialog.Content /> <Dialog.Trigger /></Dialog.Root>
<custom-dialog> <dialog data-dialog="dialog" class="dialog"> <button type="button" title="Close" data-component="Button" data-dialog="close" class="btn-subtle btn-sm btn-icon absolute right-4 top-4 size-24 !min-h-0 [&_svg]:size-12" > <svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 15 14"><path d="M14.57 1.41L13.16 0 7.57 5.59 1.98 0 .57 1.41 6.16 7 .57 12.59 1.98 14l5.59-5.59L13.16 14l1.41-1.41L8.98 7l5.59-5.59z"></path></svg> Close </button> <div class="flex items-center"> No content has been provided for this dialog. </div> </dialog> <button type="button" data-component="Button" data-dialog="trigger" class="btn-contained" > Open Dialog </button></custom-dialog>
Anatomy
This component is comprised of several elements that work together to create a dialog. The following are the elements that make up the Dialog component.
---import Dialog from '@/components/elements/dialog/compound-component/Dialog'---
<Dialog.Root> <Dialog.Content /> <Dialog.Trigger /></Dialog.Root>
API Reference
Root
The root element of the Dialog component. This element wraps the content and trigger elements and provides the scripting logic to open and close the dialog.
Content
The content of the Dialog component. This component is based on the <dialog>
element and can take in any additional HTML attributes. The following are the custom props that can be passed to the component.
Prop | Type | Default |
---|---|---|
hideClose | boolean | false |
Trigger
The trigger that opens the Dialog component. This element inherits from the Button component.
Prop | Type | Default |
---|---|---|
variant | "contained" | "outlined" | "subtle" | "text" | "contained" |
size | "sm" | "md" | "lg" | "md" |
title | string | — |
icon | boolean | false |
href | string | — |
disabled | boolean | false |
Accessibility
The <Dialog>
component leverages the <dialog>
element which is natively accessible. The following are some additional accessibility features that are built into the component:
- Clicking off from the dialog will close it. This is done by listening for a click event on the dialog element and checking if the target is the dialog itself.
- When the dialog element is opened, the body element will receive the
overflow-hidden
class to prevent scrolling. - When the dialog includes a form, we suggest passing
autofocus
to the first input field to ensure that the user can start interacting with the form immediately. - Ensure forms are validated before submitting. This can be done by adding the
required
attribute to the input fields and setting theformnovalidate
attribute on the cancel button.
Examples
Hide Close Button
You can hide the default close button by passing the hideClose
prop to the <Dialog.Content>
component.
<Dialog.Root> <Dialog.Content hideClose /> <Dialog.Trigger /></Dialog.Root>
<custom-dialog> <dialog data-dialog="dialog" class="dialog"> <div class="flex items-center"> No content has been provided for this dialog. </div> </dialog> <button type="button" data-component="Button" data-dialog="trigger" class="btn-contained" > Open Dialog </button></custom-dialog>
Form Dialog
You can use a dialog to handle form data. The following example demonstrates how to create a custom form dialog using the <Dialog>
component.
---import Button from '@/components/Button.astro'import TextInput from '@/components/forms/TextInput.astro'---
<Dialog.Root> <Dialog.Content> <form method="dialog" class="grid gap-24"> <div class="flex gap-16 p-2"> <TextInput label="First Name" name="first-name" placeholder="John" required inputProps={{ autofocus: true }} /> <TextInput label="Last Name" name="last-name" placeholder="Smith" required /> </div> <footer class="flex items-center justify-end gap-16"> <Button type="submit" variant="outlined" data-dialog="close" formnovalidate >Cancel</Button> <Button type="submit">Submit</Button> </footer> </form> </Dialog.Content> <Dialog.Trigger /></Dialog.Root>
<custom-dialog> <dialog data-dialog="dialog" class="dialog"> <button type="button" title="Close" data-component="Button" data-dialog="close" class="btn-subtle btn-sm btn-icon absolute right-4 top-4 size-24 !min-h-0 [&_svg]:size-12"> <svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 15 14"><path d="M14.57 1.41L13.16 0 7.57 5.59 1.98 0 .57 1.41 6.16 7 .57 12.59 1.98 14l5.59-5.59L13.16 14l1.41-1.41L8.98 7l5.59-5.59z"></path></svg> Close </button> <form method="dialog" class="grid gap-24"> <div class="flex gap-16 p-2"> <div class="field-group" data-required="" data-layout="FieldGroup" data-component="TextInput" > <label for="first-name" id="label-first-name"> <span class="field-label">First Name</span> </label> <input type="text" id="first-name" name="input-first-name" placeholder="John" required="" aria-labelledby="label-first-name" autofocus="" > </div> <div class="field-group" data-required="" data-layout="FieldGroup" data-component="TextInput" > <label for="last-name" id="label-last-name"> <span class="field-label">Last Name</span> </label> <input type="text" id="last-name" name="input-last-name" placeholder="Smith" required="" aria-labelledby="label-last-name" > </div> </div> <footer class="flex items-center justify-end gap-16"> <button type="submit" data-component="Button" data-dialog="close" formnovalidate="" class="btn-outlined" > Cancel </button> <button type="submit" data-component="Button" class="btn-contained" > Submit </button> </footer> </form> </dialog>
<button type="button" data-component="Button" data-dialog="trigger" class="btn-contained" > Open Dialog </button></custom-dialog>
Custom Trigger & Dialog
The dialog element can be customized to fit whatever context it is being used in. The following example demonstrates how to create a custom trigger and dialog.
<Dialog.Root> <Dialog.Content class="rounded-lg border-2 border-white bg-gradient-to-r from-indigo-500 from-10% via-sky-500 via-30% to-emerald-500 to-90% p-24"> <div class="flex items-center"> Culpa proident non exercitation eu consequat Lorem ipsum. </div> </Dialog.Content> <Dialog.Trigger class="rounded bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 p-8 font-bold !text-white transition hover:bg-gradient-to-tr hover:opacity-85 active:opacity-75"> Open Custom Dialog </Dialog.Trigger></Dialog.Root>
<custom-dialog> <dialog data-dialog="dialog" class="dialog rounded-lg border-2 border-white bg-gradient-to-r from-indigo-500 from-10% via-sky-500 via-30% to-emerald-500 to-90% p-24" > <button type="button" title="Close" data-component="Button" data-dialog="close" class="btn-subtle btn-sm btn-icon absolute right-4 top-4 size-24 !min-h-0 [&_svg]:size-12" > <svg xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 15 14"><path d="M14.57 1.41L13.16 0 7.57 5.59 1.98 0 .57 1.41 6.16 7 .57 12.59 1.98 14l5.59-5.59L13.16 14l1.41-1.41L8.98 7l5.59-5.59z"></path></svg> Close </button> <div class="flex items-center"> No content has been provided for this dialog. </div> </dialog> <button type="button" data-component="Button" data-dialog="trigger" class="btn-contained rounded bg-gradient-to-r from-indigo-500 via-purple-500 to-pink-500 p-8 font-bold !text-white transition hover:bg-gradient-to-tr hover:opacity-85 active:opacity-75" > Open Dialog </button></custom-dialog>
Astro Component
It’s recommended that an element like this is made as a reusable component. In this case, we have put together the following Astro compound component to serve as a blueprint for how you could possibly set up a Dialog component, and what properties to consider.
import Root from './Root.astro'import Content from './Content.astro'import Trigger from './Trigger.astro'
export default Object.assign({ Root, Content, Trigger })
---
---
<custom-dialog data-component="Dialog"> <slot /></custom-dialog>
<script> class CustomDialog extends HTMLElement { constructor() { super() }
connectedCallback() { this.dialogTrigger.addEventListener( 'click', this.handleDialogTriggerClick, ) this.dialog.addEventListener('click', this.handleDialogClick) this.dialogCloses?.forEach((dialogClose: Element) => { dialogClose.addEventListener('click', this.handleDialogCloseClick) }) this.dialogForm?.addEventListener('submit', this.handleDialogFormSubmit) this.dialog.addEventListener('keydown', this.handleDialogEsc) }
disconnectedCallback() { this.dialogTrigger.removeEventListener( 'click', this.handleDialogTriggerClick, ) this.dialog.removeEventListener('click', this.handleDialogClick) this.dialogCloses?.forEach((dialogClose: Element) => { dialogClose.removeEventListener('click', this.handleDialogCloseClick) }) this.dialogForm?.removeEventListener( 'submit', this.handleDialogFormSubmit, ) this.dialog.removeEventListener('keydown', this.handleDialogEsc) }
private static readonly DELAY_TIME = 300
private get dialogTrigger(): HTMLButtonElement { return this.querySelector('[data-dialog="trigger"]')! }
private get dialog(): HTMLDialogElement { return this.querySelector('[data-dialog="dialog"]')! }
private get dialogCloses(): NodeListOf<Element> { return this.querySelectorAll('[data-dialog="close"]') }
private get dialogForm(): HTMLFormElement | null { return this.querySelector('form') }
// Open the dialog when the user clicks the trigger private handleDialogTriggerClick = () => { if ('dialogStatus' in this.dialog.dataset) return
this.dialog.dataset.dialogStatus = 'opening'
setTimeout(() => { delete this.dialog.dataset.dialogStatus }, CustomDialog.DELAY_TIME)
this.dialog.showModal() // Lock scroll when dialog is open document.body.style.overflow = 'hidden' }
// Close the dialog when the user clicks backdrop private handleDialogClick = (event: Event) => { if ('dialogStatus' in this.dialog.dataset) return
if (event.target === this.dialog) this.handleDialogClose() }
// Intercept default escape and add custom close behavior private handleDialogEsc = (event: KeyboardEvent) => { if (this.dialog.open && event.key === 'Escape') { event.preventDefault() this.handleDialogClose() } }
// Close the dialog when the user clicks a close button private handleDialogCloseClick = (event: Event) => { event.preventDefault() this.handleDialogClose() }
// Close the dialog when the form is submitted and valid private handleDialogFormSubmit = (event: Event) => { event.preventDefault()
if (this.dialogForm?.checkValidity()) this.handleDialogClose() }
// Close the dialog with delay for animation private handleDialogClose() { if ('dialogStatus' in this.dialog.dataset) return
this.dialog.dataset.dialogStatus = 'closing' setTimeout(() => { delete this.dialog.dataset.dialogStatus this.dialog.close() }, CustomDialog.DELAY_TIME)
// Unlock scroll when dialog is closed document.body.style.overflow = '' } }
customElements.define('custom-dialog', CustomDialog)</script>
---import type { HTMLAttributes } from 'astro/types'import Button from '@/components/elements/Button.astro'import close from '@/assets/svg/close.svg?raw'
interface Props extends HTMLAttributes<'dialog'> { /* Whether or not the close button should be hidden. */ hideClose?: boolean /* Classes to pass into the dialog content element. */ class: string}
const { hideClose = false, class: className, ...attrs } = Astro.props---
<dialog data-dialog="dialog" class:list={['dialog', className]} {...attrs}> { !hideClose && ( <Button icon size="sm" title="Close" variant="subtle" class="absolute right-4 top-4 size-24 !min-h-0 [&_svg]:size-12" data-dialog="close" > <Fragment set:html={close} /> Close </Button> ) }
<slot> <div class="flex items-center"> No content has been provided for this dialog. </div> </slot></dialog>
---import type { ComponentProps } from 'astro/types'import Button from '@/components/elements/Button.astro'
type ButtonProps = ComponentProps<typeof Button>
const { ...ButtonProps } = Astro.props---
<Button data-dialog="trigger" {...ButtonProps}> <slot>Open Dialog</slot></Button>