| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485 |
- import React, { useEffect, useState } from 'react';
- import { CheckCircle, AlertCircle, XCircle, Info, X } from 'lucide-react';
- export type ToastType = 'success' | 'error' | 'warning' | 'info';
- export interface ToastProps {
- type: ToastType;
- title?: string;
- message: string;
- duration?: number;
- onClose: () => void;
- }
- const Toast: React.FC<ToastProps> = ({ type, title, message, duration = 5000, onClose }) => {
- const [isVisible, setIsVisible] = useState(true);
- useEffect(() => {
- const timer = setTimeout(() => {
- setIsVisible(false);
- setTimeout(onClose, 300); // 等待动画完成
- }, duration);
- return () => clearTimeout(timer);
- }, [duration, onClose]);
- const getIcon = () => {
- switch (type) {
- case 'success':
- return <CheckCircle className="w-5 h-5 text-green-500" />;
- case 'error':
- return <XCircle className="w-5 h-5 text-red-500" />;
- case 'warning':
- return <AlertCircle className="w-5 h-5 text-yellow-500" />;
- case 'info':
- return <Info className="w-5 h-5 text-blue-500" />;
- }
- };
- const getStyles = () => {
- switch (type) {
- case 'success':
- return 'bg-green-50 border-green-200 text-green-800';
- case 'error':
- return 'bg-red-50 border-red-200 text-red-800';
- case 'warning':
- return 'bg-yellow-50 border-yellow-200 text-yellow-800';
- case 'info':
- return 'bg-blue-50 border-blue-200 text-blue-800';
- }
- };
- return (
- <div
- role="alert"
- className={`relative w-full max-w-sm transition-all duration-300 transform ${isVisible ? 'translate-x-0 opacity-100' : 'translate-x-full opacity-0'
- }`}
- >
- <div className={`rounded-lg border shadow-lg p-4 backdrop-blur-sm ${getStyles()}`}>
- <div className="flex items-start gap-3">
- <div className="flex-shrink-0 mt-0.5">
- {getIcon()}
- </div>
- <div className="flex-1 min-w-0">
- {title && (
- <p className="text-sm font-semibold mb-1">{title}</p>
- )}
- <p className="text-sm font-medium leading-relaxed">{message}</p>
- </div>
- <button
- onClick={() => {
- setIsVisible(false);
- setTimeout(onClose, 300);
- }}
- className="flex-shrink-0 ml-2 p-1 rounded-full opacity-60 hover:opacity-100 hover:bg-black/5 transition-all"
- aria-label="Close"
- >
- <X className="w-4 h-4" />
- </button>
- </div>
- </div>
- </div>
- );
- };
- export default Toast;
|