678 lines
24 KiB
TypeScript
678 lines
24 KiB
TypeScript
import React, { useState, useRef, useEffect } from 'react';
|
||
import { Send, Share2, Settings2, ChevronLeft } from 'lucide-react';
|
||
import { Button } from "@/components/ui/button";
|
||
import { Input } from "@/components/ui/input";
|
||
import { ScrollArea } from "@/components/ui/scroll-area";
|
||
import { Avatar, AvatarImage, AvatarFallback } from "@/components/ui/avatar";
|
||
|
||
import {
|
||
Tooltip,
|
||
TooltipContent,
|
||
TooltipProvider,
|
||
TooltipTrigger
|
||
} from "@/components/ui/tooltip";
|
||
|
||
import {generateAICharacters} from "@/config/aiCharacters";
|
||
import { groups } from "@/config/groups";
|
||
import type { AICharacter } from "@/config/aiCharacters";
|
||
import ReactMarkdown from 'react-markdown';
|
||
import remarkGfm from 'remark-gfm'
|
||
import remarkMath from 'remark-math'
|
||
import rehypeKatex from 'rehype-katex'
|
||
import { SharePoster } from '@/components/SharePoster';
|
||
import { MembersManagement } from '@/components/MembersManagement';
|
||
import Sidebar from './Sidebar';
|
||
import { AdBanner, AdBannerMobile } from './AdSection';
|
||
// 使用本地头像数据,避免外部依赖
|
||
const getAvatarData = (name: string) => {
|
||
const colors = ['#1abc9c', '#3498db', '#9b59b6', '#f1c40f', '#e67e22'];
|
||
const index = (name.charCodeAt(0) + (name.charCodeAt(1) || 0 )) % colors.length;
|
||
return {
|
||
backgroundColor: colors[index],
|
||
text: name[0],
|
||
};
|
||
};
|
||
|
||
// 单个完整头像
|
||
const SingleAvatar = ({ user }: { user: User | AICharacter }) => {
|
||
// 如果有头像就使用头像,否则使用默认的文字头像
|
||
if ('avatar' in user && user.avatar) {
|
||
return (
|
||
<div className="w-full h-full">
|
||
<img src={user.avatar} alt={user.name} className="w-full h-full object-cover" />
|
||
</div>
|
||
);
|
||
}
|
||
const avatarData = getAvatarData(user.name);
|
||
return (
|
||
<div
|
||
className="w-full h-full flex items-center justify-center text-xs text-white font-medium"
|
||
style={{ backgroundColor: avatarData.backgroundColor }}
|
||
>
|
||
{avatarData.text}
|
||
</div>
|
||
);
|
||
};
|
||
|
||
// 左右分半头像
|
||
const HalfAvatar = ({ user, isFirst }: { user: User, isFirst: boolean }) => {
|
||
if ('avatar' in user && user.avatar) {
|
||
return (
|
||
<div
|
||
className="w-1/2 h-full"
|
||
style={{
|
||
borderRight: isFirst ? '1px solid white' : 'none'
|
||
}}
|
||
>
|
||
<img src={user.avatar} alt={user.name} className="w-full h-full object-cover" />
|
||
</div>
|
||
);
|
||
}
|
||
const avatarData = getAvatarData(user.name);
|
||
return (
|
||
<div
|
||
className="w-1/2 h-full flex items-center justify-center text-xs text-white font-medium"
|
||
style={{
|
||
backgroundColor: avatarData.backgroundColor,
|
||
borderRight: isFirst ? '1px solid white' : 'none'
|
||
}}
|
||
>
|
||
{avatarData.text}
|
||
</div>
|
||
);
|
||
};
|
||
|
||
// 四分之一头像
|
||
const QuarterAvatar = ({ user, index }: { user: User, index: number }) => {
|
||
if ('avatar' in user && user.avatar) {
|
||
return (
|
||
<div
|
||
className="aspect-square"
|
||
style={{
|
||
borderRight: index % 2 === 0 ? '1px solid white' : 'none',
|
||
borderBottom: index < 2 ? '1px solid white' : 'none'
|
||
}}
|
||
>
|
||
<img src={user.avatar} alt={user.name} className="w-full h-full object-cover" />
|
||
</div>
|
||
);
|
||
}
|
||
const avatarData = getAvatarData(user.name);
|
||
return (
|
||
<div
|
||
className="aspect-square flex items-center justify-center text-[8px] text-white font-medium"
|
||
style={{
|
||
backgroundColor: avatarData.backgroundColor,
|
||
borderRight: index % 2 === 0 ? '1px solid white' : 'none',
|
||
borderBottom: index < 2 ? '1px solid white' : 'none'
|
||
}}
|
||
>
|
||
{avatarData.text}
|
||
</div>
|
||
);
|
||
};
|
||
|
||
// 修改 KaTeXStyle 组件
|
||
const KaTeXStyle = () => (
|
||
<style dangerouslySetInnerHTML={{ __html: `
|
||
/* 只在聊天消息内应用 KaTeX 样式 */
|
||
.chat-message .katex-html {
|
||
display: none;
|
||
}
|
||
|
||
.chat-message .katex {
|
||
font: normal 1.1em KaTeX_Main, Times New Roman, serif;
|
||
line-height: 1.2;
|
||
text-indent: 0;
|
||
white-space: nowrap;
|
||
text-rendering: auto;
|
||
}
|
||
|
||
.chat-message .katex-display {
|
||
display: block;
|
||
margin: 1em 0;
|
||
text-align: center;
|
||
}
|
||
|
||
/* 其他必要的 KaTeX 样式 */
|
||
@import "katex/dist/katex.min.css";
|
||
`}} />
|
||
);
|
||
|
||
const ChatUI = () => {
|
||
// 使用当前选中的群组在 groups 数组中的索引
|
||
const [selectedGroupIndex, setSelectedGroupIndex] = useState(0); // 默认选中第1个群组
|
||
const [group, setGroup] = useState(groups[selectedGroupIndex]);
|
||
const [isGroupDiscussionMode, setIsGroupDiscussionMode] = useState(group.isGroupDiscussionMode);
|
||
const groupAiCharacters = generateAICharacters(group.name)
|
||
.filter(character => group.members.includes(character.id))
|
||
.sort((a, b) => {
|
||
return group.members.indexOf(a.id) - group.members.indexOf(b.id);
|
||
});
|
||
const allNames = groupAiCharacters.map(character => character.name);
|
||
const [users, setUsers] = useState([
|
||
{ id: 1, name: "我" },
|
||
...groupAiCharacters
|
||
]);
|
||
const [showMembers, setShowMembers] = useState(false);
|
||
const [messages, setMessages] = useState([
|
||
|
||
]);
|
||
const [showAd, setShowAd] = useState(true);
|
||
const [inputMessage, setInputMessage] = useState("");
|
||
const [isLoading, setIsLoading] = useState(false);
|
||
const [pendingContent, setPendingContent] = useState("");
|
||
const [isTyping, setIsTyping] = useState(false);
|
||
const currentMessageRef = useRef<number | null>(null);
|
||
const typewriterRef = useRef<NodeJS.Timeout | null>(null);
|
||
const accumulatedContentRef = useRef(""); // 用于跟踪完整内容
|
||
const messagesEndRef = useRef<HTMLDivElement>(null);
|
||
|
||
// 添加禁言状态
|
||
const [mutedUsers, setMutedUsers] = useState<string[]>([]);
|
||
|
||
const abortController = new AbortController();
|
||
|
||
const handleRemoveUser = (userId: number) => {
|
||
setUsers(users.filter(user => user.id !== userId));
|
||
};
|
||
|
||
const scrollToBottom = () => {
|
||
messagesEndRef.current?.scrollIntoView({ behavior: "smooth" });
|
||
};
|
||
|
||
useEffect(() => {
|
||
scrollToBottom();
|
||
}, [messages]);
|
||
|
||
// 添加禁言/取消禁言处理函数
|
||
const handleToggleMute = (userId: string) => {
|
||
setMutedUsers(prev =>
|
||
prev.includes(userId)
|
||
? prev.filter(id => id !== userId)
|
||
: [...prev, userId]
|
||
);
|
||
};
|
||
|
||
useEffect(() => {
|
||
if (messages.length > 0) {
|
||
setShowAd(false);
|
||
}
|
||
}, [messages]);
|
||
|
||
const handleSendMessage = async () => {
|
||
//判断是否Loding
|
||
if (isLoading) return;
|
||
if (!inputMessage.trim()) return;
|
||
|
||
// 添加用户消息
|
||
const userMessage = {
|
||
id: messages.length + 1,
|
||
sender: users[0],
|
||
content: inputMessage,
|
||
isAI: false
|
||
};
|
||
setMessages(prev => [...prev, userMessage]);
|
||
setInputMessage("");
|
||
setIsLoading(true);
|
||
setPendingContent("");
|
||
accumulatedContentRef.current = "";
|
||
|
||
// 构建历史消息数组
|
||
let messageHistory = messages.map(msg => ({
|
||
role: 'user',
|
||
content: msg.sender.name == "我" ? 'user:' + msg.content : msg.sender.name + ':' + msg.content,
|
||
name: msg.sender.name
|
||
}));
|
||
let selectedGroupAiCharacters = groupAiCharacters;
|
||
if (!isGroupDiscussionMode) {
|
||
const shedulerResponse = await fetch('/api/scheduler', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json',
|
||
},
|
||
body: JSON.stringify({ message: inputMessage, history: messageHistory, availableAIs: groupAiCharacters })
|
||
});
|
||
const shedulerData = await shedulerResponse.json();
|
||
const selectedAIs = shedulerData.selectedAIs;
|
||
selectedGroupAiCharacters = selectedAIs.map(ai => groupAiCharacters.find(c => c.id === ai));
|
||
}
|
||
for (let i = 0; i < selectedGroupAiCharacters.length; i++) {
|
||
//禁言
|
||
if (mutedUsers.includes(selectedGroupAiCharacters[i].id)) {
|
||
continue;
|
||
}
|
||
// 创建当前 AI 角色的消息
|
||
const aiMessage = {
|
||
id: messages.length + 2 + i,
|
||
sender: { id: selectedGroupAiCharacters[i].id, name: selectedGroupAiCharacters[i].name, avatar: selectedGroupAiCharacters[i].avatar },
|
||
content: "",
|
||
isAI: true
|
||
};
|
||
|
||
// 添加当前 AI 的消息
|
||
setMessages(prev => [...prev, aiMessage]);
|
||
|
||
try {
|
||
const response = await fetch('/api/chat', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json',
|
||
},
|
||
body: JSON.stringify({
|
||
model: selectedGroupAiCharacters[i].model,
|
||
message: inputMessage,
|
||
personality: selectedGroupAiCharacters[i].personality,
|
||
history: messageHistory,
|
||
index: i,
|
||
aiName: selectedGroupAiCharacters[i].name,
|
||
custom_prompt: selectedGroupAiCharacters[i].custom_prompt + "\n" + group.description
|
||
}),
|
||
});
|
||
|
||
if (!response.ok) {
|
||
throw new Error('请求失败');
|
||
}
|
||
|
||
const reader = response.body?.getReader();
|
||
const decoder = new TextDecoder();
|
||
|
||
if (!reader) {
|
||
throw new Error('无法获取响应流');
|
||
}
|
||
|
||
let buffer = '';
|
||
let completeResponse = ''; // 用于跟踪完整的响应
|
||
// 添加超时控制
|
||
const timeout = 5000; // 5秒超时
|
||
while (true) {
|
||
//console.log("读取中")
|
||
const startTime = Date.now();
|
||
const { done, value } = await Promise.race([
|
||
reader.read(),
|
||
new Promise((_, reject) =>
|
||
setTimeout(() => reject(new Error('响应超时')), timeout - (Date.now() - startTime))
|
||
)
|
||
]);
|
||
|
||
if (Date.now() - startTime > timeout) {
|
||
reader.cancel();
|
||
|
||
throw new Error('响应超时');
|
||
}
|
||
|
||
if (done) {
|
||
//如果completeResponse为空,
|
||
if (completeResponse.trim() === "") {
|
||
completeResponse = "这个问题难倒我了,下一位。";
|
||
setMessages(prev => {
|
||
const newMessages = [...prev];
|
||
const aiMessageIndex = newMessages.findIndex(msg => msg.id === aiMessage.id);
|
||
if (aiMessageIndex !== -1) {
|
||
newMessages[aiMessageIndex] = {
|
||
...newMessages[aiMessageIndex],
|
||
content: completeResponse
|
||
};
|
||
}
|
||
return newMessages;
|
||
});}
|
||
break;
|
||
}
|
||
|
||
buffer += decoder.decode(value, { stream: true });
|
||
|
||
let newlineIndex;
|
||
while ((newlineIndex = buffer.indexOf('\n')) >= 0) {
|
||
const line = buffer.slice(0, newlineIndex);
|
||
buffer = buffer.slice(newlineIndex + 1);
|
||
|
||
if (line.startsWith('data: ')) {
|
||
try {
|
||
const data = JSON.parse(line.slice(6));
|
||
if (data.content) {
|
||
completeResponse += data.content;
|
||
//正则去掉前面的任何AI名称:格式
|
||
completeResponse = completeResponse.replace(new RegExp(`^(${allNames.join('|')}):`, 'i'), '');
|
||
setMessages(prev => {
|
||
const newMessages = [...prev];
|
||
const aiMessageIndex = newMessages.findIndex(msg => msg.id === aiMessage.id);
|
||
if (aiMessageIndex !== -1) {
|
||
newMessages[aiMessageIndex] = {
|
||
...newMessages[aiMessageIndex],
|
||
content: completeResponse
|
||
};
|
||
}
|
||
return newMessages;
|
||
});
|
||
}
|
||
|
||
} catch (e) {
|
||
console.error('解析响应数据失败:', e);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// 将当前AI的回复添加到消息历史中,供下一个AI使用
|
||
messageHistory.push({
|
||
role: 'user',
|
||
content: aiMessage.sender.name + ':' + completeResponse,
|
||
name: aiMessage.sender.name
|
||
});
|
||
|
||
// 等待一小段时间再开始下一个 AI 的回复
|
||
if (i < groupAiCharacters.length - 1) {
|
||
await new Promise(resolve => setTimeout(resolve, 1000));
|
||
}
|
||
|
||
} catch (error) {
|
||
console.error("发送消息失败:", error);
|
||
messageHistory.push({
|
||
role: 'user',
|
||
content: aiMessage.sender.name + "我遇到错误: " + error.message + ",下一位回答。",
|
||
name: aiMessage.sender.name
|
||
});
|
||
setMessages(prev => prev.map(msg =>
|
||
msg.id === aiMessage.id
|
||
? { ...msg, content: "我遇到错误: " + error.message + ",下一位回答。", isError: true }
|
||
: msg
|
||
));
|
||
}
|
||
}
|
||
|
||
setIsLoading(false);
|
||
};
|
||
|
||
const handleCancel = () => {
|
||
abortController.abort();
|
||
};
|
||
|
||
// 清理打字机效果
|
||
useEffect(() => {
|
||
return () => {
|
||
if (typewriterRef.current) {
|
||
clearInterval(typewriterRef.current);
|
||
}
|
||
};
|
||
}, []);
|
||
|
||
// 添加对聊天区域的引用
|
||
const chatAreaRef = useRef<HTMLDivElement>(null);
|
||
|
||
// 更新分享函数
|
||
const [showPoster, setShowPoster] = useState(false);
|
||
|
||
const handleShareChat = () => {
|
||
setShowPoster(true);
|
||
};
|
||
|
||
const [sidebarOpen, setSidebarOpen] = useState(false);
|
||
|
||
// 切换侧边栏状态的函数
|
||
const toggleSidebar = () => {
|
||
setSidebarOpen(!sidebarOpen);
|
||
};
|
||
|
||
// 处理群组选择
|
||
const handleSelectGroup = (index: number) => {
|
||
setSelectedGroupIndex(index);
|
||
const newGroup = groups[index];
|
||
setGroup(newGroup);
|
||
|
||
// 重新生成当前群组的 AI 角色,并按照 members 数组的顺序排序
|
||
const newGroupAiCharacters = generateAICharacters(newGroup.name)
|
||
.filter(character => newGroup.members.includes(character.id))
|
||
.sort((a, b) => {
|
||
return newGroup.members.indexOf(a.id) - newGroup.members.indexOf(b.id);
|
||
});
|
||
|
||
// 更新用户列表
|
||
setUsers([
|
||
{ id: 1, name: "我" },
|
||
...newGroupAiCharacters
|
||
]);
|
||
setIsGroupDiscussionMode(newGroup.isGroupDiscussionMode);
|
||
|
||
// 重置消息
|
||
setMessages([]);
|
||
|
||
// 可选:关闭侧边栏(在移动设备上)
|
||
if (window.innerWidth < 768) {
|
||
setSidebarOpen(false);
|
||
}
|
||
};
|
||
|
||
return (
|
||
<>
|
||
<KaTeXStyle />
|
||
<div className="fixed inset-0 bg-gradient-to-br from-orange-50 via-orange-50/70 to-orange-100 flex items-start md:items-center justify-center overflow-hidden">
|
||
<div className="h-full flex bg-white w-full mx-auto relative shadow-xl md:max-w-5xl md:h-[96dvh] md:my-auto md:rounded-lg">
|
||
{/* 传递 selectedGroupIndex 和 onSelectGroup 回调给 Sidebar */}
|
||
<Sidebar
|
||
isOpen={sidebarOpen}
|
||
toggleSidebar={toggleSidebar}
|
||
selectedGroupIndex={selectedGroupIndex}
|
||
onSelectGroup={handleSelectGroup}
|
||
/>
|
||
|
||
{/* 聊天主界面 */}
|
||
<div className="flex flex-col flex-1">
|
||
{/* Header */}
|
||
<header className="bg-white shadow flex-none md:rounded-t-lg">
|
||
<div className="flex items-center justify-between px-0 py-1.5">
|
||
{/* 左侧群组信息 */}
|
||
<div className="flex items-center md:px-2.5">
|
||
<div
|
||
className="md:hidden flex items-center justify-center m-1 cursor-pointer"
|
||
onClick={toggleSidebar}
|
||
>
|
||
<ChevronLeft className="w-6 h-6" />
|
||
</div>
|
||
|
||
<h1 className="font-medium text-base -ml-1">{group.name}({users.length})</h1>
|
||
</div>
|
||
|
||
|
||
{/* 右侧头像组和按钮 */}
|
||
<div className="flex items-center">
|
||
{/* 广告位 手机端不展示*/}
|
||
<div className="hidden md:block">
|
||
<AdBanner show={showAd} closeAd={() => setShowAd(false)} />
|
||
</div>
|
||
<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>
|
||
<Button variant="ghost" size="icon" onClick={() => setShowMembers(true)}>
|
||
<Settings2 className="w-5 h-5" />
|
||
</Button>
|
||
</div>
|
||
</div>
|
||
</header>
|
||
|
||
{/* Main Chat Area */}
|
||
<div className="flex-1 overflow-hidden bg-gray-100">
|
||
|
||
<ScrollArea className={`h-full ${!showAd ? 'px-2 py-1' : ''} md:px-2 md:py-1`} ref={chatAreaRef}>
|
||
<div className="md:hidden">
|
||
<AdBannerMobile show={showAd} closeAd={() => setShowAd(false)} />
|
||
</div>
|
||
<div className="space-y-4">
|
||
{messages.map((message) => (
|
||
<div key={message.id}
|
||
className={`flex items-start gap-2 ${message.sender.name === "我" ? "justify-end" : ""}`}>
|
||
{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 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 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>
|
||
</ScrollArea>
|
||
</div>
|
||
|
||
{/* Input Area */}
|
||
<div className="bg-white border-t py-3 px-2 md:rounded-b-lg">
|
||
<div className="flex gap-1 pb-[env(safe-area-inset-bottom)]">
|
||
{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>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
{/* Members Management Dialog */}
|
||
<MembersManagement
|
||
showMembers={showMembers}
|
||
setShowMembers={setShowMembers}
|
||
users={users}
|
||
mutedUsers={mutedUsers}
|
||
handleToggleMute={handleToggleMute}
|
||
isGroupDiscussionMode={isGroupDiscussionMode}
|
||
onToggleGroupDiscussion={() => setIsGroupDiscussionMode(!isGroupDiscussionMode)}
|
||
getAvatarData={getAvatarData}
|
||
/>
|
||
</div>
|
||
|
||
{/* 添加 SharePoster 组件 */}
|
||
<SharePoster
|
||
isOpen={showPoster}
|
||
onClose={() => setShowPoster(false)}
|
||
chatAreaRef={chatAreaRef}
|
||
/>
|
||
</>
|
||
);
|
||
};
|
||
|
||
export default ChatUI; |