Skip to content

Commit

Permalink
chore: fixes and adding draft1 of stories
Browse files Browse the repository at this point in the history
  • Loading branch information
macci001 committed Dec 25, 2024
1 parent 2a73f8a commit 3f8c09f
Show file tree
Hide file tree
Showing 9 changed files with 219 additions and 62 deletions.
29 changes: 6 additions & 23 deletions packages/components/toast/src/toast-provider.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
import {ToastOptions, ToastQueue, useToastQueue} from "@react-stately/toast";
import {ToastVariantProps} from "@nextui-org/theme";

import {ToastRegion} from "./toast-region";
import {ToastType} from "./use-toast";
import {ToastProps} from "./use-toast";

let globalToastQueue: ToastQueue<ToastType> | null = null;
let globalToastQueue: ToastQueue<ToastProps> | null = null;

interface ToastProviderProps {
maxVisibleToasts?: number;
Expand All @@ -30,31 +29,15 @@ export const ToastProvider = ({maxVisibleToasts = 5}: ToastProviderProps) => {
return <ToastRegion toastQueue={toastQueue} />;
};

export const addToast = ({
title,
description,
priority,
timeout,
...config
}: {
title: string;
description: string;
} & ToastOptions &
ToastVariantProps) => {
export const addToast = ({...props}: ToastProps & ToastOptions) => {
if (!globalToastQueue) {
return;
}

const content: ToastType = {
title,
description,
config: config,
};

const options: Partial<ToastOptions> = {
timeout,
priority,
timeout: props?.timeout,
priority: props?.priority,
};

globalToastQueue.add(content, options);
globalToastQueue.add(props, options);
};
10 changes: 4 additions & 6 deletions packages/components/toast/src/toast-region.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,13 @@ import {useToastRegion, AriaToastRegionProps} from "@react-aria/toast";
import {QueuedToast, ToastState} from "@react-stately/toast";

import Toast from "./toast";
import {ToastType} from "./use-toast";
import {ToastProps} from "./use-toast";

interface ToastRegionProps<T> extends AriaToastRegionProps {
toastQueue: ToastState<T>;
}

export function ToastRegion<T extends ToastType>({toastQueue, ...props}: ToastRegionProps<T>) {
export function ToastRegion<T extends ToastProps>({toastQueue, ...props}: ToastRegionProps<T>) {
const ref = useRef(null);
const {regionProps} = useToastRegion(props, toastQueue, ref);

Expand All @@ -20,10 +20,8 @@ export function ToastRegion<T extends ToastType>({toastQueue, ...props}: ToastRe
ref={ref}
className="fixed bottom-6 right-6 w-screen flex flex-col items-end justify-center"
>
{toastQueue.visibleToasts.map((toast: QueuedToast<ToastType>) => {
return (
<Toast key={toast.key} state={toastQueue} toast={toast} {...toast.content.config} />
);
{toastQueue.visibleToasts.map((toast: QueuedToast<ToastProps>) => {
return <Toast key={toast.key} state={toastQueue} toast={toast} {...toast.content} />;
})}
</div>
</>
Expand Down
25 changes: 22 additions & 3 deletions packages/components/toast/src/toast.tsx
Original file line number Diff line number Diff line change
@@ -1,20 +1,36 @@
import {forwardRef} from "@nextui-org/system";
import {Button, ButtonProps} from "@nextui-org/button";
import {CloseIcon} from "@nextui-org/shared-icons";
import {
CloseIcon,
DangerIcon,
InfoFilledIcon,
SuccessIcon,
WarningIcon,
} from "@nextui-org/shared-icons";
import {motion, AnimatePresence} from "framer-motion";
import {Progress} from "@nextui-org/progress";
import {cloneElement, isValidElement} from "react";

import {UseToastProps, useToast} from "./use-toast";

export interface ToastProps extends UseToastProps {}

const iconMap = {
primary: InfoFilledIcon,
secondary: InfoFilledIcon,
success: SuccessIcon,
warning: WarningIcon,
danger: DangerIcon,
} as const;

const Toast = forwardRef<"div", ToastProps>((props, ref) => {
const {
Component,
Icon,
icon,
domRef,
endContent,
closeProgressBarValue,
color,
getToastProps,
getContentProps,
getTitleProps,
Expand All @@ -33,6 +49,9 @@ const Toast = forwardRef<"div", ToastProps>((props, ref) => {
exit: {opacity: 0, y: 50},
};

const customIcon = icon && isValidElement(icon) ? cloneElement(icon, getIconProps()) : null;
const IconComponent = iconMap[color] || iconMap.primary;

return (
<AnimatePresence>
<motion.div
Expand All @@ -44,7 +63,7 @@ const Toast = forwardRef<"div", ToastProps>((props, ref) => {
>
<Component ref={domRef} {...getToastProps()}>
<main {...getContentProps()}>
<Icon {...getIconProps()} />
{customIcon || <IconComponent {...getIconProps()} />}
<div>
<div {...getTitleProps()}>{props.toast.content.title}</div>
<div {...getDescriptionProps()}>{props.toast.content.description}</div>
Expand Down
36 changes: 15 additions & 21 deletions packages/components/toast/src/use-toast.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,33 +8,26 @@ import {ReactNode, useCallback, useEffect, useMemo, useState} from "react";
import {useToast as useToastAria, AriaToastProps} from "@react-aria/toast";
import {mergeProps} from "@react-aria/utils";
import {QueuedToast, ToastState} from "@react-stately/toast";
import {InfoFilledIcon} from "@nextui-org/shared-icons";

export type ToastType = {
title: string;
description: string;
config: ToastVariantProps;
};

interface Props<T> extends HTMLNextUIProps<"div"> {
/**
* Ref to the DOM node.
*/

export interface ToastProps extends ToastVariantProps {
ref?: ReactRef<HTMLElement | null>;
toast: QueuedToast<T>;
state: ToastState<T>;
title?: string;
description?: string;
classNames?: SlotsToClasses<ToastSlots>;
/**
* Content to be displayed in the end side of the alert
*/
endContent?: ReactNode;
icon?: ReactNode;
}

interface Props<T> extends HTMLNextUIProps<"div">, ToastProps {
toast: QueuedToast<T>;
state: ToastState<T>;
}

export type UseToastProps<T = ToastType> = Props<T> &
export type UseToastProps<T = ToastProps> = Props<T> &
ToastVariantProps &
Omit<AriaToastProps<T>, "div">;

export function useToast<T extends ToastType>(originalProps: UseToastProps<T>) {
export function useToast<T extends ToastProps>(originalProps: UseToastProps<T>) {
const [props, variantProps] = mapPropsVariants(originalProps, toastTheme.variantKeys);

const [closeProgressBarValue, setCloseProgressBarValue] = useState(0);
Expand All @@ -56,7 +49,7 @@ export function useToast<T extends ToastType>(originalProps: UseToastProps<T>) {
const {ref, as, className, classNames, toast, endContent, ...otherProps} = props;

const Component = as || "div";
let Icon = InfoFilledIcon;
const icon: ReactNode = props.icon;

const domRef = useDOMRef(ref);
const baseStyles = clsx(className, classNames?.base);
Expand Down Expand Up @@ -166,11 +159,12 @@ export function useToast<T extends ToastType>(originalProps: UseToastProps<T>) {

return {
Component,
Icon,
icon,
styles,
domRef,
classNames,
closeProgressBarValue,
color: variantProps["color"],
getToastProps,
getTitleProps,
getContentProps,
Expand Down
144 changes: 140 additions & 4 deletions packages/components/toast/stories/toast.stories.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import React from "react";
import {Meta} from "@storybook/react";
import {toast} from "@nextui-org/theme";
import {cn, toast} from "@nextui-org/theme";
import {Button} from "@nextui-org/button";

import {Toast, ToastProps, ToastProvider, addToast} from "../src";
Expand Down Expand Up @@ -43,7 +43,7 @@ const Template = (args: ToastProps) => (
<Button
onPress={() => {
addToast({
title: "Title",
title: "Toast Title",
description: "Toast description",
...args,
});
Expand All @@ -54,17 +54,153 @@ const Template = (args: ToastProps) => (
</>
);

const TimeoutTemplate = (args: ToastProps) => {
return (
<>
<ToastProvider />
<Button
onPress={() => {
addToast({
title: "Toast Title",
description: "Toast Description",
timeout: 3000,
...args,
});
}}
>
Toast
</Button>
</>
);
};

const WithEndContentTemplate = (args) => {
return (
<>
<ToastProvider />
<Button
onPress={() => {
addToast({
title: "Toast Title",
description: "Toast Description",
endContent: (
<Button color="warning" size="sm" variant="flat">
Upgrade
</Button>
),
color: "warning",
variant: "faded",
...args,
});
}}
>
Toast
</Button>
</>
);
};

const CustomToastComponent = (args) => {
const color = args.color;
const colorMap = {
primary: "before:bg-primary border-primary-200 dark:border-primary-100",
secondary: "before:bg-secondary border-secondary-200 dark:border-secondary-100",
success: "before:bg-success border-success-200 dark:border-success-100",
warning: "before:bg-warning border-warning-200 dark:border-warning-100",
danger: "before:bg-danger border-danger-200 dark:border-danger-100",
};

return (
<>
<Button
color={color}
variant="bordered"
onPress={() => {
addToast({
title: "Toast Title",
description: "Toast Description",
classNames: {
base: cn([
"bg-default-50 dark:bg-background shadow-sm",
"border-1",
"relative before:content-[''] before:absolute before:z-10",
"before:left-0 before:top-[-1px] before:bottom-[-1px] before:w-1",
"rounded-l-none border-l-0",
colorMap[color],
]),
},
color: color,
});
}}
>
Toast
</Button>
</>
);
};

const CustomToastTemplate = () => {
const colors = ["primary", "secondary", "warning", "danger", "success"];

return (
<>
<ToastProvider />
<div className="flex gap-2">
{colors.map((color, idx) => (
<CustomToastComponent key={idx} color={color} />
))}
</div>
</>
);
};

export const Default = {
render: Template,
args: {
...defaultProps,
},
};

export const WithTimeout = {
export const WithIcon = {
render: Template,
args: {
...defaultProps,
timeout: 3000,
title: "Custom Icon",
icon: (
<svg height={24} viewBox="0 0 24 24" width={24}>
<g
fill="none"
stroke="currentColor"
strokeLinecap="round"
strokeLinejoin="round"
strokeMiterlimit={10}
strokeWidth={1.5}
>
<path
d="M11.845 21.662C8.153 21.662 5 21.088 5 18.787s3.133-4.425 6.845-4.425c3.692 0 6.845 2.1 6.845 4.4s-3.134 2.9-6.845 2.9z"
data-name="Stroke 1"
/>
<path d="M11.837 11.174a4.372 4.372 0 10-.031 0z" data-name="Stroke 3" />
</g>
</svg>
),
},
};

export const WithTimeout = {
render: TimeoutTemplate,
args: {
...defaultProps,
},
};

export const WithEndContent = {
render: WithEndContentTemplate,
args: {
...defaultProps,
},
};

export const CustomTemplate = {
render: CustomToastTemplate,
};
Loading

0 comments on commit 3f8c09f

Please sign in to comment.