Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Dialog component #681

Merged
merged 34 commits into from
Nov 21, 2023
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
c297408
Scaffold and some basic concepts
dlnr Oct 23, 2023
24738c6
Long content story and some props
dlnr Oct 25, 2023
809e4d1
Tokens filled
dlnr Oct 26, 2023
48e3895
Return value and docs recommendations
dlnr Oct 26, 2023
7b81e91
Merged with main
dlnr Nov 7, 2023
654259a
Some documentation and docs css
dlnr Nov 8, 2023
473fd73
Button to cancel dialog
dlnr Nov 8, 2023
e4951a6
Merge branch 'main' of https://github.com/Amsterdam/design-system int…
dlnr Nov 8, 2023
0d9f119
CSS comment
dlnr Nov 8, 2023
f9205e9
Spacing token removed
dlnr Nov 9, 2023
dcb86fa
Unit tests
dlnr Nov 9, 2023
9a0bb0d
Merge branch 'main' of https://github.com/Amsterdam/design-system int…
dlnr Nov 9, 2023
ddbd78f
Merge branch 'main' of https://github.com/Amsterdam/design-system int…
dlnr Nov 14, 2023
1bbdafb
Feedback processed
dlnr Nov 14, 2023
f652248
Close modal by ID HTMLDialogElement
dlnr Nov 14, 2023
192a5dd
Hidden dialog tests
dlnr Nov 14, 2023
d9554cc
Update packages/css/src/dialog/README.md
dlnr Nov 16, 2023
9508c43
Update packages/css/src/dialog/README.md
dlnr Nov 16, 2023
352fc83
Failed test in jest succesful in interaction test
dlnr Nov 16, 2023
e9459eb
Remove test
dlnr Nov 16, 2023
1463d62
Replaced close onclick function
dlnr Nov 16, 2023
edd5748
Feedback on flex-start
dlnr Nov 16, 2023
0baac51
You're right
dlnr Nov 16, 2023
e920bff
Merge branch 'main' of https://github.com/Amsterdam/design-system int…
dlnr Nov 16, 2023
746b2c8
Add two cases we currently can't test
alimpens Nov 16, 2023
cbfcc9f
Merge branch 'feature/DES-253-modal-dialog' of https://github.com/Ams…
alimpens Nov 16, 2023
3fdb41f
Close button role removed
dlnr Nov 16, 2023
fc5f089
Removed css file for storybook
dlnr Nov 17, 2023
f46d959
Removed css import
dlnr Nov 17, 2023
ea0ffb2
Disabled background for this story
dlnr Nov 17, 2023
2e20d04
Dialog test fix
dlnr Nov 17, 2023
4f49e14
Disabled storybook test
dlnr Nov 21, 2023
7ae480b
Remove commented out code
dlnr Nov 21, 2023
c0130e7
Merge branch 'develop' into feature/DES-253-modal-dialog
alimpens Nov 21, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 22 additions & 0 deletions packages/css/src/dialog/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
# Dialog

Modals provide feedback or reveal additional information or functionality while maintaining the context of an underlying page.
alimpens marked this conversation as resolved.
Show resolved Hide resolved

## Guidelines

Use modals sparingly because they interrupt the user's workflow.
dlnr marked this conversation as resolved.
Show resolved Hide resolved

## Keyboard Support

| key | function |
| :---------- | :----------------------------------------------------------- |
| Tab | Moves focus to next focusable element inside the dialog. |
| Shift + Tab | Moves focus to previous focusable element inside the dialog. |
| Escape | Closes the dialog. |

## References

