Back to Tamagui

Toast

code/tamagui.dev/data/docs/components/toast/1.8.0.mdx

1.144.410.9 KB
Original Source

Toast <Beta />

<Description>Use to show feedback to user interactions</Description>

<HeroContainer showAnimationDriverControl> <ToastDemo /> </HeroContainer>
tsx

<Highlights features={[ Automatically closes, Pause closing on hover, focus, window blur and mobile touch, Supports closing via swipe gesture, Easily animatable with Tamagui's animation drivers, Native toasts included for Android, iOS and web, ]} />

Install

For native support, run yarn add burnt to add burnt, then rebuild your React Native app. React Native requires sub-dependencies with native dependencies always be hoisted to your apps package.json and Toast relies on the amazing Burnt library by Fernando Rojo to provide its native functionality.

Anatomy

tsx
import { Toast, ToastProvider, ToastViewport } from 'tamagui' // or '@tamagui/toast'

export default () => (
  <ToastProvider>
    <Toast>
      <Toast.Title />
      <Toast.Description />
      <Toast.Action />
      <Toast.Close />
    </Toast>

    <ToastViewport />
  </ToastProvider>
)

Usage

Single Toast

tsx
export default () => {
  const [open, setOpen] = React.useState(false)
  const timerRef = React.useRef(0)

  React.useEffect(() => {
    return () => clearTimeout(timerRef.current)
  }, [])

  return (
    <YStack ai="center">
      <Button
        onPress={() => {
          setOpen(false)
          window.clearTimeout(timerRef.current)
          timerRef.current = window.setTimeout(() => {
            setOpen(true)
          }, 150)
        }}
      >
        Single Toast
      </Button>
      <Toast
        onOpenChange={setOpen}
        open={open}
        animation="100ms"
        enterStyle={{ x: -20, opacity: 0 }}
        exitStyle={{ x: -20, opacity: 0 }}
        opacity={1}
        x={0}
      >
        <Toast.Title>Subscribed!</Toast.Title>
        <Toast.Description>We'll be in touch.</Toast.Description>
      </Toast>
    </YStack>
  )
}

Duplicate Toasts

tsx
export default () => {
  const [savedCount, setSavedCount] = React.useState(0)

  return (
    <YStack ai="center">
      <Button
        onPress={() => {
          setSavedCount((old) => old + 1)
        }}
      >
        Show toast
      </Button>
      {[...Array(savedCount)].map((_, index) => (
        <Toast
          key={index}
          animation="100ms"
          enterStyle={{ x: -20, opacity: 0 }}
          exitStyle={{ x: -20, opacity: 0 }}
          opacity={1}
          x={0}
        >
          <Toast.Title>Subscribed!</Toast.Title>
          <Toast.Description>We'll be in touch.</Toast.Description>
        </Toast>
      ))}
    </YStack>
  )
}

Using createToast

tsx
import { Button } from 'tamagui' // or '@tamagui/button'
import { Toast, ToastProvider, createToast } from 'tamagui' // or '@tamagui/toast'

export const { ImperativeToastProvider, useToast } = createToast()

export default () => (
  <ToastProvider>
    <ImperativeToastProvider>
      <CurrentToast />
      <MyPage />
    </ImperativeToastProvider>

    <ToastViewport />
  </ToastProvider>
)

const CurrentToast = () => {
  const { currentToast } = useToast()

  if (!currentToast) return
  return (
    <Toast key={currentToast.id}>
      <Toast.Title>{currentToast.title}</Toast.Title>
      <Toast.Description>{currentToast.message}</Toast.Description>
    </Toast>
  )
}

const MyPage = () => {
  const { show } = useToast()

  return (
    <Button onPress={() => show('Done!', { message: 'Form submitted successfully.' })}>
      Show Toast
    </Button>
  )
}

API

ToastProvider

Your toasts should be wrapped within a ToastProvider. This is usually done at the root of your application.

<PropsTable data={[ { name: 'label', required: false, type: 'string', description: An author-localized label for each toast. Used to help screen reader users associate the interruption with a toast., default: 'Notification', }, { name: 'duration', required: false, type: 'number', description: Time in milliseconds that each toast should remain visible for. This could be overwritten at the toast level as well., default: 5000, }, { name: 'swipeDirection', required: false, type: 'SwipeDirection', description: Direction of pointer swipe that should close the toast., default: 'right', }, { name: 'swipeThreshold', required: false, type: 'number', description: Distance in pixels that the swipe must pass before a close is triggered., default: 50, }, { name: 'id', required: false, type: 'string', default: 'A unique generated ID', }, ]} />

ToastViewport

The portal for toasts to be directed to. Should be inside ToastProvider. Beyond Stack Props, adds:

<PropsTable data={[ { name: 'hotkey', type: 'string[]', default: "['F8']", required: false, description: The keys to use as the keyboard shortcut that will move focus to the toast viewport., }, { name: 'label', type: 'string', default: 'Notifications ({hotkey})', required: false, description: An author-localized label for the toast viewport to provide context for screen reader users when navigating page landmarks. The available \{hotkey}` placeholder will be replaced for you., }, { name: 'name', type: 'string', required: false, description: Used to reference the viewport if you want to have multiple viewports in the same provider.`, }, ]} />

Toast

Contains the Title, Description, Action and Close component. Should be inside ToastProvider. Extends Stack and adds:

<PropsTable data={[ { name: 'forceMount', type: 'boolean', required: false, description: Used to force mounting when more control is needed. Useful when controlling animation with React animation libraries., }, { name: 'type', type: "'foreground' | 'background'", required: false, description: Control the sensitivity of the toast for accessibility purposes. For toasts that are the result of a user action, choose foreground. Toasts generated from background tasks should use background., }, { name: 'duration', type: 'number', required: false, description: Time in milliseconds that toast should remain visible for. Overrides value given to \ToastProvider`., }, { name: 'defaultOpen', type: 'boolean', required: false, description: The open state of the dialog when it is initially rendered. Use when you do not need to control its open state., }, { name: 'open', type: 'boolean', required: false, description: The controlled open state of the dialog. Must be used in conjunction with `onOpenChange`., }, { name: 'onOpenChange', type: '(open: boolean): void', required: false, description: Event handler called when the open state of the dialog changes., }, { name: 'onEscapeKeyDown', type: "(): DismissableProps['onEscapeKeyDown']", required: false, description: Event handler called when the escape key is down. It can be prevented by calling `event.preventDefault`., }, { name: 'onPause', type: '(): void', required: false, description: Event handler called when the dismiss timer is paused. On web, this occurs when the pointer is moved over the viewport, the viewport is focused or when the window is blurred. On mobile, this occurs when the toast is touched., }, { name: 'onResume', type: '(): void', required: false, description: Event handler called when the dismiss timer is resumed. On web, this occurs when the pointer is moved away from the viewport, the viewport is blurred or when the window is focused. On mobile, this occurs when the toast is released., }, { name: 'onSwipeStart', type: '(event: SwipeEvent): void', required: false, description: Event handler called when starting a swipe interaction. It can be prevented by calling `event.preventDefault`., }, { name: 'onSwipeMove', type: '(event: SwipeEvent): void', required: false, description: Event handler called during a swipe interaction. It can be prevented by calling `event.preventDefault`., }, { name: 'onSwipeCancel', type: '(event: SwipeEvent): void', required: false, description: Event handler called at the cancellation of a swipe interaction. It can be prevented by calling `event.preventDefault`., }, { name: 'onSwipeEnd', type: '(event: SwipeEvent): void', required: false, description: Event handler called at the end of a swipe interaction. It can be prevented by calling `event.preventDefault`., }, { name: 'viewportName', type: 'string', required: false, description: The viewport's name to send the toast to. Used when using multiple viewports and want to forward toasts to different ones.`, default: 'default', }, ]} />

Toast.Title

Should be inside Toast. Extends SizableText.

Toast.Description

Should be inside Toast. Extends SizableText.

Toast.Close

Should be inside Toast. Extends Stack. You can pass asChild to this component and use a custom <Button> inside.

Toast.Action

Should be inside Toast. Extends Stack. You can pass asChild to this component and use a custom <Button> inside.

createToast

An alternative way to work with toasts.

<PropsTable title="Arguments" data={[ { name: 'native', type: "boolean | 'web' | 'mobile'", description: Will show a native toast if is true or is set to the current platform. On iOS, it wraps \SPIndicator` and `SPAlert`. On Android, it wraps `ToastAndroid`. On web, it wraps Notification API. Mobile's native features are handled by `burnt`.`, default: 'false', }, ]} />

This function, then returns the following:

ImperativeToastProvider

Wrap children within this provider so that they can use useToast(). Takes no params.

useToast

You may export this hook to use it throughout your app.

<PropsTable title="Returns" data={[ { name: 'currentToast', type: 'ToastData | null', description: The information about the current toast to showm such as title, message, duration, etc., }, { name: show, type: '(title: string, showOptions?: ToastOptions): void', description: 'Call it to show a new toast.', }, ]} />