UNPKG

@nlabs/gothamjs

Version:
192 lines (191 loc) 23.7 kB
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; import { Transition } from '@headlessui/react'; import { useFluxListener } from '@nlabs/arkhamjs-utils-react'; import { cn } from '@nlabs/utils'; import { Fragment, useEffect, useState } from 'react'; import { GothamConstants } from '../../constants/GothamConstants.js'; import { Svg } from '../Svg/Svg.js'; // Custom Button component const Button = ({ children, onClick, className = '' })=>/*#__PURE__*/ _jsx("button", { type: "button", onClick: onClick, className: cn('inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm', 'text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500', className), children: children }); // Custom IconButton component const IconButton = ({ children, onClick, className = '' })=>/*#__PURE__*/ _jsx("button", { type: "button", onClick: onClick, className: cn('p-1 rounded-full hover:bg-gray-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500', className), children: children }); // Custom Alert component const Alert = ({ children, severity, onClose })=>{ const bgColors = { error: 'bg-red-500', info: 'bg-blue-500', success: 'bg-green-500', warning: 'bg-yellow-500' }; return /*#__PURE__*/ _jsxs("div", { className: cn('rounded-md p-4 w-full flex items-center justify-between', bgColors[severity] || 'bg-gray-500', 'text-white'), children: [ /*#__PURE__*/ _jsx("div", { children: children }), onClose && /*#__PURE__*/ _jsxs("button", { type: "button", onClick: onClose, className: "ml-auto -mx-1.5 -my-1.5 rounded-md p-1.5 inline-flex text-white hover:bg-opacity-20 hover:bg-black focus:outline-none focus:ring-2 focus:ring-white", children: [ /*#__PURE__*/ _jsx("span", { className: "sr-only", children: "Dismiss" }), /*#__PURE__*/ _jsx("svg", { className: "h-5 w-5", xmlns: "http://www.w3.org/2000/svg", viewBox: "0 0 20 20", fill: "currentColor", children: /*#__PURE__*/ _jsx("path", { fillRule: "evenodd", d: "M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z", clipRule: "evenodd" }) }) ] }) ] }); }; export const Notify = ()=>{ const [isOpen, setOpen] = useState(false); const [notification, setNotification] = useState({}); const [timeoutId, setTimeoutId] = useState(null); const notifyClose = ()=>setOpen(false); useEffect(()=>{ if (isOpen && notification.autoHideDuration) { if (timeoutId) { clearTimeout(timeoutId); } const id = setTimeout(()=>{ setOpen(false); }, notification.autoHideDuration); setTimeoutId(id); return ()=>{ clearTimeout(id); }; } return undefined; }, [ isOpen, notification.autoHideDuration, timeoutId ]); const notifyOpen = ({ actions = [], autoHideDuration = 3000, message, severity, anchorOrigin = { horizontal: 'left', vertical: 'bottom' }, ...restProps })=>{ let action; if (actions.length) { action = (key)=>/*#__PURE__*/ _jsx("div", { className: "flex space-x-2", children: actions.map(({ icon, label, onClick }, index)=>/*#__PURE__*/ _jsx(Fragment, { children: icon ? /*#__PURE__*/ _jsx(IconButton, { onClick: ()=>onClick(key), children: /*#__PURE__*/ _jsx(Svg, { color: "inherit", height: 24, name: icon, width: 24 }) }) : /*#__PURE__*/ _jsx(Button, { onClick: ()=>onClick(key), children: label }) }, index)) }); } setNotification({ ...restProps, actions: [ action ], anchorOrigin, autoHideDuration, message: severity ? /*#__PURE__*/ _jsx(Alert, { onClose: notifyClose, severity: severity, children: message }) : message, severity }); setOpen(true); }; useFluxListener(GothamConstants.NOTIFY_OPEN, notifyOpen); useFluxListener(GothamConstants.NOTIFY_CLOSE, notifyClose); const positionClasses = (()=>{ const { horizontal = 'left', vertical = 'bottom' } = notification.anchorOrigin || {}; const positions = { bottom: { center: 'bottom-4 left-1/2 transform -translate-x-1/2', left: 'bottom-4 left-4', right: 'bottom-4 right-4' }, top: { center: 'top-4 left-1/2 transform -translate-x-1/2', left: 'top-4 left-4', right: 'top-4 right-4' } }; return positions[vertical][horizontal]; })(); return /*#__PURE__*/ _jsx(Transition, { show: isOpen, as: Fragment, enter: "transform ease-out duration-300 transition", enterFrom: "translate-y-2 opacity-0 sm:translate-y-0 sm:translate-x-2", enterTo: "translate-y-0 opacity-100 sm:translate-x-0", leave: "transition ease-in duration-100", leaveFrom: "opacity-100", leaveTo: "opacity-0", children: /*#__PURE__*/ _jsx("div", { className: cn('fixed z-50 max-w-sm w-full shadow-lg rounded-lg pointer-events-auto overflow-hidden', positionClasses), children: /*#__PURE__*/ _jsx("div", { className: "ring-1 ring-black ring-opacity-5 bg-white", children: !notification.severity ? /*#__PURE__*/ _jsx("div", { className: "p-4", children: /*#__PURE__*/ _jsxs("div", { className: "flex items-start", children: [ /*#__PURE__*/ _jsx("div", { className: "flex-1", children: notification.message }), notification.actions?.length && /*#__PURE__*/ _jsx("div", { className: "ml-4 flex-shrink-0 flex", children: notification.actions.map(({ icon, label, onClick }, index)=>/*#__PURE__*/ _jsx(Fragment, { children: icon ? /*#__PURE__*/ _jsx(IconButton, { onClick: ()=>onClick('notification'), children: /*#__PURE__*/ _jsx(Svg, { color: "inherit", height: 24, name: icon, width: 24 }) }) : /*#__PURE__*/ _jsx(Button, { onClick: ()=>onClick('notification'), children: label }) }, index)) }) ] }) }) : /*#__PURE__*/ _jsx("div", { children: notification.message }) }) }) }); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/gothamjs/src/components/Notify/Notify.tsx"],"sourcesContent":["import {Transition} from '@headlessui/react';\nimport {useFluxListener} from '@nlabs/arkhamjs-utils-react';\nimport {cn} from '@nlabs/utils';\nimport {Fragment, useEffect, useState} from 'react';\n\nimport {GothamConstants} from '../../constants/GothamConstants.js';\nimport {Svg} from '../Svg/Svg.js';\n\nimport type {ReactElement} from 'react';\n\nexport interface GothamNotifyAction {\n  readonly icon?: string;\n  readonly label?: string;\n  readonly onClick: (key: string) => void;\n}\n\nexport type GothamSeverity = 'error' | 'info' | 'success' | 'warning';\n\nexport interface GothamNotifyParams {\n  readonly actions?: GothamNotifyAction[];\n  readonly anchorOrigin?: {\n    vertical: 'top' | 'bottom';\n    horizontal: 'left' | 'center' | 'right';\n  };\n  readonly autoHideDuration?: number;\n  readonly message?: ReactElement | string;\n  readonly severity?: GothamSeverity;\n}\n\n// Custom Button component\nconst Button = ({children, onClick, className = ''}) => (\n  <button\n    type=\"button\"\n    onClick={onClick}\n    className={cn(\n      'inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm',\n      'text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500',\n      className\n    )}\n  >\n    {children}\n  </button>\n);\n\n// Custom IconButton component\nconst IconButton = ({children, onClick, className = ''}) => (\n  <button\n    type=\"button\"\n    onClick={onClick}\n    className={cn(\n      'p-1 rounded-full hover:bg-gray-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500',\n      className\n    )}\n  >\n    {children}\n  </button>\n);\n\n// Custom Alert component\nconst Alert = ({children, severity, onClose}) => {\n  const bgColors = {\n    error: 'bg-red-500',\n    info: 'bg-blue-500',\n    success: 'bg-green-500',\n    warning: 'bg-yellow-500'\n  };\n\n  return (\n    <div className={cn(\n      'rounded-md p-4 w-full flex items-center justify-between',\n      bgColors[severity] || 'bg-gray-500',\n      'text-white'\n    )}>\n      <div>{children}</div>\n      {onClose && (\n        <button\n          type=\"button\"\n          onClick={onClose}\n          className=\"ml-auto -mx-1.5 -my-1.5 rounded-md p-1.5 inline-flex text-white hover:bg-opacity-20 hover:bg-black focus:outline-none focus:ring-2 focus:ring-white\"\n        >\n          <span className=\"sr-only\">Dismiss</span>\n          <svg className=\"h-5 w-5\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 20 20\" fill=\"currentColor\">\n            <path fillRule=\"evenodd\" d=\"M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z\" clipRule=\"evenodd\" />\n          </svg>\n        </button>\n      )}\n    </div>\n  );\n};\n\nexport const Notify = () => {\n  const [isOpen, setOpen] = useState(false);\n  const [notification, setNotification] = useState<GothamNotifyParams>({});\n  const [timeoutId, setTimeoutId] = useState<NodeJS.Timeout | null>(null);\n\n  const notifyClose = () => setOpen(false);\n\n  useEffect(() => {\n    if (isOpen && notification.autoHideDuration) {\n      if (timeoutId) {\n        clearTimeout(timeoutId);\n      }\n\n      const id = setTimeout(() => {\n        setOpen(false);\n      }, notification.autoHideDuration);\n\n      setTimeoutId(id);\n\n      return () => {\n        clearTimeout(id);\n      };\n    }\n\n    return undefined;\n  }, [isOpen, notification.autoHideDuration, timeoutId]);\n\n  const notifyOpen = ({\n    actions = [],\n    autoHideDuration = 3000,\n    message,\n    severity,\n    anchorOrigin = { horizontal: 'left', vertical: 'bottom' },\n    ...restProps\n  }: GothamNotifyParams) => {\n    let action;\n\n    if(actions.length) {\n      action = (key: string) => (\n        <div className=\"flex space-x-2\">\n          {actions.map(({icon, label, onClick}, index) => (\n            <Fragment key={index}>\n              {icon ? (\n                <IconButton onClick={() => onClick(key)}>\n                  <Svg color=\"inherit\" height={24} name={icon} width={24} />\n                </IconButton>\n              ) : (\n                <Button onClick={() => onClick(key)}>{label}</Button>\n              )}\n            </Fragment>\n          ))}\n        </div>\n      );\n    }\n\n    setNotification({\n      ...restProps,\n      actions: [action as GothamNotifyAction],\n      anchorOrigin,\n      autoHideDuration,\n      message: severity ? (\n        <Alert\n          onClose={notifyClose}\n          severity={severity}\n        >\n          {message}\n        </Alert>\n      ) : message,\n      severity\n    });\n    setOpen(true);\n  };\n\n  useFluxListener(GothamConstants.NOTIFY_OPEN, notifyOpen);\n  useFluxListener(GothamConstants.NOTIFY_CLOSE, notifyClose);\n\n  const positionClasses = (() => {\n    const {horizontal = 'left', vertical = 'bottom'} = notification.anchorOrigin || {};\n\n    const positions = {\n      bottom: {\n        center: 'bottom-4 left-1/2 transform -translate-x-1/2',\n        left: 'bottom-4 left-4',\n        right: 'bottom-4 right-4'\n      },\n      top: {\n        center: 'top-4 left-1/2 transform -translate-x-1/2',\n        left: 'top-4 left-4',\n        right: 'top-4 right-4'\n      }\n    };\n\n    return positions[vertical][horizontal];\n  })();\n\n  return (\n    <Transition\n      show={isOpen}\n      as={Fragment}\n      enter=\"transform ease-out duration-300 transition\"\n      enterFrom=\"translate-y-2 opacity-0 sm:translate-y-0 sm:translate-x-2\"\n      enterTo=\"translate-y-0 opacity-100 sm:translate-x-0\"\n      leave=\"transition ease-in duration-100\"\n      leaveFrom=\"opacity-100\"\n      leaveTo=\"opacity-0\"\n    >\n      <div className={cn(\n        'fixed z-50 max-w-sm w-full shadow-lg rounded-lg pointer-events-auto overflow-hidden',\n        positionClasses\n      )}>\n        <div className=\"ring-1 ring-black ring-opacity-5 bg-white\">\n          {!notification.severity ? (\n            <div className=\"p-4\">\n              <div className=\"flex items-start\">\n                <div className=\"flex-1\">\n                  {notification.message}\n                </div>\n                {notification.actions?.length && (\n                  <div className=\"ml-4 flex-shrink-0 flex\">\n                    {notification.actions.map(({icon, label, onClick}, index) => (\n                      <Fragment key={index}>\n                        {icon ? (\n                          <IconButton onClick={() => onClick('notification')}>\n                            <Svg color=\"inherit\" height={24} name={icon} width={24} />\n                          </IconButton>\n                        ) : (\n                          <Button onClick={() => onClick('notification')}>{label}</Button>\n                        )}\n                      </Fragment>\n                    ))}\n                  </div>\n                )}\n              </div>\n            </div>\n          ) : (\n            <div>\n              {notification.message as ReactElement}\n            </div>\n          )}\n        </div>\n      </div>\n    </Transition>\n  );\n};\n"],"names":["Transition","useFluxListener","cn","Fragment","useEffect","useState","GothamConstants","Svg","Button","children","onClick","className","button","type","IconButton","Alert","severity","onClose","bgColors","error","info","success","warning","div","span","svg","xmlns","viewBox","fill","path","fillRule","d","clipRule","Notify","isOpen","setOpen","notification","setNotification","timeoutId","setTimeoutId","notifyClose","autoHideDuration","clearTimeout","id","setTimeout","undefined","notifyOpen","actions","message","anchorOrigin","horizontal","vertical","restProps","action","length","key","map","icon","label","index","color","height","name","width","NOTIFY_OPEN","NOTIFY_CLOSE","positionClasses","positions","bottom","center","left","right","top","show","as","enter","enterFrom","enterTo","leave","leaveFrom","leaveTo"],"mappings":"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"}