@nlabs/gothamjs
Version:
Platform
152 lines (151 loc) • 20.3 kB
JavaScript
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": []
}
