Skip to content
This repository was archived by the owner on Feb 10, 2025. It is now read-only.

Commit 5e3cba1

Browse files
nijoe1hussedev
andauthored
creates modular toast component (#39)
* creates modular toast component * chore: extended tailwind theme * chore: toast - changed styles and added override of close button * removed shadcn toast * seperated toast and toaster migrated story to components/toaster * fixed interface nicer --------- Co-authored-by: Hussein Martinez <husse.dev@gmail.com>
1 parent 438dbd1 commit 5e3cba1

File tree

13 files changed

+549
-158
lines changed

13 files changed

+549
-158
lines changed

src/assets/icons/index.ts

+4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,11 @@
11
// Status Icons
22
export { default as CheckIcon } from "./check.svg?react";
3+
// Solid Check Icon
4+
export { default as CheckSolidIcon } from "./solid/check.svg?react";
35
export { default as ExclamationCircleIcon } from "./exclamationCircle.svg?react";
46
export { default as XIcon } from "./x.svg?react";
7+
// Solid X Icon
8+
export { default as XSolidIcon } from "./solid/x.svg?react";
59

610
// Time Icons
711
export { default as ClockIcon } from "./clock.svg?react";

src/assets/icons/solid/check.svg

+3
Loading

src/assets/icons/solid/x.svg

+3
Loading

src/components/Toaster/Toaster.mdx

+87
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
import { Meta } from "@storybook/blocks";
2+
3+
import * as ToasterStories from "./Toaster.stories";
4+
5+
<Meta of={ToasterStories} />
6+
7+
# Toaster Component
8+
9+
The `Toaster` component works with the `useToast` hook to display toast notifications in your app.
10+
11+
## Usage
12+
13+
1. **Add the Toaster**: Place the `Toaster` at the root of your app.
14+
15+
```tsx
16+
import { Toaster } from "./Toaster";
17+
18+
function App() {
19+
return (
20+
<>
21+
<Toaster />
22+
</>
23+
);
24+
}
25+
```
26+
27+
2. **Trigger a Toast**: Use the `useToast` hook in your components.
28+
29+
```tsx
30+
import { useToast } from "@/hooks/use-toast";
31+
import { Button } from "@/primitives/Button";
32+
33+
function TriggerButton() {
34+
const { toast } = useToast();
35+
36+
const triggerToast = () => {
37+
toast({
38+
status: "error", // Variant of the toast (success, error, warning, info)
39+
description: "Something went wrong! Please try again.", // Message content
40+
timeout: 5000, // Duration before auto-dismissal
41+
toastPosition: "bottom-right", // Where the toast appears on screen
42+
toastSize: "large", // Size of the toast (small, medium, large)
43+
descriptionSize: "large", // Size of the description text
44+
toastCloseVariant: "alwaysVisible", // Style for the close action
45+
});
46+
};
47+
}
48+
```
49+
50+
# ToasterStories Component
51+
52+
The `ToasterStories` component is a flexible UI element used to display brief messages to users. It can be customized in various ways to suit different needs.
53+
54+
## Toast Component Variations
55+
56+
### Status Variations
57+
58+
- **Success**: Indicates a successful operation.
59+
- **Error**: Represents an error or failure.
60+
- **Info**: Provides informational messages. `(WIP)`
61+
- **Warning**: Alerts the user to a potential issue. `(WIP)`
62+
63+
### Description
64+
65+
- A string that provides additional context or information about the toast message.
66+
67+
### Timeout
68+
69+
- An optional number specifying how long the toast should be visible before automatically dismissing.
70+
71+
### Position Variations (`toastPosition`)
72+
73+
- Determines where on the viewport the toast will appear. The position is defined by the keys of the `viewportVariants.variants.position` object.
74+
75+
### Description Size Variations (`descriptionSize`)
76+
77+
- Specifies the size of the description text, based on the keys of the `toastDescriptionVariants.variants.size` object.
78+
79+
### Toast Size Variations (`toastSize`)
80+
81+
- Defines the overall size of the toast, based on the keys of the `toastVariants.variants.size` object.
82+
83+
### Toast Close Variant (`toastCloseVariant`)
84+
85+
- Specifies the style of the close button, based on the keys of the `toastCloseVariants.variants.variant` object.
86+
87+
These variations allow the `Toast` component to be highly customizable, providing flexibility in both appearance and behavior to suit different use cases.
+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
// Toast.stories.tsx
2+
import { Meta } from "@storybook/react";
3+
4+
import { useToast } from "@/hooks/use-toast";
5+
import { Button } from "@/primitives/Button";
6+
7+
import { Toaster } from "./Toaster";
8+
9+
export default {
10+
title: "components/Toaster",
11+
decorators: [
12+
(Story) => (
13+
<>
14+
<Story />
15+
<Toaster />
16+
</>
17+
),
18+
],
19+
} as Meta;
20+
21+
export const SuccessToast = () => {
22+
const { toast } = useToast();
23+
24+
const showToast = () => {
25+
toast({
26+
status: "success",
27+
description: "Your evaluation has been saved",
28+
timeout: 5000,
29+
});
30+
};
31+
return <Button onClick={showToast} variant="primary" value="Show Default Success Toast" />;
32+
};
33+
34+
export const ErrorToast = () => {
35+
const { toast } = useToast();
36+
37+
const showToast = () => {
38+
toast({
39+
status: "error",
40+
description: "Error: Your evaluation has not been saved. Please try again.",
41+
timeout: 5000,
42+
});
43+
};
44+
45+
return <Button onClick={showToast} variant="primary" value="Show Default Error Toast" />;
46+
};
47+
48+
// You can add more stories for different positions and variants as needed.
49+
50+
export const SuccessToastTopLeft = () => {
51+
const { toast } = useToast();
52+
53+
const showToast = () => {
54+
toast({
55+
status: "success",
56+
description: "Your evaluation has been saved",
57+
timeout: 5000,
58+
toastPosition: "top-left",
59+
});
60+
};
61+
return <Button onClick={showToast} variant="primary" value="Show Success Toast Top Left" />;
62+
};
63+
64+
export const ErrorToastTopRight = () => {
65+
const { toast } = useToast();
66+
67+
const showToast = () => {
68+
toast({
69+
status: "error",
70+
description: "Error: Your evaluation has not been saved. Please try again.",
71+
timeout: 5000,
72+
toastPosition: "top-right",
73+
});
74+
};
75+
return <Button onClick={showToast} variant="primary" value="Show Error Toast Top Right" />;
76+
};
77+
78+
export const SuccessToastBottomLeft = () => {
79+
const { toast } = useToast();
80+
81+
const showToast = () => {
82+
toast({
83+
status: "success",
84+
description: "Your evaluation has been saved",
85+
timeout: 5000,
86+
toastPosition: "bottom-left",
87+
});
88+
};
89+
return <Button onClick={showToast} variant="primary" value="Show Success Toast Bottom Left" />;
90+
};
91+
92+
export const ErrorToastTopCenter = () => {
93+
const { toast } = useToast();
94+
95+
const showToast = () => {
96+
toast({
97+
status: "error",
98+
description: "Error: Your evaluation has not been saved. Please try again.",
99+
timeout: 5000,
100+
toastPosition: "top-center",
101+
});
102+
};
103+
return <Button onClick={showToast} variant="primary" value="Show Error Toast Bottom Right" />;
104+
};

src/components/Toaster/Toaster.tsx

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// Toast.tsx
2+
import { match } from "ts-pattern";
3+
4+
import { useToast, ToasterToast } from "@/hooks/use-toast";
5+
import { Icon, IconType } from "@/primitives/Icon";
6+
import {
7+
Toast,
8+
ToastProvider,
9+
ToastViewport,
10+
type viewportVariants,
11+
} from "@/primitives/Toast/Toast";
12+
13+
const Toaster = () => {
14+
const { toasts } = useToast();
15+
16+
// Group toasts by their toastPosition
17+
const toastsByPosition = toasts.reduce(
18+
(acc, toast) => {
19+
const position = (toast.toastPosition || "bottom-right") as string;
20+
if (!acc[position]) {
21+
acc[position] = [];
22+
}
23+
acc[position].push(toast);
24+
return acc;
25+
},
26+
{} as Record<string, ToasterToast[]>,
27+
);
28+
29+
return (
30+
<ToastProvider>
31+
{Object.entries(toastsByPosition).map(([position, toasts]) => (
32+
<ToastViewport
33+
key={position}
34+
position={position as keyof typeof viewportVariants.variants.position}
35+
>
36+
{toasts.map((toast) => {
37+
const ToastIcon = match(toast.status)
38+
.with("success", () => (
39+
<Icon type={IconType.SOLID_CHECK} className="size-5 rounded-full" />
40+
))
41+
.with("error", () => <Icon type={IconType.SOLID_X} className="size-5 rounded-full" />)
42+
// .with("info", () => (
43+
// <Icon type={IconType.SOLID_INFO} className="size-5 rounded-full" />
44+
// ))
45+
// .with("warning", () => (
46+
// <Icon type={IconType.SOLID_WARNING} className="size-5 rounded-full" />
47+
// ))
48+
.otherwise(() => <Icon type={IconType.SOLID_X} className="size-5 rounded-full" />);
49+
return (
50+
<Toast
51+
toast={{
52+
...toast,
53+
icon: ToastIcon ?? IconType.SOLID_X,
54+
}}
55+
/>
56+
);
57+
})}
58+
</ToastViewport>
59+
))}
60+
</ToastProvider>
61+
);
62+
};
63+
64+
export { Toaster };

src/hooks/use-toast.ts

+25-6
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,19 @@
11
import * as React from "react";
22

3-
import type { ToastActionElement, ToastProps } from "@/ui-shadcn/toast";
3+
import type { ToastActionElement } from "@/primitives/Toast/Toast";
4+
import { ToastProps } from "@/primitives/Toast/types";
45

56
const TOAST_LIMIT = 1;
67
const TOAST_REMOVE_DELAY = 1000000;
78

8-
type ToasterToast = ToastProps & {
9+
export interface ToasterToast extends ToastProps {
910
id: string;
11+
icon?: JSX.Element;
12+
open?: boolean;
13+
onOpenChange?: (open: boolean) => void;
1014
title?: React.ReactNode;
11-
description?: React.ReactNode;
1215
action?: ToastActionElement;
13-
};
16+
}
1417

1518
const actionTypes = {
1619
ADD_TOAST: "ADD_TOAST",
@@ -121,7 +124,7 @@ export const reducer = (state: State, action: Action): State => {
121124
}
122125
};
123126

124-
const listeners: Array<(state: State) => void> = [];
127+
const listeners: ((state: State) => void)[] = [];
125128

126129
let memoryState: State = { toasts: [] };
127130

@@ -134,7 +137,16 @@ function dispatch(action: Action) {
134137

135138
type Toast = Omit<ToasterToast, "id">;
136139

137-
function toast({ ...props }: Toast) {
140+
function toast({
141+
status,
142+
description,
143+
timeout = 5000,
144+
toastPosition = "bottom-right",
145+
descriptionSize = "medium",
146+
toastSize = "medium",
147+
toastCloseVariant = "alwaysVisible",
148+
...props
149+
}: Toast) {
138150
const id = genId();
139151

140152
const update = (props: ToasterToast) =>
@@ -149,6 +161,13 @@ function toast({ ...props }: Toast) {
149161
toast: {
150162
...props,
151163
id,
164+
status,
165+
description,
166+
timeout,
167+
toastPosition,
168+
descriptionSize,
169+
toastSize,
170+
toastCloseVariant,
152171
open: true,
153172
onOpenChange: (open) => {
154173
if (!open) dismiss();

src/primitives/Icon/Icon.tsx

+6
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
// Status Icons
22
import {
33
CheckIcon,
4+
CheckSolidIcon,
45
ClockIcon,
56
ExclamationCircleIcon,
67
SparklesIcon,
78
XIcon,
9+
XSolidIcon,
810
CalendarIcon,
911
VerifiedBadgeIcon,
1012
} from "@/assets/icons";
@@ -23,10 +25,12 @@ import {
2325
export enum IconType {
2426
// Status Icons
2527
CHECK = "check",
28+
SOLID_CHECK = "solid-check",
2629
CLOCK = "clock",
2730
EXCLAMATION_CIRCLE = "exclamation-circle",
2831
SPARKLES = "sparkles",
2932
X = "x",
33+
SOLID_X = "solid-x",
3034
CALENDAR = "calendar",
3135
VERIFIEDBADGE = "verifiedBadge",
3236

@@ -50,10 +54,12 @@ export type IconProps = React.SVGProps<SVGSVGElement> & {
5054

5155
const iconComponents: Record<IconProps["type"], React.FC<React.SVGProps<SVGSVGElement>>> = {
5256
check: CheckIcon,
57+
"solid-check": CheckSolidIcon,
5358
clock: ClockIcon,
5459
"exclamation-circle": ExclamationCircleIcon,
5560
sparkles: SparklesIcon,
5661
x: XIcon,
62+
"solid-x": XSolidIcon,
5763
twitter: TwitterIcon,
5864
github: GithubIcon,
5965
eth: ETHIcon,

0 commit comments

Comments
 (0)