UNPKG

@nlabs/gothamjs

Version:
152 lines (151 loc) 20.3 kB
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"; import { jsx, jsxs } from "react/jsx-runtime"; const Button = ({ children, onClick, className = "" }) => /* @__PURE__ */ jsx( "button", { type: "button", 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 } ); const IconButton = ({ children, onClick, className = "" }) => /* @__PURE__ */ jsx( "button", { type: "button", 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 } ); 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 }), 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" }) }) ] } ) ] }); }; 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 void 0; }, [isOpen, notification.autoHideDuration, timeoutId]); const notifyOpen = ({ actions = [], autoHideDuration = 3e3, 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, 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 }) }) }) } ); }; export { Notify }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../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"],
  "mappings": "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",
  "names": []
}
