Skip to content

Commit

Permalink
docs(ToastNotification): add interactive implementation example
Browse files Browse the repository at this point in the history
- demo autodismiss behavior with stack of notifications
- use state to track
  • Loading branch information
booc0mtaco committed Aug 20, 2024
1 parent db6203e commit 56b2998
Show file tree
Hide file tree
Showing 2 changed files with 142 additions and 63 deletions.
63 changes: 0 additions & 63 deletions src/components/ToastNotification/ToastNotification.stories.ts

This file was deleted.

142 changes: 142 additions & 0 deletions src/components/ToastNotification/ToastNotification.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
import { Transition } from '@headlessui/react';
import type { StoryObj, Meta } from '@storybook/react';
import React from 'react';

import type { ComponentProps } from 'react';

import { ToastNotification } from './ToastNotification';

import Button from '../Button';

export default {
title: 'Components/ToastNotification',
component: ToastNotification,
parameters: {
layout: 'centered',
badges: ['intro-1.0', 'current-2.0'],
},
argTypes: {
onDismiss: { action: 'trigger dismiss' },
timeout: { table: { disable: true } },
},
args: {
title: "You've got a temporary notification!",
className: 'w-96',
},
} as Meta<Args>;

type Args = ComponentProps<typeof ToastNotification>;
type Story = StoryObj<Args>;

export const Default: Story = {};

export const Favorable: Story = {
args: {
status: 'favorable',
},
};

/**
* Notifications can have different status, to indicate errors or destructive actions have completed.
*/
export const Critical: Story = {
args: {
status: 'critical',
},
};

/**
* We can restrict the ability to dismiss the notification by not specifying the `onDismiss` method.
*/
export const NotDismissable: Story = {
args: {
...Default.args,
onDismiss: undefined,
},
};

/**
* Tooltips can be instructed to auto-close after a certain period. After the timeout, the component will call the defined
* `onDismiss` method. The behavior of the dissmisal is left up to the user, which allows for complete control.
*/
export const AutoDismiss: Story = {
args: {
...Default.args,
dissmissType: 'auto',
timeout: 500,
onDismiss: () => console.log('trigger onDismiss'),
},
};

let toastId = 0;
const ToastNotificationManager = (args: Args) => {
const [toasts, setToasts] = React.useState<
{ id: number | string; text: string; show?: boolean }[]
>([]);

// TODO: clean up `toasts` after .show is set to false (using useEffect?)

return (
<div className="flex h-[90vh] w-[90vw] items-center justify-center">
<Button
onClick={() => {
setToasts([
...toasts,
{
id: toastId++,
text: 'New Toast',
show: true,
},
]);
}}
>
Add Toast
</Button>
<div
className="absolute bottom-0 left-0 flex flex-col gap-size-2"
id="toast-container"
>
{toasts.map((toast) => (
<Transition
appear
enter="transition-all duration-300"
enterFrom="opacity-0 transform-gpu scale-0"
enterTo="opacity-100 transform-gpu scale-100"
key={toast.id}
leave="ease-in-out transition-all duration-300"
leaveFrom="opacity-100 transform-gpu translate-x-[0px]"
leaveTo="opacity-0 transform-gpu translate-x-[-100%]"
show={toast.show}
>
<ToastNotification
{...args}
dissmissType="auto"
onDismiss={() => {
setToasts(
toasts.map((thisToast) => {
return thisToast.id === toast.id
? { ...thisToast, show: false }
: thisToast;
}),
);
}}
title={'You got a new toast: ' + toast.text + toast.id}
/>
</Transition>
))}
</div>
</div>
);
};

/**
* This implementation example shows how you can use toasts with state to ghandle multiple, stacking notifications
*/
export const IEDismissingToasts: Story = {
render: (args) => <ToastNotificationManager {...args} />,
parameters: {
// For interactive use, low value in snap testing again since already covered in other stories.
chromatic: { disableSnapshot: true },
snapshot: { skip: true },
},
};

0 comments on commit 56b2998

Please sign in to comment.