cal.pub0.org/packages/ui/components/alert/Alert.tsx

69 lines
2.6 KiB
TypeScript

import { CheckCircleIcon, ExclamationIcon, InformationCircleIcon, XCircleIcon } from "@heroicons/react/solid";
import classNames from "classnames";
import type { ReactNode } from "react";
import { FiInfo } from "../icon";
export interface AlertProps {
title?: ReactNode;
// @TODO: Message should be children, more flexible?
message?: ReactNode;
// @TODO: Provide action buttons so style is always the same.
actions?: ReactNode;
className?: string;
iconClassName?: string;
// @TODO: Success and info shouldn't exist as per design?
severity: "success" | "warning" | "error" | "info" | "neutral";
}
export function Alert(props: AlertProps) {
const { severity, iconClassName } = props;
return (
<div
className={classNames(
"rounded-md border border-opacity-20 p-3",
props.className,
severity === "error" && "border-red-900 bg-red-50 text-red-800",
severity === "warning" && "border-yellow-700 bg-yellow-50 text-yellow-700",
severity === "info" && "border-sky-700 bg-sky-50 text-sky-700",
severity === "success" && "bg-gray-900 text-white",
severity === "neutral" && "border-none bg-gray-100"
)}>
<div className="relative flex flex-col md:flex-row">
<div className="flex-shrink-0">
{severity === "error" && (
<XCircleIcon className={classNames("h-5 w-5 text-red-400", iconClassName)} aria-hidden="true" />
)}
{severity === "warning" && (
<ExclamationIcon
className={classNames("h-5 w-5 text-yellow-400", iconClassName)}
aria-hidden="true"
/>
)}
{severity === "info" && (
<InformationCircleIcon
className={classNames("h-5 w-5 text-sky-400", iconClassName)}
aria-hidden="true"
/>
)}
{severity === "neutral" && (
<FiInfo className={classNames("h-5 w-5 text-gray-800", iconClassName)} aria-hidden="true" />
)}
{severity === "success" && (
<CheckCircleIcon
className={classNames("h-5 w-5 text-gray-400", iconClassName)}
aria-hidden="true"
/>
)}
</div>
<div className="ml-3 flex-grow">
<h3 className="text-sm font-medium">{props.title}</h3>
<div className="text-sm">{props.message}</div>
</div>
{/* @TODO: Shouldn't be absolute. This makes it harder to give margin etc. */}
{props.actions && <div className="absolute top-1 right-1 text-sm md:relative">{props.actions}</div>}
</div>
</div>
);
}