Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/temzasse/react-modal-sheet
Flexible bottom sheet component built with Framer Motion to provide buttery smooth UX while keeping accessibility in mind ๐ช
https://github.com/temzasse/react-modal-sheet
accessibility bottom-sheet framer-motion modal reactjs
Last synced: 2 days ago
JSON representation
Flexible bottom sheet component built with Framer Motion to provide buttery smooth UX while keeping accessibility in mind ๐ช
- Host: GitHub
- URL: https://github.com/temzasse/react-modal-sheet
- Owner: Temzasse
- License: mit
- Created: 2020-05-17T20:06:18.000Z (over 4 years ago)
- Default Branch: main
- Last Pushed: 2024-10-27T20:51:01.000Z (3 months ago)
- Last Synced: 2024-10-29T14:50:48.233Z (3 months ago)
- Topics: accessibility, bottom-sheet, framer-motion, modal, reactjs
- Language: TypeScript
- Homepage: https://temzasse.github.io/react-modal-sheet/
- Size: 42.6 MB
- Stars: 840
- Watchers: 4
- Forks: 80
- Open Issues: 22
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
ยท
Flexible bottom sheet component for your React apps
ยท
| ![](media/1.gif) | ![](media/2.gif) | ![](media/3.gif) | ![](media/4.gif) |
| :--------------: | :--------------: | :--------------: | :--------------: |## ๐ฆ Installation
```sh
npm install react-modal-sheet
```### Peer dependencies
The gestures and animations are handled by the excellent [Motion](https://motion.dev/) library so before you can start using this library you need to install `motion`:
```sh
npm install motion
```> [!IMPORTANT]
> If you are still using the old `framer-motion` package you need to upgrade to `motion` as `react-modal-sheet` is now built on top of the new `motion` package. **Version `v3.5.0` and older are compatible with `framer-motion`**.---
๐ Table of contents
- [Usage](#-usage)
- [Props](#%EF%B8%8F-props)
- [Methods and properties](#%EF%B8%8F-methods-and-properties)
- [Compound Components](#-compound-components)
- [Advanced usage](#-advanced-usage)
- [Customization](#-customization)
- [Accessibility](#%EF%B8%8F-accessibility)
- [Troubleshooting](#-troubleshooting)---
## ๐ป Usage
```tsx
import { Sheet } from 'react-modal-sheet';
import { useState } from 'react';function Example() {
const [isOpen, setOpen] = useState(false);return (
<>
setOpen(true)}>Open sheetsetOpen(false)}>
{/* Your sheet content goes here */}
>
);
}
```The `Sheet` component follows the [Compound Component pattern](https://kentcdodds.com/blog/compound-components-with-react-hooks) in order to provide a flexible yet powerful API for creating highly customizable bottom sheet components.
Since the final bottom sheet is composed from smaller building blocks (`Container`, `Content`, `Scroller`, `Header`, and `Backdrop`) you are in total control over the rendering output. So for example, if you don't want to have any backdrop in your sheet then you can just skip rendering it instead of passing some prop like `renderBackdrop={false}` to the main sheet component. Cool huh? ๐
Also, by constructing the sheet from smaller pieces makes it easier to apply any necessary accessibility related properties to the right components without requiring the main sheet component to be aware of them. You can read more about accessibility in the [Accessibility](#Accessibility) section.
## ๐๏ธ Props
| Name | Required | Default | Description |
| ----------------------- | -------- | ------------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
| `children` | yes | | Use `Sheet.Container/Content/Header/Backdrop` to compose your bottom sheet. |
| `isOpen` | yes | | Boolean that indicates whether the sheet is open or not. |
| `onClose` | yes | | Callback fn that is called when the sheet is closed by the user. |
| `disableDrag` | no | false | Disable drag for the whole sheet. |
| `disableScrollLocking` | no | false | Disable scroll locking for the `body` element while sheet is open. Can be useful if you face issues with input elements and the iOS virtual keyboard. See related [issue](https://github.com/Temzasse/react-modal-sheet/issues/135). |
| `detent` | no | `'full-height'` | The [detent](https://developer.apple.com/design/human-interface-guidelines/components/presentation/sheets#ios-ipados) in which the sheet should be in when opened. Available values: `'full-height'` or `'content-height'`. |
| `onOpenStart` | no | | Callback fn that is called when the sheet opening animation starts. |
| `onOpenEnd` | no | | Callback fn that is called when the sheet opening animation is completed. |
| `onCloseStart` | no | | Callback fn that is called when the sheet closing animation starts. |
| `onCloseEnd` | no | | Callback fn that is called when the sheet closing animation is completed. |
| `onSnap` | no | | Callback fn that is called with the current snap point index when the sheet snaps to a new snap point. Requires `snapPoints` prop. |
| `snapPoints` | no | | Eg. `[-50, 0.5, 100, 0]` - where positive values are pixels from the bottom of the screen and negative from the top. Values between 0-1 represent percentages, eg. `0.5` means 50% of window height from the bottom of the sceen. |
| `initialSnap` | no | 0 | Initial snap point when sheet is opened (index from `snapPoints`). |
| `rootId` | no | | The id of the element where the main app is mounted, eg. "root". Enables iOS modal effect. |
| `tweenConfig` | no | `{ ease: 'easeOut', duration: 0.2 }` | Overrides the config for the sheet [tween](https://motion.dev/docs/react-transitions#tween) transition when the sheet is opened, closed, or snapped to a point. |
| `mountPoint` | no | `document.body` | HTML element that should be used as the mount point for the sheet. |
| `prefersReducedMotion` | no | false | Skip sheet animations (sheet instantly snaps to desired location). |
| `dragVelocityThreshold` | no | 500 | How fast the sheet must be flicked down to close. Higher values make the sheet harder to close. |
| `dragCloseThreshold` | no | 0.6 | The portion of the sheet which must be dragged off-screen before it will close. |## โ๏ธ Methods and properties
### `snapTo(index)`
Imperative method that can be accessed via a ref for snapping to a snap point in given index.
```tsx
import { Sheet, SheetRef } from 'react-modal-sheet';
import { useState, useRef } from 'react';function Example() {
const [isOpen, setOpen] = useState(false);
const ref = useRef(null);
const snapTo = (i: number) => ref.current?.snapTo(i);return (
<>
setOpen(true)}>Open sheet{/* Opens to 400 since initial index is 1 */}
setOpen(false)}
snapPoints={[600, 400, 100, 0]}
initialSnap={1}
onSnap={(snapIndex) =>
console.log('> Current snap point index:', snapIndex)
}
>
snapTo(0)}>Snap to index 0
snapTo(1)}>Snap to index 1
snapTo(2)}>Snap to index 2
snapTo(3)}>Snap to index 3
>
);
}
```### Motion value `y`
The `y` value is an internal `MotionValue` that represents the distance to the top most position of the sheet when it is fully open. So for example the `y` value is zero when the sheet is completely open.
Similarly to the `snapTo` method the `y` value can be accessed via a ref.
The `y` value can be useful for certain situtation eg. when you want to combine snap points with scrollable sheet content and ensure that the content stays properly scrollable in any snap point. Below you can see a simplified example of this situation and for a more detailed example take a look at the [ScrollableSnapPoints](example/components/ScrollableSnapPoints.tsx) component in the example app.
```tsx
import { Sheet, SheetRef } from 'react-modal-sheet';
import { useState, useRef } from 'react';function Example() {
const [isOpen, setOpen] = useState(false);
const ref = useRef(null);return (
<>
setOpen(true)}>Open sheetsetOpen(false)}
snapPoints={[600, 400, 100, 0]}
initialSnap={1}
>
{/**
* Since `Sheet.Content` is a `motion.div` it can receive motion values
* in it's style prop which allows us to utilise the exposed `y` value.
*
* By syncing the padding bottom with the `y` motion value we introduce
* an offset that ensures that the sheet content can be scrolled all the
* way to the bottom in every snap point.
*/}
{/* Some content here that makes the sheet content scrollable */}
>
);
}
```### Detents
By default the sheet will take the full height of the page minus top padding and safe area inset. If you want the sheet height to be based on it's content you can pass `detent="content-height"` prop to the `Sheet` component:
```tsx
function Example() {
const [isOpen, setOpen] = useState(false);return (
setOpen(false)}
detent="content-height"
>
Some content
);
}
```If the sheet height changes dynamically the sheet will grow until it hits the maximum full height after which it becomes scrollable.
It is possible to use snap points with `detent="content-height"` **but** the snap points are restricted by the content height. For example if one of the snap points is 800px and the sheet height is only 700px then snapping to the 800px snap point would only snap to 700px since otherwise the sheet would become detached from the bottom.
> โน๏ธ If you are wondering where the term `detent` comes from it's from Apple's [Human Interface Guidelines](https://developer.apple.com/design/human-interface-guidelines/components/presentation/sheets#ios-ipados).
## ๐งฉ Compound Components
### `Sheet`
Sheet is the root element that wraps the whole sheet. It renders a fixed positioned wrapper element that covers the whole screen to contain the actual sheet that is animated in (don't worry the root element is not interactive when the sheet is closed). All sheet compound components should be rendered within `Sheet`.
> ๐ฅ Rendered element: `motion.div`.
### `Sheet.Container`
Sheet container is positioned above the sheet backdrop and by default adds a small shadow and rounded corners to the sheet. `Sheet.Content` and `Sheet.Header` should be rendered inside `Sheet.Container`.
> ๐ฅ Rendered element: `motion.div`.
### `Sheet.Header`
Sheet header acts as a drag target and has a dragging direction indicator. Rendering any children inside `Sheet.Header` replaces the default header.
> ๐ฅ Rendered element: `motion.div`.
#### Header props
| Name | Required | Default | Description |
| ------------- | -------- | ------- | ---------------------------------- |
| `disableDrag` | no | false | Disable drag for the sheet header. |### `Sheet.Content`
Sheet content acts as a drag target and can be used in conjunction with `Sheet.Scroller` to make sure that content which doesn't fit inside the sheet becomes scrollable.
> ๐ฅ Rendered element: `motion.div`.
#### Content props
| Name | Required | Default | Description |
| ------------- | -------- | ------- | ----------------------------------- |
| `disableDrag` | no | false | Disable drag for the sheet content. |### `Sheet.Scroller`
Sheet scroller can be used to make the whole sheet content or parts of it scrollable in a way that drag gestures are properly disabled and enabled based on the scroll state. See the [Scrolling on touch devices](#scrolling-on-touch-devices) section for more details.
> ๐ฅ Rendered element: `motion.div`.
#### Scroller props
| Name | Required | Default | Description |
| ------------- | -------- | ------- | -------------------------------------------------------------------------------------------------------------------------------------- |
| `draggableAt` | no | `"top"` | Should the drag be enabled when the element is scrolled either to the top, bottom, or both. Available values: `top`, `bottom`, `both`. |### `Sheet.Backdrop`
Sheet backdrop is a translucent overlay that helps to separate the sheet from it's background. By default the backdrop doesn't have any interaction attached to it but if you, for example, want to close the sheet when the backdrop is clicked you can provide tap handler to it which will change the rendered element from `div` to `button`.
**โ ๏ธ Note:** as the element is a motion component you need to use [`onTap`](https://motion.dev/docs/react-gestures#tap) instead of `onClick` if you want to add a click handler to it.
> ๐ฅ Rendered element: `motion.div` or `motion.button`.
## โจ Advanced usage
### โจ๏ธ Virtual keyboard avoidance
A common problem with React Modal Sheet is that since the sheet is rendered as a fixed positioned element it doesn't move when the virtual keyboard is opened which can lead to the keyboard covering the input elements.
Handling the virtual keyboard on mobile devices can be quite tricky. It doesn't help that the Web doesn't yet have widely supported APIs for working with the virtual keyboard.
**โ ๏ธ Note: This library does not provide any official built-in solution for keyboard avoidance!**
This is because it is not possible for the library to know how your sheet content is structured and how the sheet should behave when the keyboard is opened.
However, there are some strategies that you can use to handle the virtual keyboard:
1. Design your sheet content in a way that the input elements are positioned at the top of the sheet so that they are not covered by the keyboard. This is the easiest solution and if you look at popular mobile apps you can see that they often have input elements at the top of the screen exactly for this reason.
2. Listen to changes in the `visualViewport`, see [Visual Viewport API](https://developer.mozilla.org/en-US/docs/Web/API/Visual_Viewport_API), and change styles like padding on the sheet content to move the input up when the keyboard is opened. See the `AvoidKeyboard` component [code](example/components/AvoidKeyboard.tsx) and [demo](https://temzasse.github.io/react-modal-sheet/#/avoid-keyboard) in the example app for a simple example of this. The example is based on a custom `useAnimatedVirtualKeyboard` hook which you can copy to your project at your own risk. **Note that at least on iOS there is a noticeable delay when the keyboard is opened and when the `resize` event is fired for the `visualViewport`**.
3. Use the experimental [VirtualKeyboard API](https://developer.mozilla.org/en-US/docs/Web/API/VirtualKeyboard_API) to detect when the virtual keyboard is opened and adjust the sheet content accordingly. **Note that this API is not yet widely supported in browsers, but it will most likely produce the best result.**
### ๐ฑ Scrolling on touch devices
Scrolling and dragging are **the same** gesture on touch devices which can create problems when you want to have scrollable content inside the sheet. React Modal Sheet provides a `Sheet.Scroller` component that is able to automatically disable and enable dragging inside the `Sheet.Content` component based on the scroll state. There are three modes for the `Sheet.Scroller` reflected in the `draggableAt` prop:
1. Enable dragging when the scroller is not yet scrolled - it is at the `top` position.
2. Enable dragging when the scroller is scrolled all the way to the bottom - it is at the `bottom` position.
3. Enable dragging in `both` positions.The scroller component is in-between these states when the user has scrolled only some amount. Dragging is always **disabled** in this in-between state in order to avoid it getting mixed with dragging gestures. The `Sheet.Scroller` component applies the special handling only for touch devices.
The default value for the `draggableAt` prop is `top` which should be a good default for most use cases. You shouldn't need `bottom` or `both` unless you have scrollable content inside a sheet that also has snap points.
```tsx
function ScrollableExample() {
return (
{/*...*/}
);
}
```### ๐ช iOS Modal View effect
In addition to the `Sheet.Backdrop` it's possible to apply a scaling effect to the main app element to highlight the modality of the bottom sheet. This effect mimics the [iOS Modal View](https://developer.apple.com/design/human-interface-guidelines/ios/app-architecture/modality/) presentation style to bring more focus to the sheet and add some delight to the user experience.
| ![](media/3.gif) | ![](media/4.gif) |
| :--------------: | :--------------: |To enable this effect you can provide the id of the root element where your application is mounted:
```tsx
function Example() {
return {/*...*/};
}
```> โ ๏ธ **Limitations**: Since the effect is applied to the root element it will NOT work as desired if the HTML body element is scrolled down at all. One way to avoid this is to use something like `height: 100vh;` and `overflow: auto;` on the root element to make it fill the whole screen and be scrollable instead of the body element.
## ๐จ Customization
The default styles for the `Sheet` component somewhat follows the styles of the previously mentioned iOS Modal View. However, if these default styles are not to your liking it's easy to make changes to them: you can provide a custom header or you can overwrite any style with CSS via the exposed class names.
### Custom header
Adding a custom header is as simple as providing your own header as the child component to `Sheet.Header`:
```tsx
function Example() {
return (
{/*...*/}
);
}
```### Custom styles
You can add your own styles or override the default sheet styles via the exposed class names. Note that you might need to use `!important` for style overrides since the inner styles are applied as inline styles which have higher specificity.
#### Vanilla CSS
```css
.react-modal-sheet-backdrop {
/* custom styles */
}
.react-modal-sheet-container {
/* custom styles */
}
.react-modal-sheet-header {
/* custom styles */
}
.react-modal-sheet-drag-indicator {
/* custom styles */
}
.react-modal-sheet-content {
/* custom styles */
}
```#### CSS-in-JS
```tsx
import { Sheet } from 'react-modal-sheet';
import { styled } from 'styled-components';
import { useState } from 'react';const CustomSheet = styled(Sheet)`
.react-modal-sheet-backdrop {
/* custom styles */
}
.react-modal-sheet-container {
/* custom styles */
}
.react-modal-sheet-header {
/* custom styles */
}
.react-modal-sheet-drag-indicator {
/* custom styles */
}
.react-modal-sheet-content {
/* custom styles */
}
`;function Example() {
const [isOpen, setOpen] = useState(false);return (
<>
setOpen(true)}>Open sheetsetOpen(false)}>
{/*...*/}
>
);
}
```## โฟ๏ธ Accessibility
By default, react-modal-sheet doesn't include any built-in accessibility properties in order to not make any assumptions and to support a wide range of use cases. Additionally, not including 3rd party libraries for features like focus trapping or screen reader support makes it possible to utilize any accessibility libraries that your project may already use, eg. [React Aria](https://react-spectrum.adobe.com/react-aria/getting-started.html). This also helps to reduce JS bloat by not including similar libraries multiple times in your app bundle.
The example below utilizes React Aria to achieve an accessible modal-like bottom sheet that can be closed via a button rendered inside a custom sheet header.
> โน๏ธ The example was built by following the React Aria's [useDialog](https://react-spectrum.adobe.com/react-aria/useDialog.html) documentation.
```tsx
import { Sheet } from 'react-modal-sheet';
import { useRef } from 'react';
import { useOverlayTriggerState } from 'react-stately';import {
useOverlay,
useModal,
OverlayProvider,
FocusScope,
useButton,
useDialog,
} from 'react-aria';function A11yExample() {
const sheetState = useOverlayTriggerState({});
const openButtonRef = useRef(null);
const openButton = useButton({ onPress: sheetState.open }, openButtonRef);return (
Open sheet
);
}function SheetComp({ sheetState }) {
const containerRef = useRef(null);
const dialog = useDialog({}, containerRef);
const overlay = useOverlay(
{ onClose: sheetState.close, isOpen: true, isDismissable: true },
containerRef
);const closeButtonRef = useRef(null);
const closeButton = useButton(
{ onPress: sheetState.close, 'aria-label': 'Close sheet' },
closeButtonRef
);useModal();
// In real world usage this would be a separate React component
const customHeader = (
Some title for sheet
๐ ง
);return (
<>
{customHeader}
{/*...*/}
>
);
}
```If you want to see a more real-world-like implementation you can take a look at the [Slack example](example/components/slack-message/index.tsx) and try out the related [demo](https://temzasse.github.io/react-modal-sheet/#/slack-message) (optimized for mobile).
### ๐ฉ Building a reusable sheet
In your projects it might make sense to build a reusable bottom sheet that has all the accessibility features included and can then be easily used in various places in the project. Take a look at the [A11ySheet](example/components/a11y/A11ySheet.tsx) example to get some insight on how to build such a component. By incorporating all the accessibility features inside your own reusable component you don't need to repeat them every time you want to use a bottom sheet in your app.
## ๐ Troubleshooting
### The sheet doesn't open when using `StrictMode`
If you are using React [StrictMode](https://react.dev/reference/react/StrictMode) the sheet animations might not work as expected. This seems to be an issue in the `motion` library and I haven't been able to find a good solution for it yet. You see all `motion` issues related to the `StrictMode` [here](https://github.com/motiondivision/motion/issues?q=is%3Aissue+StrictMode). Easiest solution is to just not to use `StrictMode` ๐คทโโ๏ธ