- [HTMLDialogElement](https://developer.mozilla.org/en-US/docs/Web/API/HTMLDialogElement)
- [Return value](https://developer.mozilla.org/en-US/docs/Web/API/HTMLDialogElement/returnValue)
- [Modal & Nonmodal Dialogs: When (& When Not) to Use Them](https://www.nngroup.com/articles/modal-nonmodal-dialog/)
- [Patterns](https://www.w3.org/WAI/ARIA/apg/patterns/dialog-modal/)
107 changes: 107 additions & 0 deletions packages/css/src/dialog/dialog.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
/**
* @license EUPL-1.2+
* Copyright (c) 2023 Gemeente Amsterdam
*/

@import "../../utils/breakpoint";

.amsterdam-dialog {
align-content: start;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
background-color: var(--amsterdam-dialog-background-color);
border: var(--amsterdam-dialog-border);
display: grid;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
inset: 0;
max-inline-size: var(--amsterdam-dialog-max-inline-size);
overflow: hidden;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
padding-block: 0;
padding-inline: 0;
position: fixed;

/* no token because dialog does not inherit from any element */
&::backdrop {
background: #0006;
}

&:not([open]) {
opacity: 0%;
pointer-events: none;
}
alimpens marked this conversation as resolved.
Show resolved Hide resolved
}

.amsterdam-dialog__form {
align-items: start;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
display: grid;
gap: var(--amsterdam-dialog-form-gap);
grid-template-rows: auto 1fr auto;
max-block-size: var(--amsterdam-dialog-form-max-block-size);
padding-block: var(--amsterdam-dialog-form-padding-block);
padding-inline: var(--amsterdam-dialog-form-padding-inline);
}

.amsterdam-dialog__article {
display: grid;
gap: 1.5rem;
justify-items: flex-start;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
max-block-size: 100%; /* safari */
overflow-y: auto;
overscroll-behavior-y: contain;
padding-inline-end: var(--amsterdam-dialog-article-padding-inline-end);

&::-webkit-scrollbar-track {
background: var(--amsterdam-dialog-background-color);
}
alimpens marked this conversation as resolved.
Show resolved Hide resolved
}

.amsterdam-dialog__header {
align-items: center;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
display: flex;
flex-direction: row;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
}

@mixin reset {
-webkit-text-size-adjust: 100%;
}

.amsterdam-dialog__title {
color: var(--amsterdam-dialog-title-color);
flex: auto;
font-family: var(--amsterdam-dialog-title-font-family);
font-size: var(--amsterdam-dialog-title-narrow-font-size);
font-weight: var(--amsterdam-dialog-title-font-weight);
line-height: var(--amsterdam-dialog-title-line-height);

@media screen and (min-width: $amsterdam-breakpoint-wide) {
alimpens marked this conversation as resolved.
Show resolved Hide resolved
font-size: var(--amsterdam-dialog-title-wide-font-size);
}

@include reset;
}

.amsterdam-dialog__footer {
display: flex;
flex-direction: column-reverse;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
grid-gap: var(--amsterdam-dialog-footer-gap);
padding-block: var(--amsterdam-dialog-footer-padding-block);

@media screen and (min-width: $amsterdam-breakpoint-medium) {
align-items: end;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
flex-direction: row;
justify-content: end;
}
}

.amsterdam-dialog__close {
background-color: var(--amsterdam-dialog-close-background-color);
border: 0;
cursor: pointer;
padding-block: 0;
padding-inline: 0;
}

.amsterdam-dialog__close svg {
fill: var(--amsterdam-dialog-close-fill);
}

.amsterdam-dialog__close:hover svg {
fill: var(--amsterdam-dialog-close-hover-fill);
}
1 change: 1 addition & 0 deletions packages/css/src/index.scss
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
*/

/* Append here */
@import "./dialog/dialog";
@import "./pagination/pagination";
@import "./accordion/accordion";
@import "./alert/alert";
Expand Down
84 changes: 84 additions & 0 deletions packages/react/src/Dialog/Dialog.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
import { render } from '@testing-library/react'
import { createRef } from 'react'
import { Dialog } from './Dialog'
import '@testing-library/jest-dom'

describe('Dialog', () => {
it('renders', () => {
const { container } = render(<Dialog open />)

const component = container.querySelector(':only-child')
alimpens marked this conversation as resolved.
Show resolved Hide resolved

expect(component).toBeInTheDocument()
expect(component).toBeVisible()
})

it('renders a design system BEM class name', () => {
const { container } = render(<Dialog />)

const component = container.querySelector(':only-child')

expect(component).toHaveClass('amsterdam-dialog')
})

it('renders an additional class name', () => {
const { container } = render(<Dialog className="extra" />)

const component = container.querySelector(':only-child')

expect(component).toHaveClass('extra')

expect(component).toHaveClass('amsterdam-dialog')
})

it('supports ForwardRef in React', () => {
const ref = createRef<HTMLDialogElement>()

const { container } = render(<Dialog ref={ref} />)

const component = container.querySelector(':only-child')

expect(ref.current).toBe(component)
})

it('is not visible when open attribute is not used', () => {
const { container } = render(<Dialog />)

const component = container.querySelector(':only-child')

expect(component).toBeInTheDocument()
expect(component).not.toBeVisible()
})

alimpens marked this conversation as resolved.
Show resolved Hide resolved
it('renders a title', () => {
const { getByText } = render(<Dialog title="Dialog Title" />)

expect(getByText('Dialog Title')).toBeInTheDocument()
})

it('renders children', () => {
const { getByText } = render(<Dialog>Dialog Content</Dialog>)

expect(getByText('Dialog Content')).toBeInTheDocument()
})

it('renders actions when provided', () => {
const { getByText } = render(<Dialog actions={<button>Click Me</button>} />)

expect(getByText('Click Me')).toBeInTheDocument()
})

it('does not render actions when not provided', () => {
const { queryByText } = render(<Dialog />)

expect(queryByText('Click Me')).not.toBeInTheDocument()
})

it('renders DialogClose button', () => {
const { container } = render(<Dialog />)

const closeButton = container.querySelector('.amsterdam-dialog__close')

expect(closeButton).toBeInTheDocument()
})
})
alimpens marked this conversation as resolved.
Show resolved Hide resolved
39 changes: 39 additions & 0 deletions packages/react/src/Dialog/Dialog.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
/**
* @license EUPL-1.2+
* Copyright (c) 2023 Gemeente Amsterdam
*/

import { CloseIcon } from '@amsterdam/design-system-react-icons'
import clsx from 'clsx'
import { DialogHTMLAttributes, ForwardedRef, forwardRef, PropsWithChildren, ReactNode } from 'react'
import { Icon } from '../Icon'
import { VisuallyHidden } from '../VisuallyHidden'

export interface DialogProps extends PropsWithChildren<DialogHTMLAttributes<HTMLDialogElement>> {
actions?: ReactNode
}

const DialogClose = forwardRef(({ ...restProps }, ref: ForwardedRef<HTMLButtonElement>) => (
<button aria-label="close" {...restProps} ref={ref} className="amsterdam-dialog__close" formNoValidate>
<VisuallyHidden>Sluiten</VisuallyHidden>
<Icon svg={CloseIcon} size="level-5" />
</button>
))

export const Dialog = forwardRef(
({ children, className, title, actions, ...restProps }: DialogProps, ref: ForwardedRef<HTMLDialogElement>) => (
<dialog {...restProps} ref={ref} className={clsx('amsterdam-dialog', className)}>
<form method="dialog" className="amsterdam-dialog__form">
<header className="amsterdam-dialog__header">
<span className="amsterdam-dialog__title">{title}</span>
<DialogClose />
</header>
<article className="amsterdam-dialog__article">{children}</article>
{actions && <footer className="amsterdam-dialog__footer">{actions}</footer>}
</form>
</dialog>
),
)

Dialog.displayName = 'Dialog'
DialogClose.displayName = 'DialogClose'
3 changes: 3 additions & 0 deletions packages/react/src/Dialog/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# React Modal component

[Modal documentation](../../../css/src/modal/README.md)
2 changes: 2 additions & 0 deletions packages/react/src/Dialog/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export { Dialog } from './Dialog'
export type { DialogProps } from './Dialog'
1 change: 1 addition & 0 deletions packages/react/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
*/

/* Append here */
export * from './Dialog'
export * from './Pagination'
export * from './Screen'
export * from './Switch'
Expand Down
44 changes: 44 additions & 0 deletions proprietary/tokens/src/components/amsterdam/dialog.tokens.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
{
"amsterdam": {
"dialog": {
"background-color": { "value": "{amsterdam.color.primary-white}" },
"border": { "value": "0" },
"max-inline-size": { "value": "min(87.69vw, 45rem)" },
"title": {
"color": { "value": "{amsterdam.color.primary-black}" },
"font-family": { "value": "{amsterdam.typography.font-family}" },
"font-weight": { "value": "{amsterdam.typography.font-weight.bold}" },
"line-height": { "value": "{amsterdam.typography.text-level.5.line-height}" },
"narrow": {
"font-size": { "value": "{amsterdam.typography.text-level.5.narrow.font-size}" }
},
"wide": {
"font-size": { "value": "{amsterdam.typography.text-level.5.wide.font-size}" }
}
},
"backdrop": {
"background": { "value": "#0006" }
},
"close": {
"background-color": { "value": "transparent" },
"fill": { "value": "{amsterdam.color.primary-black}" },
"hover": {
"fill": { "value": "{amsterdam.color.primary-blue}" }
}
},
"form": {
"gap": { "value": "1.5rem" },
"padding-block": { "value": "clamp(1.5rem, calc(1.5rem + ((1vw - 0.5337rem) * 2.1448)), 2.5rem)" },
"padding-inline": { "value": "clamp(1.5rem, calc(1.5rem + ((1vw - 0.5337rem) * 2.1448)), 2.5rem)" },
"max-block-size": { "value": "75vh" }
},
"article": {
"padding-inline-end": { "value": "1.5rem" }
},
"footer": {
"gap": { "value": "1rem" },
"padding-block": { "value": "1.5rem 0" }
}
}
}
}
2 changes: 2 additions & 0 deletions storybook/storybook-react/config/preview.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@ import '@amsterdam/design-system-assets/font/index.css'
import '@amsterdam/design-system-css/dist/index.css'
import { viewports } from './viewports'

import './story.css'
alimpens marked this conversation as resolved.
Show resolved Hide resolved

export const parameters = {
actions: {
argTypesRegex: '^on[A-Z].*',
Expand Down
5 changes: 5 additions & 0 deletions storybook/storybook-react/config/story.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
/* Workaround to mimic dark background for dialog Canvas blocks */
.amsterdam-dialog-story {
background-color: #333 !important;
alimpens marked this conversation as resolved.
Show resolved Hide resolved
transition: background-color 0.3s;
}
19 changes: 19 additions & 0 deletions storybook/storybook-react/src/Dialog/Dialog.docs.mdx
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import { Canvas, Controls, Markdown, Meta, Primary } from "@storybook/blocks";
import * as DialogStories from "./Dialog.stories.tsx";
import README from "../../../../packages/css/src/dialog/README.md?raw";

<Meta of={DialogStories} />

<Markdown>{README}</Markdown>

<Primary />

<Controls />

## Scroll Content

<Canvas of={DialogStories.ScrollContent} className="amsterdam-dialog-story" />

## No Actions

<Canvas of={DialogStories.NoActions} className="amsterdam-dialog-story" />
alimpens marked this conversation as resolved.
Show resolved Hide resolved
Loading