From 1287d6ee65e83b03c0d941295ae8e550b6e5612a Mon Sep 17 00:00:00 2001 From: Bivekich Date: Mon, 21 Jul 2025 14:53:17 +0300 Subject: [PATCH] =?UTF-8?q?=D0=9E=D0=B1=D0=BD=D0=BE=D0=B2=D0=BB=D0=B5?= =?UTF-8?q?=D0=BD=D1=8B=20=D0=BA=D0=BE=D0=BC=D0=BF=D0=BE=D0=BD=D0=B5=D0=BD?= =?UTF-8?q?=D1=82=D1=8B=20=D0=BC=D0=B5=D1=81=D1=81=D0=B5=D0=BD=D0=B4=D0=B6?= =?UTF-8?q?=D0=B5=D1=80=D0=B0:=20=D0=B8=D0=B7=D0=BC=D0=B5=D0=BD=D0=B5?= =?UTF-8?q?=D0=BD=20=D0=B8=D0=BD=D1=82=D0=B5=D1=80=D0=B2=D0=B0=D0=BB=20?= =?UTF-8?q?=D0=BE=D0=B1=D0=BD=D0=BE=D0=B2=D0=BB=D0=B5=D0=BD=D0=B8=D1=8F=20?= =?UTF-8?q?=D1=81=D0=BF=D0=B8=D1=81=D0=BA=D0=B0=20=D1=87=D0=B0=D1=82=D0=BE?= =?UTF-8?q?=D0=B2=20=D0=BD=D0=B0=201=20=D0=BC=D0=B8=D0=BD=D1=83=D1=82?= =?UTF-8?q?=D1=83=20=D0=B8=20=D0=B8=D0=B7=D0=BC=D0=B5=D0=BD=D0=B5=D0=BD?= =?UTF-8?q?=D0=B0=20=D0=BF=D0=BE=D0=BB=D0=B8=D1=82=D0=B8=D0=BA=D0=B0=20?= =?UTF-8?q?=D0=B2=D1=8B=D0=B1=D0=BE=D1=80=D0=BA=D0=B8=20=D0=BD=D0=B0=20'ca?= =?UTF-8?q?che-first'=20=D0=B4=D0=BB=D1=8F=20=D0=BF=D0=BE=D0=B2=D1=8B?= =?UTF-8?q?=D1=88=D0=B5=D0=BD=D0=B8=D1=8F=20=D1=81=D1=82=D0=B0=D0=B1=D0=B8?= =?UTF-8?q?=D0=BB=D1=8C=D0=BD=D0=BE=D1=81=D1=82=D0=B8.=20=D0=92=20=D0=BA?= =?UTF-8?q?=D0=BE=D0=BC=D0=BF=D0=BE=D0=BD=D0=B5=D0=BD=D1=82=D0=B5=20Messen?= =?UTF-8?q?gerChat=20=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB=D0=B5=D0=BD?= =?UTF-8?q?=D0=B0=20=D0=BF=D0=BE=D0=B4=D0=B4=D0=B5=D1=80=D0=B6=D0=BA=D0=B0?= =?UTF-8?q?=20=D0=B2=D0=BB=D0=BE=D0=B6=D0=B5=D0=BD=D0=B8=D0=B9=20=D0=B8=20?= =?UTF-8?q?=D1=83=D0=BB=D1=83=D1=87=D1=88=D0=B5=D0=BD=D0=B0=20=D0=BB=D0=BE?= =?UTF-8?q?=D0=B3=D0=B8=D0=BA=D0=B0=20=D0=BE=D1=82=D0=BC=D0=B5=D1=82=D0=BA?= =?UTF-8?q?=D0=B8=20=D1=81=D0=BE=D0=BE=D0=B1=D1=89=D0=B5=D0=BD=D0=B8=D0=B9?= =?UTF-8?q?=20=D0=BA=D0=B0=D0=BA=20=D0=BF=D1=80=D0=BE=D1=87=D0=B8=D1=82?= =?UTF-8?q?=D0=B0=D0=BD=D0=BD=D1=8B=D1=85=20=D1=81=20=D0=BF=D0=BB=D0=B0?= =?UTF-8?q?=D0=B2=D0=BD=D1=8B=D0=BC=20=D0=BE=D0=B1=D0=BD=D0=BE=D0=B2=D0=BB?= =?UTF-8?q?=D0=B5=D0=BD=D0=B8=D0=B5=D0=BC.=20=D0=92=20=D0=BA=D0=BE=D0=BC?= =?UTF-8?q?=D0=BF=D0=BE=D0=BD=D0=B5=D0=BD=D1=82=D0=B5=20MessengerConversat?= =?UTF-8?q?ions=20=D1=83=D0=BB=D1=83=D1=87=D1=88=D0=B5=D0=BD=D0=BE=20?= =?UTF-8?q?=D0=BE=D1=82=D0=BE=D0=B1=D1=80=D0=B0=D0=B6=D0=B5=D0=BD=D0=B8?= =?UTF-8?q?=D0=B5=20=D0=B7=D0=B0=D0=B3=D1=80=D1=83=D0=B7=D0=BA=D0=B8=20?= =?UTF-8?q?=D0=B8=20=D0=B8=D0=BD=D0=B4=D0=B8=D0=BA=D0=B0=D1=82=D0=BE=D1=80?= =?UTF-8?q?=D0=BE=D0=B2=20=D0=BD=D0=B5=D0=BF=D1=80=D0=BE=D1=87=D0=B8=D1=82?= =?UTF-8?q?=D0=B0=D0=BD=D0=BD=D1=8B=D1=85=20=D1=81=D0=BE=D0=BE=D0=B1=D1=89?= =?UTF-8?q?=D0=B5=D0=BD=D0=B8=D0=B9.=20=D0=A2=D0=B0=D0=BA=D0=B6=D0=B5=20?= =?UTF-8?q?=D0=B4=D0=BE=D0=B1=D0=B0=D0=B2=D0=BB=D0=B5=D0=BD=20=D0=B0=D0=B2?= =?UTF-8?q?=D1=82=D0=BE=D1=84=D0=BE=D0=BA=D1=83=D1=81=20=D0=BD=D0=B0=20?= =?UTF-8?q?=D0=BF=D0=BE=D0=BB=D0=B5=20=D0=B2=D0=B2=D0=BE=D0=B4=D0=B0=20?= =?UTF-8?q?=D0=BF=D1=80=D0=B8=20=D0=BE=D1=82=D0=BA=D1=80=D1=8B=D1=82=D0=B8?= =?UTF-8?q?=D0=B8=20=D1=87=D0=B0=D1=82=D0=B0.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/components/dashboard/sidebar.tsx | 5 +- .../messenger/messenger-attachments.tsx | 315 +++++++++++ .../messenger-chat-with-attachments.tsx | 518 ++++++++++++++++++ src/components/messenger/messenger-chat.tsx | 40 +- .../messenger/messenger-conversations.tsx | 50 +- .../messenger/messenger-dashboard.tsx | 12 +- 6 files changed, 890 insertions(+), 50 deletions(-) create mode 100644 src/components/messenger/messenger-attachments.tsx create mode 100644 src/components/messenger/messenger-chat-with-attachments.tsx diff --git a/src/components/dashboard/sidebar.tsx b/src/components/dashboard/sidebar.tsx index 60ed55f..69001b9 100644 --- a/src/components/dashboard/sidebar.tsx +++ b/src/components/dashboard/sidebar.tsx @@ -30,9 +30,10 @@ export function Sidebar() { // Загружаем список чатов для подсчета непрочитанных сообщений const { data: conversationsData } = useQuery(GET_CONVERSATIONS, { - pollInterval: 10000, // Обновляем каждые 10 секунд - fetchPolicy: 'cache-and-network', + pollInterval: 60000, // Обновляем каждую минуту в сайдбаре - этого достаточно + fetchPolicy: 'cache-first', errorPolicy: 'ignore', // Игнорируем ошибки чтобы не ломать сайдбар + notifyOnNetworkStatusChange: false, // Плавные обновления без мерцания }) const conversations = conversationsData?.conversations || [] diff --git a/src/components/messenger/messenger-attachments.tsx b/src/components/messenger/messenger-attachments.tsx new file mode 100644 index 0000000..6f81efb --- /dev/null +++ b/src/components/messenger/messenger-attachments.tsx @@ -0,0 +1,315 @@ +"use client" + +import { useState } from 'react' +import { useQuery } from '@apollo/client' +import { GET_MESSAGES } from '@/graphql/queries' +import { useAuth } from '@/hooks/useAuth' +import { Card } from '@/components/ui/card' +import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs' +import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar' +import { Badge } from '@/components/ui/badge' +import { + FileText, + Image, + Music, + Video, + Download, + Calendar, + User +} from 'lucide-react' + +interface Organization { + id: string + inn: string + name?: string + fullName?: string + type: 'FULFILLMENT' | 'SELLER' | 'LOGIST' | 'WHOLESALE' + users?: Array<{ id: string, avatar?: string, managerName?: string }> +} + +interface Message { + id: string + content?: string + type?: 'TEXT' | 'VOICE' | 'IMAGE' | 'FILE' + voiceUrl?: string + voiceDuration?: number + fileUrl?: string + fileName?: string + fileSize?: number + fileType?: string + senderId: string + senderOrganization: Organization + receiverOrganization: Organization + createdAt: string + isRead: boolean +} + +interface MessengerAttachmentsProps { + counterparty: Organization +} + +export function MessengerAttachments({ counterparty }: MessengerAttachmentsProps) { + const { user } = useAuth() + const [activeTab, setActiveTab] = useState('all') + + // Загружаем все сообщения для получения вложений + const { data: messagesData, loading } = useQuery(GET_MESSAGES, { + variables: { counterpartyId: counterparty.id, limit: 1000 }, // Увеличиваем лимит для получения всех файлов + fetchPolicy: 'cache-first', + }) + + const messages: Message[] = messagesData?.messages || [] + + // Фильтруем только сообщения с вложениями + const attachmentMessages = messages.filter(msg => + msg.type && ['VOICE', 'IMAGE', 'FILE'].includes(msg.type) && msg.fileUrl + ) + + // Группируем по типам + const imageMessages = attachmentMessages.filter(msg => msg.type === 'IMAGE') + const voiceMessages = attachmentMessages.filter(msg => msg.type === 'VOICE') + const fileMessages = attachmentMessages.filter(msg => msg.type === 'FILE') + + const getOrganizationName = (org: Organization) => { + return org.name || org.fullName || 'Организация' + } + + const getManagerName = (org: Organization) => { + return org.users?.[0]?.managerName || 'Управляющий' + } + + const getInitials = (org: Organization) => { + const name = getOrganizationName(org) + return name.charAt(0).toUpperCase() + } + + const formatFileSize = (bytes?: number) => { + if (!bytes) return '0 B' + const k = 1024 + const sizes = ['B', 'KB', 'MB', 'GB'] + const i = Math.floor(Math.log(bytes) / Math.log(k)) + return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i] + } + + const formatDate = (dateString: string) => { + const date = new Date(dateString) + return date.toLocaleDateString('ru-RU', { + day: '2-digit', + month: '2-digit', + year: 'numeric' + }) + } + + const formatTime = (dateString: string) => { + const date = new Date(dateString) + return date.toLocaleTimeString('ru-RU', { + hour: '2-digit', + minute: '2-digit' + }) + } + + const handleDownload = (fileUrl: string, fileName: string) => { + const link = document.createElement('a') + link.href = fileUrl + link.download = fileName + document.body.appendChild(link) + link.click() + document.body.removeChild(link) + } + + const renderFileIcon = (fileType?: string) => { + if (!fileType) return + + if (fileType.startsWith('image/')) return + if (fileType.startsWith('audio/')) return + if (fileType.startsWith('video/')) return