| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161 | 
							- 'use client'
 
- import type { ReactNode } from 'react'
 
- import React, { useEffect, useState } from 'react'
 
- import { createRoot } from 'react-dom/client'
 
- import {
 
-   RiAlertFill,
 
-   RiCheckboxCircleFill,
 
-   RiCloseLine,
 
-   RiErrorWarningFill,
 
-   RiInformation2Fill,
 
- } from '@remixicon/react'
 
- import { createContext, useContext } from 'use-context-selector'
 
- import ActionButton from '@/app/components/base/action-button'
 
- import classNames from '@/utils/classnames'
 
- export type IToastProps = {
 
-   type?: 'success' | 'error' | 'warning' | 'info'
 
-   size?: 'md' | 'sm'
 
-   duration?: number
 
-   message: string
 
-   children?: ReactNode
 
-   onClose?: () => void
 
-   className?: string
 
-   customComponent?: ReactNode
 
- }
 
- type IToastContext = {
 
-   notify: (props: IToastProps) => void
 
-   close: () => void
 
- }
 
- export const ToastContext = createContext<IToastContext>({} as IToastContext)
 
- export const useToastContext = () => useContext(ToastContext)
 
- const Toast = ({
 
-   type = 'info',
 
-   size = 'md',
 
-   message,
 
-   children,
 
-   className,
 
-   customComponent,
 
- }: IToastProps) => {
 
-   const { close } = useToastContext()
 
-   // sometimes message is react node array. Not handle it.
 
-   if (typeof message !== 'string')
 
-     return null
 
-   return <div className={classNames(
 
-     className,
 
-     'fixed w-[360px] rounded-xl my-4 mx-8 flex-grow z-[9999] overflow-hidden',
 
-     size === 'md' ? 'p-3' : 'p-2',
 
-     'border border-components-panel-border-subtle bg-components-panel-bg-blur shadow-sm',
 
-     'top-0',
 
-     'right-0',
 
-   )}>
 
-     <div className={`absolute inset-0 -z-10 opacity-40 ${
 
-       (type === 'success' && 'bg-toast-success-bg')
 
-       || (type === 'warning' && 'bg-toast-warning-bg')
 
-       || (type === 'error' && 'bg-toast-error-bg')
 
-       || (type === 'info' && 'bg-toast-info-bg')
 
-     }`}
 
-     />
 
-     <div className={`flex ${size === 'md' ? 'gap-1' : 'gap-0.5'}`}>
 
-       <div className={`flex items-center justify-center ${size === 'md' ? 'p-0.5' : 'p-1'}`}>
 
-         {type === 'success' && <RiCheckboxCircleFill className={`${size === 'md' ? 'h-5 w-5' : 'h-4 w-4'} text-text-success`} aria-hidden="true" />}
 
-         {type === 'error' && <RiErrorWarningFill className={`${size === 'md' ? 'h-5 w-5' : 'h-4 w-4'} text-text-destructive`} aria-hidden="true" />}
 
-         {type === 'warning' && <RiAlertFill className={`${size === 'md' ? 'h-5 w-5' : 'h-4 w-4'} text-text-warning-secondary`} aria-hidden="true" />}
 
-         {type === 'info' && <RiInformation2Fill className={`${size === 'md' ? 'h-5 w-5' : 'h-4 w-4'} text-text-accent`} aria-hidden="true" />}
 
-       </div>
 
-       <div className={`flex py-1 ${size === 'md' ? 'px-1' : 'px-0.5'} grow flex-col items-start gap-1`}>
 
-         <div className='flex items-center gap-1'>
 
-           <div className='system-sm-semibold text-text-primary'>{message}</div>
 
-           {customComponent}
 
-         </div>
 
-         {children && <div className='system-xs-regular text-text-secondary'>
 
-           {children}
 
-         </div>
 
-         }
 
-       </div>
 
-       {close
 
-         && (<ActionButton className='z-[1000]' onClick={close}>
 
-           <RiCloseLine className='h-4 w-4 shrink-0 text-text-tertiary' />
 
-         </ActionButton>)
 
-       }
 
-     </div>
 
-   </div>
 
- }
 
- export const ToastProvider = ({
 
-   children,
 
- }: {
 
-   children: ReactNode
 
- }) => {
 
-   const placeholder: IToastProps = {
 
-     type: 'info',
 
-     message: 'Toast message',
 
-     duration: 6000,
 
-   }
 
-   const [params, setParams] = React.useState<IToastProps>(placeholder)
 
-   const defaultDuring = (params.type === 'success' || params.type === 'info') ? 3000 : 6000
 
-   const [mounted, setMounted] = useState(false)
 
-   useEffect(() => {
 
-     if (mounted) {
 
-       setTimeout(() => {
 
-         setMounted(false)
 
-       }, params.duration || defaultDuring)
 
-     }
 
-   }, [defaultDuring, mounted, params.duration])
 
-   return <ToastContext.Provider value={{
 
-     notify: (props) => {
 
-       setMounted(true)
 
-       setParams(props)
 
-     },
 
-     close: () => setMounted(false),
 
-   }}>
 
-     {mounted && <Toast {...params} />}
 
-     {children}
 
-   </ToastContext.Provider>
 
- }
 
- Toast.notify = ({
 
-   type,
 
-   size = 'md',
 
-   message,
 
-   duration,
 
-   className,
 
-   customComponent,
 
-   onClose,
 
- }: Pick<IToastProps, 'type' | 'size' | 'message' | 'duration' | 'className' | 'customComponent' | 'onClose'>) => {
 
-   const defaultDuring = (type === 'success' || type === 'info') ? 3000 : 6000
 
-   if (typeof window === 'object') {
 
-     const holder = document.createElement('div')
 
-     const root = createRoot(holder)
 
-     root.render(
 
-       <ToastContext.Provider value={{
 
-         notify: () => { },
 
-         close: () => {
 
-           if (holder) {
 
-             root.unmount()
 
-             holder.remove()
 
-           }
 
-           onClose?.()
 
-         },
 
-       }}>
 
-         <Toast type={type} size={size} message={message} duration={duration} className={className} customComponent={customComponent} />
 
-       </ToastContext.Provider>,
 
-     )
 
-     document.body.appendChild(holder)
 
-     setTimeout(() => {
 
-       if (holder) {
 
-         root.unmount()
 
-         holder.remove()
 
-       }
 
-       onClose?.()
 
-     }, duration || defaultDuring)
 
-   }
 
- }
 
- export default Toast
 
 
  |