adjuest poster and mian chat

This commit is contained in:
hobby
2025-02-26 08:07:13 +08:00
parent a049df7147
commit 008165d77b
2 changed files with 273 additions and 270 deletions

View File

@@ -377,291 +377,293 @@ const ChatUI = () => {
return ( return (
<> <>
<KaTeXStyle /> <KaTeXStyle />
<div className="h-[100dvh] flex flex-col bg-gray-100 fixed inset-0 overflow-hidden"> <div className="min-h-screen bg-gradient-to-br from-blue-50 via-indigo-50 to-purple-50 flex items-center justify-center">
{/* Header */} <div className="h-[100dvh] flex flex-col bg-white max-w-3xl w-full mx-auto relative shadow-xl">
<header className="bg-white shadow flex-none"> {/* Header */}
<div className="flex items-center justify-between px-4 py-3"> <header className="bg-white shadow flex-none">
{/* 左侧群组信息 */} <div className="flex items-center justify-between px-4 py-3">
<div className="flex items-center gap-1.5"> {/* 左侧群组信息 */}
<div className="relative w-10 h-10"> <div className="flex items-center gap-1.5">
<div className="w-full h-full overflow-hidden bg-white border border-gray-200"> <div className="relative w-10 h-10">
{users.length === 1 ? ( <div className="w-full h-full overflow-hidden bg-white border border-gray-200">
<SingleAvatar user={users[0]} /> {users.length === 1 ? (
) : users.length === 2 ? ( <SingleAvatar user={users[0]} />
<div className="h-full flex"> ) : users.length === 2 ? (
{users.slice(0, 2).map((user, index) => ( <div className="h-full flex">
<HalfAvatar key={user.id} user={user} isFirst={index === 0} />
))}
</div>
) : users.length === 3 ? (
<div className="h-full flex flex-col">
<div className="flex h-1/2">
{users.slice(0, 2).map((user, index) => ( {users.slice(0, 2).map((user, index) => (
<HalfAvatar key={user.id} user={user} isFirst={index === 0} /> <HalfAvatar key={user.id} user={user} isFirst={index === 0} />
))} ))}
</div> </div>
<div className="h-1/2 flex justify-center"> ) : users.length === 3 ? (
<SingleAvatar user={users[2]} /> <div className="h-full flex flex-col">
<div className="flex h-1/2">
{users.slice(0, 2).map((user, index) => (
<HalfAvatar key={user.id} user={user} isFirst={index === 0} />
))}
</div>
<div className="h-1/2 flex justify-center">
<SingleAvatar user={users[2]} />
</div>
</div> </div>
</div> ) : (
) : ( <div className="h-full grid grid-cols-2">
<div className="h-full grid grid-cols-2"> {users.slice(0, 4).map((user, index) => (
{users.slice(0, 4).map((user, index) => ( <QuarterAvatar key={user.id} user={user} index={index} />
<QuarterAvatar key={user.id} user={user} index={index} /> ))}
))} </div>
)}
</div>
<div className="absolute -bottom-0.5 -right-0.5 bg-green-500 w-3 h-3 border-2 border-white"></div>
</div>
<div>
<h1 className="font-medium text-base">{group.name}</h1>
<p className="text-xs text-gray-500">{users.length} </p>
</div>
</div>
{/* 右侧头像组和按钮 */}
<div className="flex items-center">
<div className="flex -space-x-2 ">
{users.slice(0, 4).map((user) => {
const avatarData = getAvatarData(user.name);
return (
<TooltipProvider key={user.id}>
<Tooltip>
<TooltipTrigger>
<Avatar className="w-7 h-7 border-2 border-white">
{'avatar' in user && user.avatar ? (
<AvatarImage src={user.avatar} />
) : (
<AvatarFallback style={{ backgroundColor: avatarData.backgroundColor, color: 'white' }}>
{avatarData.text}
</AvatarFallback>
)}
</Avatar>
</TooltipTrigger>
<TooltipContent>
<p>{user.name}</p>
</TooltipContent>
</Tooltip>
</TooltipProvider>
);
})}
{users.length > 4 && (
<div className="w-7 h-7 rounded-full bg-gray-200 flex items-center justify-center text-xs border-2 border-white">
+{users.length - 4}
</div> </div>
)} )}
</div> </div>
<div className="absolute -bottom-0.5 -right-0.5 bg-green-500 w-3 h-3 border-2 border-white"></div> <Button variant="ghost" size="icon" onClick={() => setShowMembers(true)}>
</div> <Users className="w-5 h-5" />
<div> </Button>
<h1 className="font-medium text-base">{group.name}</h1>
<p className="text-xs text-gray-500">{users.length} </p>
</div> </div>
</div> </div>
</header>
{/* 右侧头像组和按钮 */}
<div className="flex items-center"> {/* Main Chat Area */}
<div className="flex -space-x-2 "> <div className="flex-1 overflow-hidden bg-gray-100">
{users.slice(0, 4).map((user) => { <ScrollArea className="h-full p-2" ref={chatAreaRef}>
const avatarData = getAvatarData(user.name); <div className="space-y-4">
return ( {messages.map((message) => (
<TooltipProvider key={user.id}> <div key={message.id}
<Tooltip> className={`flex items-start gap-2 ${message.sender.name === "我" ? "justify-end" : ""}`}>
<TooltipTrigger> {message.sender.name !== "我" && (
<Avatar className="w-7 h-7 border-2 border-white"> <Avatar>
{'avatar' in user && user.avatar ? ( {'avatar' in message.sender && message.sender.avatar ? (
<AvatarImage src={user.avatar} /> <AvatarImage src={message.sender.avatar} className="w-10 h-10" />
) : ( ) : (
<AvatarFallback style={{ backgroundColor: avatarData.backgroundColor, color: 'white' }}> <AvatarFallback style={{ backgroundColor: getAvatarData(message.sender.name).backgroundColor, color: 'white' }}>
{avatarData.text} {message.sender.name[0]}
</AvatarFallback> </AvatarFallback>
)} )}
</Avatar> </Avatar>
</TooltipTrigger> )}
<TooltipContent> <div className={message.sender.name === "我" ? "text-right" : ""}>
<p>{user.name}</p> <div className="text-sm text-gray-500">{message.sender.name}</div>
</TooltipContent> <div className={`mt-1 p-3 rounded-lg shadow-sm chat-message ${
</Tooltip> message.sender.name === "我" ? "bg-blue-500 text-white text-left" : "bg-white"
</TooltipProvider> }`}>
); <ReactMarkdown
})} remarkPlugins={[remarkGfm, remarkMath]}
{users.length > 4 && ( rehypePlugins={[rehypeKatex]}
<div className="w-7 h-7 rounded-full bg-gray-200 flex items-center justify-center text-xs border-2 border-white"> className={`prose dark:prose-invert max-w-none ${
+{users.length - 4} message.sender.name === "我" ? "text-white [&_*]:text-white" : ""
}
[&_h2]:py-1
[&_h2]:m-0
[&_h3]:py-1.5
[&_h3]:m-0
[&_p]:m-0
[&_pre]:bg-gray-900
[&_pre]:p-2
[&_pre]:m-0
[&_pre]:rounded-lg
[&_pre]:text-gray-100
[&_pre]:whitespace-pre-wrap
[&_pre]:break-words
[&_pre_code]:whitespace-pre-wrap
[&_pre_code]:break-words
[&_code]:text-sm
[&_code]:text-gray-400
[&_code:not(:where([class~="language-"]))]:text-pink-500
[&_code:not(:where([class~="language-"]))]:bg-transparent
[&_a]:text-blue-500
[&_a]:no-underline
[&_ul]:my-2
[&_ol]:my-2
[&_li]:my-1
[&_blockquote]:border-l-4
[&_blockquote]:border-gray-300
[&_blockquote]:pl-4
[&_blockquote]:my-2
[&_blockquote]:italic`}
>
{message.content}
</ReactMarkdown>
{message.isAI && isTyping && currentMessageRef.current === message.id && (
<span className="typing-indicator ml-1"></span>
)}
</div>
</div>
{message.sender.name === "我" && (
<Avatar>
{'avatar' in message.sender && message.sender.avatar ? (
<AvatarImage src={message.sender.avatar} className="w-10 h-10" />
) : (
<AvatarFallback style={{ backgroundColor: getAvatarData(message.sender.name).backgroundColor, color: 'white' }}>
{message.sender.name[0]}
</AvatarFallback>
)}
</Avatar>
)}
</div> </div>
)} ))}
<div ref={messagesEndRef} />
{/* 添加一个二维码 */}
<div id="qrcode" className="flex flex-col items-center hidden">
<img src="/img/qr.png" alt="QR Code" className="w-24 h-24" />
<p className="text-sm text-gray-500 mt-2 font-medium tracking-tight bg-gray-50 px-3 py-1 rounded-full">AI群聊</p>
</div>
</div> </div>
<Button variant="ghost" size="icon" onClick={() => setShowMembers(true)}> </ScrollArea>
<Users className="w-5 h-5" /> </div>
{/* Input Area */}
<div className="bg-white border-t pb-[calc(0.75rem+env(safe-area-inset-bottom))] pt-3 px-4">
<div className="flex gap-1">
{messages.length > 0 && (
<TooltipProvider>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="outline"
size="icon"
onClick={handleShareChat}
className="px-3"
>
<Share2 className="w-4 h-4" />
</Button>
</TooltipTrigger>
<TooltipContent>
<p></p>
</TooltipContent>
</Tooltip>
</TooltipProvider>
)}
<Input
placeholder="输入消息..."
className="flex-1"
value={inputMessage}
onChange={(e) => setInputMessage(e.target.value)}
onKeyPress={(e) => e.key === 'Enter' && handleSendMessage()}
/>
<Button
onClick={handleSendMessage}
disabled={isLoading}
>
{isLoading ? (
<div className="w-4 h-4 mr-2 animate-spin rounded-full border-2 border-white border-t-transparent" />
) : (
<Send className="w-4 h-4" />
)}
</Button> </Button>
</div> </div>
</div> </div>
</header>
{/* Main Chat Area */} {/* Members Management Dialog */}
<div className="flex-1 overflow-hidden"> <Dialog open={showMembers} onOpenChange={setShowMembers}>
<ScrollArea className="h-full p-2" ref={chatAreaRef}> <DialogContent className="max-w-md">
<div className="space-y-4"> <DialogHeader>
{messages.map((message) => ( <DialogTitle></DialogTitle>
<div key={message.id} </DialogHeader>
className={`flex items-start gap-2 ${message.sender.name === "我" ? "justify-end" : ""}`}> <div className="mt-4">
{message.sender.name !== "我" && ( <div className="flex justify-between items-center mb-4">
<Avatar> <span className="text-sm text-gray-500">{users.length}</span>
{'avatar' in message.sender && message.sender.avatar ? ( <Button variant="outline" size="sm">
<AvatarImage src={message.sender.avatar} className="w-10 h-10" /> <UserPlus className="w-4 h-4 mr-2" />
) : (
<AvatarFallback style={{ backgroundColor: getAvatarData(message.sender.name).backgroundColor, color: 'white' }}> </Button>
{message.sender.name[0]}
</AvatarFallback>
)}
</Avatar>
)}
<div className={message.sender.name === "我" ? "text-right" : ""}>
<div className="text-sm text-gray-500">{message.sender.name}</div>
<div className={`mt-1 p-3 rounded-lg shadow-sm chat-message ${
message.sender.name === "我" ? "bg-blue-500 text-white text-left" : "bg-white"
}`}>
<ReactMarkdown
remarkPlugins={[remarkGfm, remarkMath]}
rehypePlugins={[rehypeKatex]}
className={`prose dark:prose-invert max-w-none ${
message.sender.name === "我" ? "text-white [&_*]:text-white" : ""
}
[&_h2]:py-1
[&_h2]:m-0
[&_h3]:py-1.5
[&_h3]:m-0
[&_p]:m-0
[&_pre]:bg-gray-900
[&_pre]:p-2
[&_pre]:m-0
[&_pre]:rounded-lg
[&_pre]:text-gray-100
[&_pre]:whitespace-pre-wrap
[&_pre]:break-words
[&_pre_code]:whitespace-pre-wrap
[&_pre_code]:break-words
[&_code]:text-sm
[&_code]:text-gray-400
[&_code:not(:where([class~="language-"]))]:text-pink-500
[&_code:not(:where([class~="language-"]))]:bg-transparent
[&_a]:text-blue-500
[&_a]:no-underline
[&_ul]:my-2
[&_ol]:my-2
[&_li]:my-1
[&_blockquote]:border-l-4
[&_blockquote]:border-gray-300
[&_blockquote]:pl-4
[&_blockquote]:my-2
[&_blockquote]:italic`}
>
{message.content}
</ReactMarkdown>
{message.isAI && isTyping && currentMessageRef.current === message.id && (
<span className="typing-indicator ml-1"></span>
)}
</div>
</div>
{message.sender.name === "我" && (
<Avatar>
{'avatar' in message.sender && message.sender.avatar ? (
<AvatarImage src={message.sender.avatar} className="w-10 h-10" />
) : (
<AvatarFallback style={{ backgroundColor: getAvatarData(message.sender.name).backgroundColor, color: 'white' }}>
{message.sender.name[0]}
</AvatarFallback>
)}
</Avatar>
)}
</div> </div>
))} <ScrollArea className="h-[300px]">
<div ref={messagesEndRef} /> <div className="space-y-2">
{/* 添加一个二维码 */} {users.map((user) => (
<div id="qrcode" className="flex flex-col items-center hidden"> <div key={user.id} className="flex items-center justify-between p-2 hover:bg-gray-100 rounded-lg">
<img src="/img/qr.png" alt="QR Code" className="w-24 h-24" /> <div className="flex items-center gap-3">
<p className="text-sm text-gray-500 mt-2 font-medium tracking-tight bg-gray-50 px-3 py-1 rounded-full">AI群聊</p> <Avatar>
</div> {'avatar' in user && user.avatar ? (
</div> <AvatarImage src={user.avatar} className="w-10 h-10" />
</ScrollArea> ) : (
</div> <AvatarFallback style={{ backgroundColor: getAvatarData(user.name).backgroundColor, color: 'white' }}>
{user.name[0]}
{/* Input Area */} </AvatarFallback>
<div className="bg-white border-t pb-[calc(0.75rem+env(safe-area-inset-bottom))] pt-3 px-4"> )}
<div className="flex gap-1"> </Avatar>
{messages.length > 0 && ( <div className="flex flex-col">
<TooltipProvider> <span>{user.name}</span>
<Tooltip> {mutedUsers.includes(user.id) && (
<TooltipTrigger asChild> <span className="text-xs text-red-500"></span>
<Button )}
variant="outline" </div>
size="icon"
onClick={handleShareChat}
className="px-3"
>
<Share2 className="w-4 h-4" />
</Button>
</TooltipTrigger>
<TooltipContent>
<p></p>
</TooltipContent>
</Tooltip>
</TooltipProvider>
)}
<Input
placeholder="输入消息..."
className="flex-1"
value={inputMessage}
onChange={(e) => setInputMessage(e.target.value)}
onKeyPress={(e) => e.key === 'Enter' && handleSendMessage()}
/>
<Button
onClick={handleSendMessage}
disabled={isLoading}
>
{isLoading ? (
<div className="w-4 h-4 mr-2 animate-spin rounded-full border-2 border-white border-t-transparent" />
) : (
<Send className="w-4 h-4" />
)}
</Button>
</div>
</div>
{/* Members Management Dialog */}
<Dialog open={showMembers} onOpenChange={setShowMembers}>
<DialogContent className="max-w-md">
<DialogHeader>
<DialogTitle></DialogTitle>
</DialogHeader>
<div className="mt-4">
<div className="flex justify-between items-center mb-4">
<span className="text-sm text-gray-500">{users.length}</span>
<Button variant="outline" size="sm">
<UserPlus className="w-4 h-4 mr-2" />
</Button>
</div>
<ScrollArea className="h-[300px]">
<div className="space-y-2">
{users.map((user) => (
<div key={user.id} className="flex items-center justify-between p-2 hover:bg-gray-100 rounded-lg">
<div className="flex items-center gap-3">
<Avatar>
{'avatar' in user && user.avatar ? (
<AvatarImage src={user.avatar} className="w-10 h-10" />
) : (
<AvatarFallback style={{ backgroundColor: getAvatarData(user.name).backgroundColor, color: 'white' }}>
{user.name[0]}
</AvatarFallback>
)}
</Avatar>
<div className="flex flex-col">
<span>{user.name}</span>
{mutedUsers.includes(user.id) && (
<span className="text-xs text-red-500"></span>
)}
</div> </div>
{user.name !== "我" && (
<div className="flex gap-2">
<TooltipProvider>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="icon"
onClick={() => handleToggleMute(user.id)}
>
{mutedUsers.includes(user.id) ? (
<MicOff className="w-4 h-4 text-red-500" />
) : (
<Mic className="w-4 h-4 text-green-500" />
)}
</Button>
</TooltipTrigger>
<TooltipContent>
{mutedUsers.includes(user.id) ? '取消禁言' : '禁言'}
</TooltipContent>
</Tooltip>
</TooltipProvider>
{/*<Button
variant="ghost"
size="icon"
onClick={() => handleRemoveUser(user.id)}
>
<UserMinus className="w-4 h-4 text-red-500" />
</Button>*/}
</div>
)}
</div> </div>
{user.name !== "我" && ( ))}
<div className="flex gap-2"> </div>
<TooltipProvider> </ScrollArea>
<Tooltip> </div>
<TooltipTrigger asChild> </DialogContent>
<Button </Dialog>
variant="ghost" </div>
size="icon"
onClick={() => handleToggleMute(user.id)}
>
{mutedUsers.includes(user.id) ? (
<MicOff className="w-4 h-4 text-red-500" />
) : (
<Mic className="w-4 h-4 text-green-500" />
)}
</Button>
</TooltipTrigger>
<TooltipContent>
{mutedUsers.includes(user.id) ? '取消禁言' : '禁言'}
</TooltipContent>
</Tooltip>
</TooltipProvider>
{/*<Button
variant="ghost"
size="icon"
onClick={() => handleRemoveUser(user.id)}
>
<UserMinus className="w-4 h-4 text-red-500" />
</Button>*/}
</div>
)}
</div>
))}
</div>
</ScrollArea>
</div>
</DialogContent>
</Dialog>
</div> </div>
{/* 添加 SharePoster 组件 */} {/* 添加 SharePoster 组件 */}

View File

@@ -72,17 +72,18 @@ export function SharePoster({ isOpen, onClose, chatAreaRef }: SharePosterProps)
const dataUrl = await domtoimage.toSvg(messageContainer as HTMLElement, { const dataUrl = await domtoimage.toSvg(messageContainer as HTMLElement, {
bgcolor: '#f3f4f6', bgcolor: '#f3f4f6',
scale: 1, // 回到较安全的值 scale: 1, // 回到较安全的值
width: targetWidth + (extraSpace * 2), width: targetWidth + (extraSpace * 5),
height: adjustedHeight + (extraSpace * 2), height: adjustedHeight + (extraSpace * 5),
style: { style: {
padding: `${extraSpace}px`, padding: `${extraSpace}px`,
margin: '0', margin: '0 auto',
width: '120%', width: '120%',
height: '110%', height: '110%',
transform: `scale(${scale})`, transform: `scale(${scale})`,
transformOrigin: 'top left', transformOrigin: 'top left',
background: '#f3f4f6', background: '#f3f4f6',
boxSizing: 'border-box' boxSizing: 'border-box'
}, },
quality: 1.0 quality: 1.0
}); });
@@ -168,7 +169,7 @@ export function SharePoster({ isOpen, onClose, chatAreaRef }: SharePosterProps)
onClose(); onClose();
} }
}}> }}>
<DialogContent className="max-w-[100vw] w-full sm:max-w-[100vw] max-h-[90vh] flex flex-col p-0"> <DialogContent className="max-w-[100vw] w-full sm:max-w-[50vw] max-h-[90vh] flex flex-col p-0">
{/* 图片容器 */} {/* 图片容器 */}
<div className="flex-1 overflow-auto "> <div className="flex-1 overflow-auto ">
{posterImage && ( {posterImage && (