@linktr.ee/messaging-react 1.19.0 โ†’ 1.19.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/providers/MessagingProvider.tsx","../src/hooks/useMessaging.ts","../src/utils/formatRelativeTime.ts","../src/components/Avatar/getAvatarEmoji.ts","../src/components/Avatar/index.tsx","../src/components/CustomMessage/MessageTag.tsx","../src/components/ChannelList/CustomChannelPreview.tsx","../src/components/ChannelList/index.tsx","../src/components/ActionButton/index.tsx","../src/components/IconButton/index.tsx","../src/components/CloseButton/index.tsx","../src/components/CustomDateSeparator/index.tsx","../src/components/CustomMessage/index.tsx","../src/components/CustomLinkPreviewList/index.tsx","../src/components/CustomMessageInput/index.tsx","../src/components/CustomSystemMessage/index.tsx","../src/components/MessagingShell/ChannelEmptyState.tsx","../src/components/Loading/index.tsx","../src/components/MessagingShell/LoadingState.tsx","../src/components/ChannelView.tsx","../src/components/SearchInput/index.tsx","../src/components/ParticipantPicker/index.tsx","../src/components/MessagingShell/EmptyState.tsx","../src/components/MessagingShell/ErrorState.tsx","../src/components/MessagingShell/index.tsx","../src/components/FaqList/FaqListItem.tsx","../src/components/FaqList/index.tsx","../src/hooks/useParticipants.ts"],"sourcesContent":["import { StreamChatService } from '@linktr.ee/messaging-core'\nimport React, {\n createContext,\n useContext,\n useEffect,\n useState,\n useRef,\n useCallback,\n} from 'react'\nimport type { StreamChat } from 'stream-chat'\nimport { Chat } from 'stream-chat-react'\n\nimport type { MessagingProviderProps, MessagingCapabilities } from '../types'\n\n/**\n * Context value for messaging state and service\n */\nexport interface MessagingContextValue {\n service: StreamChatService | null\n client: StreamChat | null // Stream Chat client\n isConnected: boolean\n isLoading: boolean\n error: string | null\n capabilities: MessagingCapabilities\n refreshConnection: () => Promise<void>\n debug: boolean\n}\n\nconst MessagingContext = createContext<MessagingContextValue>({\n service: null,\n client: null,\n isConnected: false,\n isLoading: false,\n error: null,\n capabilities: {},\n refreshConnection: async () => {},\n debug: false,\n})\n\n/**\n * Hook to access messaging context\n */\nexport const useMessagingContext = () => useContext(MessagingContext)\n\n/**\n * Provider component that wraps messaging-core with React state management\n */\nexport const MessagingProvider: React.FC<MessagingProviderProps> = ({\n children,\n user,\n serviceConfig,\n apiKey,\n capabilities = {},\n debug = false,\n}) => {\n // Create debug logger that respects the debug prop\n const debugLog = useCallback(\n (message: string, ...args: unknown[]) => {\n if (debug) {\n console.log(`๐Ÿ”ฅ [MessagingProvider] ${message}`, ...args)\n }\n },\n [debug]\n )\n\n debugLog('๐Ÿ”„ RENDER START', {\n userId: user?.id,\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfig: !!serviceConfig,\n capabilities: Object.keys(capabilities),\n })\n\n const [service, setService] = useState<StreamChatService | null>(null)\n const [client, setClient] = useState<StreamChat | null>(null)\n const [isConnected, setIsConnected] = useState(false)\n const [isLoading, setIsLoading] = useState(false)\n const [error, setError] = useState<string | null>(null)\n\n // Prevent multiple concurrent connection attempts\n const connectingRef = useRef(false)\n\n // Track renders and prop changes\n const prevPropsRef = useRef({\n userId: user?.id,\n apiKey,\n serviceConfig,\n capabilities,\n })\n const renderCountRef = useRef(0)\n renderCountRef.current++\n\n debugLog('๐Ÿ“Š RENDER INFO', {\n renderCount: renderCountRef.current,\n currentProps: { userId: user?.id, apiKey: apiKey?.substring(0, 8) + '...' },\n propChanges: {\n userChanged: prevPropsRef.current.userId !== user?.id,\n apiKeyChanged: prevPropsRef.current.apiKey !== apiKey,\n serviceConfigChanged:\n prevPropsRef.current.serviceConfig !== serviceConfig,\n capabilitiesChanged: prevPropsRef.current.capabilities !== capabilities,\n },\n })\n\n prevPropsRef.current = {\n userId: user?.id,\n apiKey,\n serviceConfig,\n capabilities,\n }\n\n // Initialize service when config changes\n useEffect(() => {\n const currentRender = renderCountRef.current\n debugLog('๐Ÿ”ง SERVICE INIT EFFECT TRIGGERED', {\n renderCount: currentRender,\n apiKey: !!apiKey,\n serviceConfig: !!serviceConfig,\n dependencies: {\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfigRef: serviceConfig,\n serviceConfigStable:\n prevPropsRef.current.serviceConfig === serviceConfig,\n apiKeyStable: prevPropsRef.current.apiKey === apiKey,\n },\n })\n\n if (!apiKey || !serviceConfig) {\n debugLog('โš ๏ธ SERVICE INIT SKIPPED', {\n renderCount: currentRender,\n reason: 'Missing apiKey or serviceConfig',\n })\n return\n }\n\n debugLog('๐Ÿš€ CREATING NEW SERVICE', {\n renderCount: currentRender,\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfigChanged:\n prevPropsRef.current.serviceConfig !== serviceConfig,\n })\n\n const newService = new StreamChatService({\n ...serviceConfig,\n apiKey,\n debug,\n })\n\n setService(newService)\n debugLog('โœ… SERVICE SET', {\n renderCount: currentRender,\n serviceInstance: !!newService,\n })\n\n return () => {\n debugLog('๐Ÿงน SERVICE CLEANUP', {\n renderCount: currentRender,\n reason: 'Effect cleanup',\n })\n newService.disconnectUser().catch(console.error)\n }\n }, [apiKey, serviceConfig, debug, debugLog]) // Use serviceConfig object directly, not individual properties\n\n // Track if we've already connected this user with this service to prevent duplicate connections\n const connectedUserRef = useRef<{\n serviceId: StreamChatService\n userId: string\n } | null>(null)\n\n // Connect user when service and user are available\n useEffect(() => {\n debugLog('๐Ÿ”— USER CONNECTION EFFECT TRIGGERED', {\n hasService: !!service,\n hasUser: !!user,\n userId: user?.id,\n isConnecting: connectingRef.current,\n isConnected: isConnected,\n dependencies: { service: !!service, userId: user?.id },\n })\n\n if (!service || !user) {\n debugLog('โš ๏ธ USER CONNECTION SKIPPED', 'Missing service or user')\n return\n }\n\n if (connectingRef.current) {\n debugLog('โš ๏ธ USER CONNECTION SKIPPED', 'Already connecting')\n return\n }\n\n // Check if we've already connected this exact user with this exact service instance\n if (\n connectedUserRef.current?.serviceId === service &&\n connectedUserRef.current?.userId === user.id\n ) {\n debugLog(\n 'โš ๏ธ USER CONNECTION SKIPPED',\n 'Already connected this user with this service'\n )\n return\n }\n\n const connectUser = async () => {\n debugLog('๐Ÿš€ STARTING USER CONNECTION', { userId: user.id })\n connectingRef.current = true\n setIsLoading(true)\n setError(null)\n\n try {\n debugLog('๐Ÿ“ž CALLING SERVICE.CONNECTUSER', { userId: user.id })\n const streamClient = await service.connectUser(user)\n setClient(streamClient)\n setIsConnected(true)\n connectedUserRef.current = { serviceId: service, userId: user.id } // Mark as connected\n debugLog('โœ… USER CONNECTION SUCCESS', {\n userId: user.id,\n clientId: streamClient.userID,\n })\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'Connection failed'\n setError(errorMessage)\n debugLog('โŒ USER CONNECTION ERROR', {\n userId: user.id,\n error: errorMessage,\n })\n } finally {\n setIsLoading(false)\n connectingRef.current = false\n debugLog('๐Ÿ”„ USER CONNECTION FINISHED', {\n userId: user.id,\n isConnected,\n })\n }\n }\n\n connectUser()\n }, [service, user, debugLog, isConnected]) // Remove isConnected to prevent circular dependency\n\n // Disconnect when user is removed (cleanup effect)\n useEffect(() => {\n debugLog('๐Ÿ”Œ CLEANUP EFFECT REGISTERED', {\n hasService: !!service,\n isConnected,\n })\n return () => {\n if (service && isConnected) {\n debugLog(\n '๐Ÿงน CLEANUP EFFECT TRIGGERED',\n 'Cleaning up connection on unmount'\n )\n connectedUserRef.current = null // Reset connection tracking\n service.disconnectUser().catch(console.error)\n } else {\n debugLog('๐Ÿ”‡ CLEANUP EFFECT SKIPPED', {\n hasService: !!service,\n isConnected,\n })\n }\n }\n }, [service, isConnected, debugLog])\n\n const refreshConnection = useCallback(async () => {\n debugLog('๐Ÿ”„ REFRESH CONNECTION CALLED', {\n hasService: !!service,\n hasUser: !!user,\n })\n\n if (!service || !user) {\n debugLog('โš ๏ธ REFRESH CONNECTION SKIPPED', 'Missing service or user')\n return\n }\n\n debugLog('๐Ÿš€ STARTING CONNECTION REFRESH', { userId: user.id })\n setIsLoading(true)\n try {\n debugLog('๐Ÿ”Œ DISCONNECTING FOR REFRESH')\n await service.disconnectUser()\n debugLog('๐Ÿ“ž RECONNECTING FOR REFRESH')\n const streamClient = await service.connectUser(user)\n setClient(streamClient)\n setIsConnected(true)\n setError(null)\n debugLog('โœ… CONNECTION REFRESH SUCCESS', { userId: user.id })\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Refresh failed'\n setError(errorMessage)\n debugLog('โŒ CONNECTION REFRESH ERROR', {\n userId: user.id,\n error: errorMessage,\n })\n } finally {\n setIsLoading(false)\n debugLog('๐Ÿ”„ CONNECTION REFRESH FINISHED', { userId: user.id })\n }\n }, [service, user, debugLog])\n\n // Memoize context value to prevent unnecessary re-renders\n const contextValue: MessagingContextValue = React.useMemo(() => {\n debugLog('๐Ÿ’ซ CONTEXT VALUE MEMOIZATION', {\n hasService: !!service,\n hasClient: !!client,\n isConnected,\n isLoading,\n hasError: !!error,\n capabilitiesKeys: Object.keys(capabilities),\n })\n\n return {\n service,\n client,\n isConnected,\n isLoading,\n error,\n capabilities,\n refreshConnection,\n debug,\n }\n }, [\n service,\n client,\n isConnected,\n isLoading,\n error,\n capabilities,\n refreshConnection,\n debug,\n debugLog,\n ])\n\n debugLog('๐Ÿ”„ RENDER END', {\n renderCount: renderCountRef.current,\n willRenderChat: !!(client && isConnected),\n contextValueReady: !!contextValue,\n })\n\n return (\n <MessagingContext.Provider value={contextValue}>\n {client && isConnected ? (\n <Chat\n client={client}\n customClasses={{\n channelList:\n 'str-chat__channel-list str-chat__channel-list-react bg-transparent lg:border-r-2 border-r-0 border-[#0000000A]',\n }}\n >\n {children}\n </Chat>\n ) : (\n children\n )}\n </MessagingContext.Provider>\n )\n}\n","import { useMessagingContext } from '../providers/MessagingProvider';\nimport type { MessagingContextValue } from '../providers/MessagingProvider';\n\n/**\n * Hook to access messaging service and state\n */\nexport const useMessaging = (): MessagingContextValue => {\n return useMessagingContext();\n};\n","/**\n * Get the number of days between two dates (calendar days in UTC, not 24-hour periods)\n * Uses UTC to ensure consistent day calculations globally since messages are stored in UTC\n */\nconst getDaysDifference = (date1: Date, date2: Date): number => {\n const d1 = new Date(\n Date.UTC(date1.getUTCFullYear(), date1.getUTCMonth(), date1.getUTCDate())\n )\n const d2 = new Date(\n Date.UTC(date2.getUTCFullYear(), date2.getUTCMonth(), date2.getUTCDate())\n )\n const diffTime = d2.getTime() - d1.getTime()\n return Math.floor(diffTime / (1000 * 60 * 60 * 24))\n}\n\n/**\n * Format a date - shows time for today, relative time for older messages\n * (e.g., \"Just now\", \"2:08 PM\" for today, \"Yesterday\" for yesterday, \"2d\" for 2 days ago)\n */\nexport const formatRelativeTime = (date: Date): string => {\n const now = new Date()\n const diffInSeconds = Math.floor((now.getTime() - date.getTime()) / 1000)\n\n // Less than 1 minute\n if (diffInSeconds < 60) {\n return 'Just now'\n }\n\n // Check if it's today (same calendar day)\n const daysDiff = getDaysDifference(date, now)\n\n // If today, show time in 12-hour format (e.g., \"8:40 PM\")\n if (daysDiff === 0) {\n return date.toLocaleTimeString([], {\n hour: 'numeric',\n minute: '2-digit',\n hour12: true,\n })\n }\n\n // Yesterday\n if (daysDiff === 1) {\n return 'Yesterday'\n }\n\n // Less than 7 days - show days\n if (daysDiff < 7) {\n return `${daysDiff}d`\n }\n\n // Less than 4 weeks - show weeks\n if (daysDiff < 28) {\n const weeks = Math.floor(daysDiff / 7)\n return `${weeks}w`\n }\n\n // More than 4 weeks - show date as MM/DD/YY\n return date.toLocaleDateString('en-US', {\n month: 'numeric',\n day: 'numeric',\n year: '2-digit',\n })\n}\n","/**\n * Generate a fruit emoji based on a string id\n * Returns a consistent fruit emoji for the same id\n */\nconst EMOJIS = [\n '๐ŸŽ', // Apple\n '๐ŸŒ', // Banana\n '๐Ÿ‡', // Grape\n '๐ŸŠ', // Orange\n '๐Ÿ“', // Strawberry\n '๐Ÿฅฅ', // Coconut\n '๐Ÿ’', // Cherry\n '๐Ÿฅญ', // Mango\n '๐Ÿ‰', // Watermelon\n '๐Ÿ‹', // Lemon\n '๐Ÿฅ', // Kiwi\n '๐Ÿซ’', // Olive\n '๐Ÿˆ', // Melon\n]\n\n/**\n * Simple hash function to convert string to number\n */\nfunction hashString(str: string): number {\n let hash = 0\n for (let i = 0; i < str.length; i++) {\n const char = str.charCodeAt(i)\n hash = (hash << 5) - hash + char\n hash = hash & hash // Convert to 32-bit integer\n }\n return Math.abs(hash)\n}\n\n/**\n * Get a fruit emoji based on an id string\n * @param id - The string id to generate emoji from\n * @returns A fruit emoji string\n */\nexport function getAvatarEmoji(id: string): string {\n const hash = hashString(id)\n const index = hash % EMOJIS.length\n return EMOJIS[index]\n}\n\n","import classNames from 'classnames'\nimport React from 'react'\n\nimport { getAvatarEmoji } from './getAvatarEmoji'\n\nexport interface AvatarProps {\n id: string\n name: string\n image?: string\n size?: number\n className?: string\n shape?: 'squircle' | 'circle'\n}\n\n/**\n * Avatar component that displays a user image or colored initial fallback\n */\nexport const Avatar: React.FC<AvatarProps> = ({\n id,\n image,\n size = 40,\n className,\n shape = 'squircle',\n}) => {\n const emoji = getAvatarEmoji(id)\n\n // Determine font size based on avatar size\n const getFontSizeClass = () => {\n if (size < 32) return 'text-xs'\n if (size < 56) return 'text-sm'\n return 'text-lg'\n }\n\n const fontSizeClass = getFontSizeClass()\n\n const borderStyle =\n shape === 'circle'\n ? { borderRadius: '50%' }\n : {\n borderRadius: '33%',\n 'corner-shape': 'superellipse(1.3)',\n }\n\n return (\n <div\n className={classNames('flex-shrink-0 overflow-hidden', className)}\n style={{\n width: `${size}px`,\n height: `${size}px`,\n ...borderStyle,\n }}\n >\n {image ? (\n <img\n src={image}\n alt=\"\"\n className=\"h-full w-full object-cover aspect-square\"\n />\n ) : (\n <div\n aria-hidden=\"true\"\n className={classNames(\n 'avatar-fallback flex h-full w-full items-center justify-center font-semibold bg-[#E6E5E3] select-none transition-colors',\n fontSizeClass\n )}\n >\n {emoji}\n </div>\n )}\n </div>\n )\n}\n","import { GiftIcon } from '@phosphor-icons/react'\nimport { LocalMessage } from 'stream-chat'\n\ninterface MessageTagProps {\n message: LocalMessage\n /** When true, renders as a standalone bubble instead of a small tag */\n standalone?: boolean\n}\n\nconst SparkleIcon = () => (\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path\n d=\"M10.003 5a.705.705 0 0 1-.469.67L6.7 6.7 5.67 9.535a.715.715 0 0 1-1.34 0L3.3 6.7.466 5.67a.715.715 0 0 1 0-1.34L3.3 3.3 4.33.466a.715.715 0 0 1 1.34 0L6.7 3.3l2.834 1.03a.705.705 0 0 1 .469.67\"\n fill=\"currentColor\"\n />\n </svg>\n)\n\n/** Check if a message is a tip based on metadata */\nexport const isTipMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_TIP'\n}\n\n/** Check if a message is a paid message based on metadata */\nexport const isPaidMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_PAID'\n}\n\n/** Check if a message is a chatbot message based on metadata */\nexport const isChatbotMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_CHATBOT'\n}\n\n/** Check if a message has a tip/paid tag (both render the same) */\nexport const isTipOrPaidMessage = (message: LocalMessage): boolean => {\n return isTipMessage(message) || isPaidMessage(message)\n}\n\n/** Check if a message is a tip/paid-only message (no text) */\nexport const isTipOnlyMessage = (message: LocalMessage): boolean => {\n return isTipOrPaidMessage(message) && !message.text?.trim()\n}\n\nexport const MessageTag = ({\n message,\n standalone = false,\n}: MessageTagProps) => {\n const isTipOrPaid = isTipOrPaidMessage(message)\n const isChatbot = isChatbotMessage(message)\n\n if (!isTipOrPaid && !isChatbot) {\n return null\n }\n\n if (isTipOrPaid) {\n const amountText = message.metadata?.amount_text\n if (!amountText) return null\n\n const className = standalone\n ? 'message-tip-standalone'\n : 'message-tag message-tag--tip'\n\n const label = standalone\n ? `${amountText} tip`\n : `Delivered with ${amountText} tip`\n\n return (\n <div className={className}>\n <GiftIcon size={standalone ? 14 : 12} />\n <span>{label}</span>\n </div>\n )\n }\n\n // Chatbot tag\n return (\n <div className=\"message-tag message-tag--chatbot\">\n <span className=\"message-tag__icon\" style={{ marginTop: -1 }}>\n <SparkleIcon />\n </span>\n <span className=\"message-tag__label\">Chatbot</span>\n </div>\n )\n}\n","import classNames from 'classnames'\nimport React from 'react'\nimport { Channel, LocalMessage } from 'stream-chat'\nimport { ChannelPreviewUIComponentProps } from 'stream-chat-react'\n\nimport { formatRelativeTime } from '../../utils/formatRelativeTime'\nimport { Avatar } from '../Avatar'\nimport { isChatbotMessage } from '../CustomMessage/MessageTag'\n\ntype CustomChannelPreviewProps = ChannelPreviewUIComponentProps & {\n selectedChannel?: Channel | null\n onChannelSelect: (channel: Channel) => void\n debug?: boolean\n renderMessagePreview?: (\n message: LocalMessage | undefined,\n defaultPreview?: string\n ) => React.ReactNode\n}\n\n/**\n * Custom channel preview that handles selection\n */\nconst CustomChannelPreview = React.memo<CustomChannelPreviewProps>(\n ({\n channel,\n selectedChannel,\n onChannelSelect,\n debug = false,\n unread,\n renderMessagePreview,\n }) => {\n const isSelected = selectedChannel?.id === channel?.id\n\n const handleClick = () => {\n if (channel) {\n onChannelSelect(channel)\n }\n }\n\n // Get participant info\n const members = Object.values(channel?.state?.members || {})\n const participant = members.find(\n (member) => member.user?.id && member.user.id !== channel?._client?.userID\n )\n const participantName = participant?.user?.name || 'Conversation'\n const participantImage = participant?.user?.image\n\n // Get last message and format timestamp\n const lastMessage =\n channel?.state?.messages?.[channel.state.messages.length - 1]\n\n const getLastMessageText = () => {\n if (lastMessage?.text) return lastMessage.text\n\n const attachment = lastMessage?.attachments?.[0]\n if (attachment) {\n // Link previews - show the actual URL\n if (attachment.og_scrape_url) return attachment.og_scrape_url\n\n // Type-based detection\n if (attachment.type === 'image') return '๐Ÿ“ท Sent an image'\n if (attachment.type === 'video') return '๐ŸŽฅ Sent a video'\n if (attachment.type === 'audio') return '๐ŸŽต Sent audio'\n if (attachment.type === 'file') return '๐Ÿ“Ž Sent a file'\n\n // Fallback\n return '๐Ÿ“Ž Sent an attachment'\n }\n\n return 'No messages yet'\n }\n\n const lastMessageText = getLastMessageText()\n const lastMessageTime = lastMessage?.created_at\n ? formatRelativeTime(new Date(lastMessage.created_at))\n : ''\n const isLastMessageFromChatbot = lastMessage\n ? isChatbotMessage(lastMessage)\n : false\n\n const messagePreview = renderMessagePreview\n ? renderMessagePreview(lastMessage, lastMessageText)\n : `${isLastMessageFromChatbot ? 'โœจ ' : ''}${lastMessageText}`\n\n // Use the unread prop passed by Stream Chat (reactive and updates automatically)\n const unreadCount = unread ?? 0\n\n if (debug) {\n console.log('๐Ÿ“บ [ChannelList] ๐Ÿ“‹ CHANNEL PREVIEW RENDER', {\n channelId: channel?.id,\n isSelected,\n participantName,\n unreadCount,\n hasTimestamp: !!lastMessageTime,\n })\n }\n\n return (\n <button\n type=\"button\"\n onClick={handleClick}\n className={classNames(\n 'group w-full px-4 py-3 transition-colors text-left max-w-full overflow-hidden focus-ring',\n {\n 'bg-primary-alt/10 border-l-4 border-l-primary': isSelected,\n 'hover:bg-sand': !isSelected,\n }\n )}\n >\n <div className=\"flex items-start gap-3\">\n {/* Avatar */}\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={44}\n className=\"[&_.avatar-fallback]:group-hover:bg-[#eeeeee]\"\n />\n\n {/* Content column */}\n <div className=\"flex-1 min-w-0 flex flex-col gap-1\">\n {/* Name and timestamp row */}\n <div className=\"flex items-center justify-between gap-2\">\n <h3\n className={classNames(\n 'text-sm font-medium truncate',\n isSelected ? 'text-primary' : 'text-charcoal'\n )}\n >\n {participantName}\n </h3>\n {lastMessageTime && (\n <span className=\"text-xs text-stone flex-shrink-0\">\n {lastMessageTime}\n </span>\n )}\n </div>\n\n {/* Message and unread badge row */}\n <div className=\"flex items-center justify-between gap-2 min-w-0\">\n <p className=\"text-xs text-stone mr-2 flex-1 line-clamp-2\">\n {messagePreview}\n </p>\n {unreadCount > 0 && (\n <span className=\"bg-[#7f22fe] text-white text-xs px-2 py-0.5 rounded-full min-w-[20px] text-center flex-shrink-0\">\n {unreadCount > 99 ? '99+' : unreadCount}\n </span>\n )}\n </div>\n </div>\n </div>\n </button>\n )\n }\n)\n\nexport default CustomChannelPreview\nCustomChannelPreview.displayName = 'CustomChannelPreview'\n","import classNames from 'classnames'\nimport React from 'react'\nimport type { ChannelPreviewUIComponentProps } from 'stream-chat-react'\nimport { ChannelList as StreamChannelList } from 'stream-chat-react'\n\nimport { useMessagingContext } from '../../providers/MessagingProvider'\nimport type { ChannelListProps } from '../../types'\n\nimport CustomChannelPreview from './CustomChannelPreview'\n\n/**\n * Channel list component with customizable header and actions\n */\nexport const ChannelList = React.memo<ChannelListProps>(\n ({\n onChannelSelect,\n selectedChannel,\n filters,\n className,\n customEmptyStateIndicator,\n renderMessagePreview,\n }) => {\n // Track renders\n const renderCountRef = React.useRef(0)\n renderCountRef.current++\n\n // Get debug flag from context\n const { debug = false } = useMessagingContext()\n\n if (debug) {\n console.log('๐Ÿ“บ [ChannelList] ๐Ÿ”„ RENDER START', {\n renderCount: renderCountRef.current,\n selectedChannelId: selectedChannel?.id,\n filters,\n })\n }\n\n // Memoize Preview component to prevent re-renders\n const PreviewComponent = React.useMemo(() => {\n const Preview = (props: ChannelPreviewUIComponentProps) => (\n <CustomChannelPreview\n {...props}\n selectedChannel={selectedChannel}\n onChannelSelect={onChannelSelect}\n debug={debug}\n renderMessagePreview={renderMessagePreview}\n />\n )\n return Preview\n }, [selectedChannel, onChannelSelect, debug, renderMessagePreview])\n\n return (\n <div\n className={classNames(\n 'messaging-channel-list h-full flex flex-col min-w-0 overflow-hidden',\n className\n )}\n >\n {/* Channel List */}\n <div className=\"flex-1 overflow-hidden min-w-0\">\n <StreamChannelList\n key={JSON.stringify(filters)}\n filters={filters}\n sort={{ last_message_at: -1 }}\n options={{ limit: 30 }}\n Preview={PreviewComponent}\n EmptyStateIndicator={customEmptyStateIndicator}\n />\n </div>\n </div>\n )\n }\n)\nChannelList.displayName = 'ChannelList'\n","import classNames from \"classnames\";\nimport React from \"react\";\n\ninterface ActionButtonProps\n extends React.ButtonHTMLAttributes<HTMLButtonElement> {\n variant?: \"default\" | \"danger\";\n}\n\nconst ActionButton = ({\n variant = \"default\",\n className,\n children,\n ...rest\n}: ActionButtonProps) => {\n const isDanger = variant === \"danger\";\n return (\n <button\n type=\"button\"\n className={classNames(\n \"flex w-full items-center gap-3 rounded-lg px-4 py-3 text-left text-sm transition-colors focus-ring disabled:cursor-not-allowed disabled:opacity-60\",\n isDanger\n ? \"text-danger hover:bg-danger/50\"\n : \"text-charcoal hover:bg-sand\",\n className,\n )}\n {...rest}\n >\n {children}\n </button>\n );\n};\n\nexport default ActionButton;\n","import classNames from \"classnames\";\nimport React from \"react\";\n\ninterface IconButtonProps\n extends Omit<\n React.ButtonHTMLAttributes<HTMLButtonElement>,\n \"type\" | \"children\"\n > {\n label: string;\n children: React.ReactNode;\n className?: string;\n}\n\nexport function IconButton({ label, className, children, ...rest }: IconButtonProps) {\n return (\n <button\n type=\"button\"\n className={classNames(\n \"rounded-full p-2 transition-colors focus-ring\",\n {\n \"cursor-not-allowed opacity-50\": rest.disabled,\n \"hover:bg-sand\": !rest.disabled,\n },\n className,\n )}\n {...rest}\n >\n <span className=\"sr-only\">{label}</span>\n {children}\n </button>\n );\n}","import { XIcon } from '@phosphor-icons/react'\n\nimport { IconButton } from '../IconButton'\n\ninterface CloseButtonProps {\n onClick: () => void\n}\n\nexport function CloseButton({ onClick }: CloseButtonProps) {\n return (\n <IconButton label=\"Close\" onClick={onClick} className=\"p-1\">\n <XIcon className=\"h-5 w-5 text-stone\" weight=\"bold\" />\n </IconButton>\n )\n}\n","import { DateSeparator, type DateSeparatorProps } from 'stream-chat-react'\n\nexport const CustomDateSeparator = (props: DateSeparatorProps) => (\n <DateSeparator {...props} position=\"center\" />\n)\n","import classNames from 'classnames'\nimport React, { useMemo, useState } from 'react'\nimport {\n Attachment as DefaultAttachment,\n EditMessageModal as DefaultEditMessageModal,\n MessageBounceModal,\n MessageBouncePrompt as DefaultMessageBouncePrompt,\n MessageBlocked as DefaultMessageBlocked,\n MessageDeleted as DefaultMessageDeleted,\n MessageErrorIcon,\n MessageIsThreadReplyInChannelButtonIndicator as DefaultMessageIsThreadReplyInChannelButtonIndicator,\n MessageRepliesCountButton as DefaultMessageRepliesCountButton,\n MessageText,\n Poll,\n ReminderNotification as DefaultReminderNotification,\n StreamedMessageText as DefaultStreamedMessageText,\n areMessageUIPropsEqual,\n isDateSeparatorMessage,\n isMessageBlocked,\n isMessageBounced,\n messageHasAttachments,\n messageHasReactions,\n useComponentContext,\n useChatContext,\n useMessageContext,\n useMessageReminder,\n type MessageContextValue,\n type MessageUIComponentProps,\n} from 'stream-chat-react'\n\nimport { Avatar } from '../Avatar'\n\nimport { MessageTag, isTipOnlyMessage } from './MessageTag'\n\ntype CustomMessageWithContextProps = MessageContextValue\n\nconst CustomMessageWithContext = (props: CustomMessageWithContextProps) => {\n const {\n additionalMessageInputProps,\n editing,\n endOfGroup,\n firstOfGroup,\n groupedByUser,\n handleAction,\n handleOpenThread,\n handleRetry,\n highlighted,\n isMessageAIGenerated,\n isMyMessage,\n message,\n renderText,\n threadList,\n } = props\n\n const { client } = useChatContext('CustomMessage')\n const [isBounceDialogOpen, setIsBounceDialogOpen] = useState(false)\n const reminder = useMessageReminder(message.id)\n\n const {\n Attachment = DefaultAttachment,\n EditMessageModal = DefaultEditMessageModal,\n MessageBlocked = DefaultMessageBlocked,\n MessageBouncePrompt = DefaultMessageBouncePrompt,\n MessageDeleted = DefaultMessageDeleted,\n MessageIsThreadReplyInChannelButtonIndicator = DefaultMessageIsThreadReplyInChannelButtonIndicator,\n MessageRepliesCountButton = DefaultMessageRepliesCountButton,\n ReminderNotification = DefaultReminderNotification,\n StreamedMessageText = DefaultStreamedMessageText,\n PinIndicator,\n } = useComponentContext('CustomMessage')\n\n const hasAttachment = messageHasAttachments(message)\n const hasReactions = messageHasReactions(message)\n const isAIGenerated = useMemo(\n () => isMessageAIGenerated?.(message),\n [isMessageAIGenerated, message]\n )\n const finalAttachments = useMemo(\n () =>\n !message.shared_location && !message.attachments\n ? []\n : !message.shared_location\n ? message.attachments\n : [message.shared_location, ...(message.attachments ?? [])],\n [message]\n )\n\n if (isDateSeparatorMessage(message)) {\n return null\n }\n\n if (message.deleted_at || message.type === 'deleted') {\n return <MessageDeleted message={message} />\n }\n\n if (isMessageBlocked(message)) {\n return <MessageBlocked />\n }\n\n const showReplyCountButton = !threadList && !!message.reply_count\n const showIsReplyInChannel =\n !threadList && message.show_in_channel && message.parent_id\n const allowRetry =\n message.status === 'failed' && message.error?.status !== 403\n const isBounced = isMessageBounced(message)\n let handleClick: (() => void) | undefined = undefined\n\n if (allowRetry) {\n handleClick = () => handleRetry(message)\n } else if (isBounced) {\n handleClick = () => setIsBounceDialogOpen(true)\n }\n\n const rootClassName = classNames(\n 'str-chat__message str-chat__message-simple',\n `str-chat__message--${message.type}`,\n `str-chat__message--${message.status}`,\n isMyMessage()\n ? 'str-chat__message--me str-chat__message-simple--me'\n : 'str-chat__message--other',\n message.text ? 'str-chat__message--has-text' : 'has-no-text',\n {\n 'str-chat__message--has-attachment': hasAttachment,\n 'str-chat__message--highlighted': highlighted,\n 'str-chat__message--pinned pinned-message': message.pinned,\n 'str-chat__message--with-reactions': hasReactions,\n 'str-chat__message-send-can-be-retried':\n message?.status === 'failed' && message?.error?.status !== 403,\n 'str-chat__message-with-thread-link':\n showReplyCountButton || showIsReplyInChannel,\n 'str-chat__virtual-message__wrapper--end': endOfGroup,\n 'str-chat__virtual-message__wrapper--first': firstOfGroup,\n 'str-chat__virtual-message__wrapper--group': groupedByUser,\n }\n )\n\n const poll = message.poll_id && client.polls.fromState(message.poll_id)\n const isTipOnly = isTipOnlyMessage(message)\n\n return (\n <>\n {editing && (\n <EditMessageModal\n additionalMessageInputProps={additionalMessageInputProps}\n />\n )}\n {isBounceDialogOpen && (\n <MessageBounceModal\n MessageBouncePrompt={MessageBouncePrompt}\n onClose={() => setIsBounceDialogOpen(false)}\n open={isBounceDialogOpen}\n />\n )}\n <div className={rootClassName} key={message.id}>\n {PinIndicator && <PinIndicator />}\n {!!reminder && <ReminderNotification reminder={reminder} />}\n {message.user && (\n <Avatar\n className=\"str-chat__avatar str-chat__message-sender-avatar\"\n id={message.user.id}\n image={message.user.image}\n name={message.user.name || message.user.id}\n />\n )}\n {/* eslint-disable-next-line jsx-a11y/no-static-element-interactions */}\n <div\n className={classNames('str-chat__message-inner', {\n 'str-chat__simple-message--error-failed': allowRetry || isBounced,\n })}\n data-testid=\"message-inner\"\n onClick={handleClick}\n onKeyDown={handleClick}\n role={handleClick ? 'button' : undefined}\n tabIndex={handleClick ? 0 : undefined}\n style={{\n // Force margins to 0 to prevent hover layout shift\n // Stream Chat CSS sets margin-inline-end/start to 78px, then 0 on hover\n marginInlineEnd: 0,\n marginInlineStart: 0,\n }}\n >\n {isTipOnly ? (\n /* Tip-only messages render as a standalone bubble */\n <MessageTag message={message} standalone />\n ) : (\n <div className=\"str-chat__message-bubble-wrapper\">\n <div className=\"str-chat__message-bubble\">\n {poll && <Poll poll={poll} />}\n {finalAttachments?.length && !message.quoted_message ? (\n <Attachment\n actionHandler={handleAction}\n attachments={finalAttachments}\n />\n ) : null}\n {isAIGenerated ? (\n <StreamedMessageText\n message={message}\n renderText={renderText}\n />\n ) : (\n <MessageText message={message} renderText={renderText} />\n )}\n <MessageErrorIcon />\n </div>\n {/* Tag positioned outside and below the bubble */}\n <MessageTag message={message} />\n </div>\n )}\n </div>\n {showReplyCountButton && (\n <MessageRepliesCountButton\n onClick={handleOpenThread}\n reply_count={message.reply_count}\n />\n )}\n {showIsReplyInChannel && (\n <MessageIsThreadReplyInChannelButtonIndicator />\n )}\n </div>\n </>\n )\n}\n\nconst MemoizedCustomMessage = React.memo(\n CustomMessageWithContext,\n areMessageUIPropsEqual\n) as typeof CustomMessageWithContext\n\nexport const CustomMessage = (props: MessageUIComponentProps) => {\n const messageContext = useMessageContext('CustomMessage')\n return <MemoizedCustomMessage {...messageContext} {...props} />\n}\n","import { XIcon } from '@phosphor-icons/react'\nimport React from 'react'\nimport {\n LinkPreview,\n LinkPreviewsManager,\n LinkPreviewsManagerState,\n} from 'stream-chat'\nimport { useMessageComposer, useStateStore } from 'stream-chat-react'\n\nconst linkPreviewsManagerStateSelector = (state: LinkPreviewsManagerState) => ({\n linkPreviews: Array.from(state.previews.values()).filter(\n (preview) =>\n LinkPreviewsManager.previewIsLoaded(preview) ||\n LinkPreviewsManager.previewIsLoading(preview)\n ),\n})\n\ninterface CustomLinkPreviewCardProps {\n link: LinkPreview\n onDismiss: (url: string) => void\n}\n\nconst CustomLinkPreviewCard: React.FC<CustomLinkPreviewCardProps> = ({\n link,\n onDismiss,\n}) => {\n const { og_scrape_url, title, image_url } = link\n\n const handleDismissLink = (e: React.MouseEvent) => {\n e.preventDefault()\n onDismiss(og_scrape_url)\n }\n\n return (\n <a\n href={og_scrape_url}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className=\"relative w-full block rounded-[24px] bg-[#121110] p-2 no-underline transition-opacity hover:opacity-90\"\n >\n {image_url && (\n <img\n src={image_url}\n alt={title || ''}\n className=\"h-[148px] w-full rounded-[20px] object-cover\"\n />\n )}\n <button\n type=\"button\"\n onClick={handleDismissLink}\n className=\"absolute right-4 top-4 flex size-6 items-center justify-center rounded-full border border-white/40 bg-white/70 backdrop-blur-2xl\"\n aria-label=\"Close link preview\"\n >\n <XIcon className=\"size-4 text-black/90\" />\n </button>\n <div className=\"p-2\">\n {title && (\n <div className=\"text-[14px] font-medium leading-5 text-white\">\n {title}\n </div>\n )}\n <div className=\"text-[12px] leading-4 text-white/55\">\n {og_scrape_url}\n </div>\n </div>\n </a>\n )\n}\n\nexport const CustomLinkPreviewList = () => {\n const { linkPreviewsManager } = useMessageComposer()\n\n const { linkPreviews: stateLinkPreviews } = useStateStore(\n linkPreviewsManager.state,\n linkPreviewsManagerStateSelector\n )\n\n const handleDismiss = (url: string) => {\n linkPreviewsManager.dismissPreview(url)\n }\n\n const showLinkPreviews = stateLinkPreviews.length > 0\n\n if (!showLinkPreviews) return null\n\n return (\n <div className=\"flex flex-col items-center w-full gap-2 mb-4\">\n {stateLinkPreviews.map((linkPreview) => (\n <CustomLinkPreviewCard\n key={linkPreview.og_scrape_url}\n link={linkPreview}\n onDismiss={handleDismiss}\n />\n ))}\n </div>\n )\n}\n","import { ArrowUpIcon } from '@phosphor-icons/react'\nimport React from 'react'\nimport {\n AttachmentPreviewList,\n MessageInput,\n QuotedMessagePreview,\n SimpleAttachmentSelector,\n TextareaComposer,\n useMessageComposerHasSendableData,\n useMessageInputContext,\n} from 'stream-chat-react'\n\nimport { CustomLinkPreviewList } from '../CustomLinkPreviewList'\n\nconst CustomMessageInputInner: React.FC = () => {\n const { handleSubmit } = useMessageInputContext()\n const hasSendableData = useMessageComposerHasSendableData()\n\n return (\n <>\n <div className=\"left-container\">\n <SimpleAttachmentSelector />\n </div>\n <div className=\"central-container min-w-0 w-full p-2 bg-white rounded-[1.5rem] shadow-[0_4px_16px_0_rgba(0,0,0,0.08),0_1px_2px_0_rgba(0,0,0,0.04),0_0_0_1px_rgba(0,0,0,0.04)]\">\n <QuotedMessagePreview />\n <CustomLinkPreviewList />\n <AttachmentPreviewList />\n <div className=\"flex\">\n <div className=\"w-full ml-2 mr-4 self-center leading-[0]\">\n <TextareaComposer\n className=\"w-full resize-none outline-none leading-6\"\n // While this might usually be considered an anti-pattern, in most\n // cases, when a message thread is rendered, we want the input to\n // gain focus automatically.\n // eslint-disable-next-line jsx-a11y/no-autofocus\n autoFocus\n maxRows={4}\n />\n </div>\n <button\n aria-label=\"Send\"\n className=\"str-chat__send-button mt-auto flex justify-center items-center flex-shrink-0 rounded-full size-8 bg-[#121110] disabled:bg-[#F1F0EE] disabled:text-black/20 text-white\"\n data-testid=\"send-button\"\n disabled={!hasSendableData}\n onClick={handleSubmit}\n type=\"button\"\n >\n <ArrowUpIcon className=\"size-4\" />\n </button>\n </div>\n </div>\n </>\n )\n}\n\nexport interface CustomMessageInputProps {\n renderActions?: () => React.ReactNode\n}\n\nexport const CustomMessageInput: React.FC<CustomMessageInputProps> = ({\n renderActions,\n}) => (\n <div className=\"message-input flex items-center gap-2 p-4\">\n {renderActions && renderActions?.()}\n <MessageInput Input={CustomMessageInputInner} />\n </div>\n)\n","import { MessageTimestamp, type EventComponentProps } from 'stream-chat-react'\n\nexport const CustomSystemMessage: React.FC<EventComponentProps> = (props) => {\n const isDateHidden = props.message.hide_date === true\n\n return (\n <div className=\"str-chat__message--system\" data-testid=\"message-system\">\n <div className=\"str-chat__message--system__text\">\n <div className=\"str-chat__message--system__line\"></div>\n <p>{props.message.text}</p>\n <div className=\"str-chat__message--system__line\"></div>\n </div>\n {!isDateHidden && <MessageTimestamp message={props.message} />}\n </div>\n )\n}\n\n","import React from 'react'\n\n/**\n * Empty state component shown when a channel has no messages\n * Returns null to show nothing - the LoadingIndicator handles the loading state\n */\nexport const ChannelEmptyState: React.FC = () => null\n","import classNames from 'classnames'\n\ntype LoadingProps = {\n className?: string\n message?: string\n}\n\nconst Loading = ({ className, message }: LoadingProps) => (\n <div\n className={classNames('flex items-center justify-center h-full', className)}\n >\n <svg viewBox=\"0 0 100 100\" className=\"size-8 fill-pebble\" stroke=\"none\">\n <circle cx=\"6\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 15 ; 0 -15; 0 15\"\n repeatCount=\"indefinite\"\n begin=\"0.1\"\n />\n </circle>\n <circle cx=\"30\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 10 ; 0 -10; 0 10\"\n repeatCount=\"indefinite\"\n begin=\"0.2\"\n />\n </circle>\n <circle cx=\"54\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 5 ; 0 -5; 0 5\"\n repeatCount=\"indefinite\"\n begin=\"0.3\"\n />\n </circle>\n </svg>\n {message && <span className=\"text-stone\">{message}</span>}\n </div>\n)\n\nexport default Loading\n","import React from 'react'\n\nimport Loading from '../Loading'\n\n/**\n * Loading state component\n */\nexport const LoadingState = React.memo(() => (\n <div className=\"messaging-loading-state flex items-center justify-center h-full\">\n <div className=\"flex items-center\">\n <Loading className=\"w-6 h-6\" />\n <span className=\"text-sm text-stone\">Loading messages</span>\n </div>\n </div>\n))\nLoadingState.displayName = 'LoadingState'\n","import {\n ArrowLeftIcon,\n DotsThreeIcon,\n FlagIcon,\n ProhibitInsetIcon,\n SignOutIcon,\n SpinnerGapIcon,\n StarIcon,\n} from '@phosphor-icons/react'\nimport classNames from 'classnames'\nimport React, { useState, useCallback, useRef, useEffect } from 'react'\nimport {\n Channel as ChannelType,\n ChannelMemberResponse,\n Event,\n} from 'stream-chat'\nimport {\n Channel,\n Window,\n MessageList,\n useChannelStateContext,\n WithComponents,\n MessageUIComponentProps,\n} from 'stream-chat-react'\n\nimport { useMessagingContext } from '../providers/MessagingProvider'\nimport type { ChannelViewProps } from '../types'\n\nimport ActionButton from './ActionButton'\nimport { Avatar } from './Avatar'\nimport { CloseButton } from './CloseButton'\nimport { CustomDateSeparator } from './CustomDateSeparator'\nimport { CustomMessage } from './CustomMessage'\nimport { CustomMessageInput } from './CustomMessageInput'\nimport { CustomSystemMessage } from './CustomSystemMessage'\nimport { ChannelEmptyState } from './MessagingShell/ChannelEmptyState'\nimport { LoadingState } from './MessagingShell/LoadingState'\n\n// Custom user type with email and username\ntype CustomUser = {\n email?: string\n username?: string\n}\n\n// Blocked user from Stream Chat API\ntype BlockedUser = {\n blocked_user_id: string\n}\n\n/**\n * Custom channel header component\n */\nconst CustomChannelHeader: React.FC<{\n onBack?: () => void\n showBackButton: boolean\n onShowInfo: () => void\n canShowInfo: boolean\n showStarButton?: boolean\n}> = ({\n onBack,\n showBackButton,\n onShowInfo,\n canShowInfo,\n showStarButton = false,\n}) => {\n const { channel } = useChannelStateContext()\n\n // Get participant info (excluding current user)\n const participant = React.useMemo(() => {\n const members = Object.values(channel.state.members || {})\n return members.find(\n (member) => member.user?.id && member.user.id !== channel._client.userID\n )\n }, [channel._client.userID, channel.state.members])\n\n const participantName =\n participant?.user?.name || participant?.user?.id || 'Unknown member'\n const participantImage = participant?.user?.image\n\n const [isStarred, setIsStarred] = useState(\n !!channel.state.membership?.pinned_at\n )\n\n useEffect(() => {\n const handleMemberUpdate = (event: Event) => {\n setIsStarred(\n event?.member\n ? !!event.member.pinned_at\n : !!channel.state.membership?.pinned_at\n )\n }\n\n channel.on('member.updated', handleMemberUpdate)\n\n return () => {\n channel.off('member.updated', handleMemberUpdate)\n }\n }, [channel])\n\n const handleStarClick = async () => {\n try {\n if (isStarred) {\n await channel.unpin()\n } else {\n await channel.pin()\n }\n } catch (error) {\n console.error(\n '[CustomChannelHeader] Failed to update pinned status:',\n error\n )\n }\n }\n\n return (\n <div className=\"@container\">\n <div className=\"grid grid-cols-[1fr_auto_1fr] w-full items-center @lg:hidden\">\n <div className=\"flex items-center gap-2\">\n {showBackButton && (\n <button\n className={classNames(\n 'size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center'\n )}\n onClick={onBack || (() => {})}\n type=\"button\"\n aria-label=\"Back to conversations\"\n >\n <ArrowLeftIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n </div>\n <div className=\"flex flex-col gap-1 items-center\">\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={40}\n />\n <h1 className=\"text-xs font-medium text-black/90\">\n {participantName}\n </h1>\n </div>\n <div className=\"flex justify-end items-center gap-2\">\n {showStarButton && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={handleStarClick}\n type=\"button\"\n aria-label={\n isStarred ? 'Unstar conversation' : 'Star conversation'\n }\n >\n <StarIcon\n className=\"size-5 text-black/90\"\n weight={isStarred ? 'fill' : 'regular'}\n />\n </button>\n )}\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={onShowInfo}\n type=\"button\"\n aria-label=\"Show info\"\n >\n <DotsThreeIcon className=\"size-5 text-black/90\" />\n </button>\n </div>\n </div>\n <div className=\"hidden @lg:flex items-center justify-between gap-3 min-h-12\">\n <div className=\"flex items-center gap-4 min-w-0\">\n {showBackButton && onBack && (\n <button\n type=\"button\"\n onClick={onBack}\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n aria-label=\"Back to conversations\"\n >\n <ArrowLeftIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n {/* Avatar */}\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={40}\n />\n <div className=\"min-w-0\">\n <h1 className=\"font-medium text-black/90 truncate\">\n {participantName}\n </h1>\n </div>\n </div>\n <div className=\"flex items-center gap-2\">\n {showStarButton && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={handleStarClick}\n type=\"button\"\n aria-label={\n isStarred ? 'Unstar conversation' : 'Star conversation'\n }\n >\n <StarIcon\n className=\"size-5 text-black/90\"\n weight={isStarred ? 'fill' : 'regular'}\n />\n </button>\n )}\n {canShowInfo && onShowInfo && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={onShowInfo}\n type=\"button\"\n aria-label=\"Show info\"\n >\n <DotsThreeIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n </div>\n </div>\n </div>\n )\n}\n\n/**\n * Channel info dialog (matching original implementation)\n */\nconst ChannelInfoDialog: React.FC<{\n dialogRef: React.RefObject<HTMLDialogElement>\n onClose: () => void\n participant: ChannelMemberResponse | undefined\n channel: ChannelType\n followerStatusLabel?: string\n onLeaveConversation?: (channel: ChannelType) => void\n onBlockParticipant?: (participantId?: string) => void\n showDeleteConversation?: boolean\n onDeleteConversationClick?: () => void\n onBlockParticipantClick?: () => void\n onReportParticipantClick?: () => void\n}> = ({\n dialogRef,\n onClose,\n participant,\n channel,\n followerStatusLabel,\n onLeaveConversation,\n onBlockParticipant,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n}) => {\n const { service, debug } = useMessagingContext()\n const [isParticipantBlocked, setIsParticipantBlocked] = useState(false)\n const [isLeaving, setIsLeaving] = useState(false)\n const [isUpdatingBlockStatus, setIsUpdatingBlockStatus] = useState(false)\n\n // Check if participant is blocked when participant changes\n const checkIsParticipantBlocked = useCallback(async () => {\n if (!service || !participant?.user?.id) return\n\n try {\n const blockedUsers = await service.getBlockedUsers()\n const isBlocked = blockedUsers.some(\n (user: BlockedUser) => user.blocked_user_id === participant?.user?.id\n )\n setIsParticipantBlocked(isBlocked)\n } catch (error) {\n console.error(\n '[ChannelInfoDialog] Failed to check blocked status:',\n error\n )\n }\n }, [service, participant?.user?.id])\n\n useEffect(() => {\n checkIsParticipantBlocked()\n }, [checkIsParticipantBlocked])\n\n const handleLeaveConversation = async () => {\n if (isLeaving) return\n\n // Fire analytics callback before action\n onDeleteConversationClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Leave conversation', channel.cid)\n }\n setIsLeaving(true)\n\n try {\n const actingUserId = channel._client?.userID ?? null\n await channel.hide(actingUserId, false)\n\n if (onLeaveConversation) {\n await onLeaveConversation(channel)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to leave conversation', error)\n } finally {\n setIsLeaving(false)\n }\n }\n\n const handleBlockUser = async () => {\n if (isUpdatingBlockStatus || !service) return\n\n // Fire analytics callback before action\n onBlockParticipantClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Block member', participant?.user?.id)\n }\n setIsUpdatingBlockStatus(true)\n\n try {\n await service.blockUser(participant?.user?.id)\n\n if (onBlockParticipant) {\n await onBlockParticipant(participant?.user?.id)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to block member', error)\n } finally {\n setIsUpdatingBlockStatus(false)\n }\n }\n\n const handleUnblockUser = async () => {\n if (isUpdatingBlockStatus || !service) return\n\n // Fire analytics callback before action\n onBlockParticipantClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Unblock member', participant?.user?.id)\n }\n setIsUpdatingBlockStatus(true)\n\n try {\n await service.unBlockUser(participant?.user?.id)\n\n if (onBlockParticipant) {\n await onBlockParticipant(participant?.user?.id)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to unblock member', error)\n } finally {\n setIsUpdatingBlockStatus(false)\n }\n }\n\n const handleReportUser = () => {\n // Fire analytics callback before action\n onReportParticipantClick?.()\n\n onClose()\n window.open(\n 'https://linktr.ee/s/about/trust-center/report',\n '_blank',\n 'noopener,noreferrer'\n )\n }\n\n if (!participant) return null\n\n const participantName =\n participant.user?.name || participant.user?.id || 'Unknown member'\n const participantImage = participant.user?.image\n const participantEmail = (participant.user as CustomUser)?.email\n const participantUsername = (participant.user as CustomUser)?.username\n const participantSecondary = participantEmail\n ? participantEmail\n : participantUsername\n ? `linktr.ee/${participantUsername}`\n : undefined\n const participantId = participant.user?.id || 'unknown'\n\n return (\n // eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-noninteractive-element-interactions\n <dialog\n ref={dialogRef}\n className=\"mes-dialog group\"\n onClose={onClose}\n onClick={(e) => {\n if (e.target === dialogRef.current) {\n onClose()\n }\n }}\n >\n <div className=\"ml-auto flex h-full w-full flex-col bg-white shadow-none transition-shadow duration-200 group-open:shadow-max-elevation-light\">\n <div className=\"flex items-center justify-between border-b border-sand px-4 py-3\">\n <h2 className=\"text-base font-semibold text-charcoal\">Chat info</h2>\n <CloseButton onClick={onClose} />\n </div>\n\n <div className=\"flex-1 px-2 overflow-y-auto w-full\">\n <div\n className=\"flex flex-col items-center gap-3 self-stretch px-4 py-2 mt-6 rounded-lg border border-black/[0.04]\"\n style={{ backgroundColor: '#FBFAF9' }}\n >\n <div className=\"flex items-center gap-3 w-full\">\n <Avatar\n id={participantId}\n name={participantName}\n image={participantImage}\n size={88}\n shape=\"circle\"\n />\n <div className=\"flex flex-col min-w-0 flex-1\">\n <p className=\"truncate text-base font-semibold text-charcoal\">\n {participantName}\n </p>\n {participantSecondary && (\n <p className=\"truncate text-sm text-[#00000055]\">\n {participantSecondary}\n </p>\n )}\n {followerStatusLabel && (\n <span\n className=\"mt-1 rounded-full text-xs font-normal w-fit\"\n style={{\n padding: '4px 8px',\n backgroundColor:\n followerStatusLabel === 'Subscribed to you'\n ? '#DCFCE7'\n : '#F5F5F4',\n color:\n followerStatusLabel === 'Subscribed to you'\n ? '#008236'\n : '#78716C',\n lineHeight: '133.333%',\n letterSpacing: '0.21px',\n }}\n >\n {followerStatusLabel}\n </span>\n )}\n </div>\n </div>\n </div>\n\n <ul className=\"flex flex-col gap-2 mt-2\">\n {showDeleteConversation && (\n <li>\n <ActionButton\n onClick={handleLeaveConversation}\n disabled={isLeaving}\n aria-busy={isLeaving}\n >\n {isLeaving ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <SignOutIcon className=\"h-5 w-5\" />\n )}\n <span>Delete Conversation</span>\n </ActionButton>\n </li>\n )}\n <li>\n {isParticipantBlocked ? (\n <ActionButton\n onClick={handleUnblockUser}\n disabled={isUpdatingBlockStatus}\n aria-busy={isUpdatingBlockStatus}\n >\n {isUpdatingBlockStatus ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <ProhibitInsetIcon className=\"h-5 w-5\" />\n )}\n <span>Unblock</span>\n </ActionButton>\n ) : (\n <ActionButton\n onClick={handleBlockUser}\n disabled={isUpdatingBlockStatus}\n aria-busy={isUpdatingBlockStatus}\n >\n {isUpdatingBlockStatus ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <ProhibitInsetIcon className=\"h-5 w-5\" />\n )}\n <span>Block</span>\n </ActionButton>\n )}\n </li>\n <li>\n <ActionButton variant=\"danger\" onClick={handleReportUser}>\n <FlagIcon className=\"h-5 w-5\" />\n <span>Report</span>\n </ActionButton>\n </li>\n </ul>\n </div>\n </div>\n </dialog>\n )\n}\n\n/**\n * Inner component that has access to channel context\n */\nconst ChannelViewInner: React.FC<{\n onBack?: () => void\n showBackButton: boolean\n renderMessageInputActions?: (channel: ChannelType) => React.ReactNode\n onLeaveConversation?: (channel: ChannelType) => void\n onBlockParticipant?: (participantId?: string) => void\n CustomChannelEmptyState?: React.ComponentType\n showDeleteConversation?: boolean\n onDeleteConversationClick?: () => void\n onBlockParticipantClick?: () => void\n onReportParticipantClick?: () => void\n showStarButton?: boolean\n}> = ({\n onBack,\n showBackButton,\n renderMessageInputActions,\n onLeaveConversation,\n onBlockParticipant,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n showStarButton = false,\n}) => {\n const { channel } = useChannelStateContext()\n const infoDialogRef = useRef<HTMLDialogElement>(null)\n\n // Get participant info for info dialog\n const participant = React.useMemo(() => {\n const members = Object.values(channel.state.members || {})\n return members.find(\n (member) => member.user?.id && member.user.id !== channel._client.userID\n )\n }, [channel._client.userID, channel.state.members])\n\n // Get follower status label from channel data\n const followerStatusLabel = React.useMemo(() => {\n const channelExtraData = (channel.data ?? {}) as {\n followerStatus?: string\n isFollower?: boolean\n }\n\n // If explicit followerStatus is provided, use it\n if (channelExtraData.followerStatus) {\n return String(channelExtraData.followerStatus)\n }\n // If isFollower is explicitly defined, use it to determine status\n if (channelExtraData.isFollower !== undefined) {\n return channelExtraData.isFollower\n ? 'Subscribed to you'\n : 'Not subscribed'\n }\n // Otherwise, don't show any status\n return undefined\n }, [channel.data])\n\n const handleShowInfo = useCallback(() => {\n infoDialogRef.current?.showModal()\n }, [])\n\n const handleCloseInfo = useCallback(() => {\n infoDialogRef.current?.close()\n }, [])\n\n return (\n <>\n <WithComponents\n overrides={{\n Message: (props: MessageUIComponentProps) => (\n <CustomMessage {...props} />\n ),\n }}\n >\n <Window>\n {/* Custom Channel Header */}\n <div className=\"p-4\">\n <CustomChannelHeader\n onBack={onBack}\n showBackButton={showBackButton}\n onShowInfo={handleShowInfo}\n canShowInfo={Boolean(participant)}\n showStarButton={showStarButton}\n />\n </div>\n\n {/* Message List */}\n <div className=\"flex-1 overflow-hidden relative\">\n <MessageList\n hideDeletedMessages\n hideNewMessageSeparator={false}\n messageActions={undefined}\n />\n </div>\n\n {/* Message Input */}\n <CustomMessageInput\n renderActions={() => renderMessageInputActions?.(channel)}\n />\n </Window>\n </WithComponents>\n\n {/* Channel Info Dialog */}\n <ChannelInfoDialog\n dialogRef={infoDialogRef}\n onClose={handleCloseInfo}\n participant={participant}\n channel={channel}\n followerStatusLabel={followerStatusLabel}\n onLeaveConversation={onLeaveConversation}\n onBlockParticipant={onBlockParticipant}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n />\n </>\n )\n}\n\n/**\n * Channel view component with message list and input\n */\nexport const ChannelView = React.memo<ChannelViewProps>(\n ({\n channel,\n onBack,\n showBackButton = false,\n renderMessageInputActions,\n onLeaveConversation,\n onBlockParticipant,\n className,\n CustomChannelEmptyState = ChannelEmptyState,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n dmAgentEnabled,\n messageMetadata,\n onMessageSent,\n showStarButton = false,\n }) => {\n // Custom send message handler that:\n // 1. Applies messageMetadata if provided\n // 2. Adds skip_push and silent when DM agent is active\n // 3. Calls onMessageSent callback with full response\n // Read chatbot_paused inside callback to get current value at send time (not stale closure)\n const doSendMessageRequest = useCallback(\n async (\n _channel: ChannelType,\n message: Parameters<ChannelType['sendMessage']>[0],\n options?: Parameters<ChannelType['sendMessage']>[1]\n ) => {\n const agentPaused =\n (channel.data as { chatbot_paused?: boolean })?.chatbot_paused ===\n true\n const shouldSuppressNotifications = dmAgentEnabled && !agentPaused\n\n // Build final message with nested metadata\n const finalMessage = {\n ...message,\n ...(shouldSuppressNotifications && { silent: true }),\n ...(messageMetadata && {\n metadata: {\n ...(message.metadata ?? {}),\n ...messageMetadata,\n },\n }),\n }\n\n // Build final options\n const finalOptions = {\n ...options,\n ...(shouldSuppressNotifications && { skip_push: true }),\n }\n\n const response = await channel.sendMessage(finalMessage, finalOptions)\n\n // Fire callback with full response (includes message.id)\n onMessageSent?.(response)\n\n return response\n },\n [channel, dmAgentEnabled, messageMetadata, onMessageSent]\n )\n\n return (\n <div\n className={classNames(\n 'messaging-channel-view h-full flex flex-col',\n className\n )}\n >\n <Channel\n channel={channel}\n MessageSystem={CustomSystemMessage}\n EmptyStateIndicator={CustomChannelEmptyState}\n LoadingIndicator={LoadingState}\n DateSeparator={CustomDateSeparator}\n doSendMessageRequest={doSendMessageRequest}\n >\n <ChannelViewInner\n onBack={onBack}\n showBackButton={showBackButton}\n renderMessageInputActions={renderMessageInputActions}\n onLeaveConversation={onLeaveConversation}\n onBlockParticipant={onBlockParticipant}\n CustomChannelEmptyState={CustomChannelEmptyState}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n showStarButton={showStarButton}\n />\n </Channel>\n </div>\n )\n }\n)\nChannelView.displayName = 'ChannelView'\n","import { MagnifyingGlassIcon, XIcon } from '@phosphor-icons/react'\nimport { useRef } from 'react'\n\nimport { IconButton } from '../IconButton'\n\ninterface SearchInputProps {\n searchQuery: string\n setSearchQuery: (value: string) => void\n placeholder: string\n}\n\nexport function SearchInput({\n searchQuery,\n setSearchQuery,\n placeholder,\n}: SearchInputProps) {\n const searchInputRef = useRef<HTMLInputElement>(null)\n\n return (\n <div className=\"relative\">\n <MagnifyingGlassIcon\n className=\"pointer-events-none absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-stone\"\n weight=\"bold\"\n />\n\n <input\n ref={searchInputRef}\n type=\"text\"\n placeholder={placeholder}\n value={searchQuery}\n onChange={(e) => setSearchQuery(e.target.value)}\n className=\"w-full pl-10 pr-10 py-3 text-sm border border-sand rounded-xl focus:outline-none focus:ring-2 focus:ring-black focus:border-transparent\"\n />\n\n {searchQuery && (\n <IconButton\n label=\"Clear search\"\n onClick={() => {\n setSearchQuery('')\n searchInputRef.current?.focus()\n }}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 p-1 text-stone hover:text-charcoal\"\n >\n <XIcon className=\"h-4 w-4\" weight=\"bold\" />\n </IconButton>\n )}\n </div>\n )\n}\n","import { ChatCircleDotsIcon, SpinnerGapIcon } from '@phosphor-icons/react'\nimport classNames from 'classnames'\nimport React, { useCallback, useEffect, useState, useRef } from 'react'\n\nimport { useMessagingContext } from '../../providers/MessagingProvider'\nimport type { ParticipantPickerProps, Participant } from '../../types'\nimport { Avatar } from '../Avatar'\nimport { CloseButton } from '../CloseButton'\nimport { SearchInput } from '../SearchInput'\n\n/**\n * Generic participant picker component for starting conversations\n */\nexport const ParticipantPicker: React.FC<ParticipantPickerProps> = ({\n participantSource,\n onSelectParticipant,\n onClose,\n existingParticipantIds = new Set(),\n participantLabel = 'participants',\n searchPlaceholder = 'Search participants...',\n className,\n}) => {\n const { debug } = useMessagingContext()\n const [searchQuery, setSearchQuery] = useState('')\n const [participants, setParticipants] = useState<Participant[]>([])\n const [loading, setLoading] = useState(false)\n const [error, setError] = useState<string | null>(null)\n const [startingChatWithId, setStartingChatWithId] = useState<string | null>(\n null\n )\n\n // Track if we've already loaded participants to prevent repeated loading\n const loadedRef = useRef(false)\n\n // Load participants initially - wait for participantSource to finish loading first\n useEffect(() => {\n // Wait for the participantSource to finish loading before we try to load participants\n if (participantSource.loading) {\n if (debug) {\n console.log(\n '[ParticipantPicker] Waiting for participant source to finish loading...'\n )\n }\n return\n }\n\n if (loadedRef.current) return // Prevent multiple loads\n\n const loadInitialParticipants = async () => {\n if (debug) {\n console.log('[ParticipantPicker] Loading initial participants...')\n }\n setLoading(true)\n setError(null)\n\n try {\n const result = await participantSource.loadParticipants({\n search: '', // Load all participants initially\n limit: 100,\n })\n setParticipants(result.participants)\n loadedRef.current = true // Mark as loaded\n if (debug) {\n console.log(\n '[ParticipantPicker] Participants loaded successfully:',\n result.participants.length\n )\n }\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'Failed to load participants'\n setError(errorMessage)\n console.error('[ParticipantPicker] Failed to load participants:', err)\n // Don't mark as loaded on error, allow retry\n } finally {\n setLoading(false)\n }\n }\n\n loadInitialParticipants()\n }, [participantSource.loading, debug]) // Re-run when loading state changes\n\n // Filter participants by search query and existing participants\n const availableParticipants = participants\n .filter((participant) => !existingParticipantIds.has(participant.id))\n .filter((participant) => {\n if (!searchQuery) return true\n const searchLower = searchQuery.toLowerCase()\n return (\n participant.name.toLowerCase().includes(searchLower) ||\n participant.email?.toLowerCase().includes(searchLower) ||\n false\n )\n })\n\n const handleSelectParticipant = useCallback(\n async (participant: Participant) => {\n if (startingChatWithId) return // Prevent multiple clicks\n\n setStartingChatWithId(participant.id)\n try {\n await onSelectParticipant(participant)\n } catch (error) {\n console.error('[ParticipantPicker] Failed to start chat:', error)\n // Reset the loading state on error\n setStartingChatWithId(null)\n }\n // Note: Don't reset startingChatWithId on success because the dialog will close\n },\n [onSelectParticipant, startingChatWithId]\n )\n\n const handleKeyDown = (\n event: React.KeyboardEvent,\n participant: Participant\n ) => {\n if (event.key === 'Enter' || event.key === ' ') {\n event.preventDefault()\n handleSelectParticipant(participant)\n }\n }\n\n return (\n <div className={classNames('flex flex-col h-full', className)}>\n {/* Header */}\n <div className=\"px-4 py-4 border-b border-sand bg-chalk\">\n <div className=\"flex items-center justify-between mb-3\">\n <h2 className=\"text-lg font-semibold text-charcoal\">\n Start a new Conversation\n </h2>\n <CloseButton onClick={onClose} />\n </div>\n\n <p className=\"text-xs text-stone mb-3\">\n Select a {participantLabel.slice(0, -1)} to start messaging (\n {availableParticipants.length} available)\n {participantSource.totalCount !== undefined &&\n ` โ€ข ${participantSource.totalCount} ${participantLabel} total`}\n </p>\n\n <SearchInput\n searchQuery={searchQuery}\n setSearchQuery={setSearchQuery}\n placeholder={searchPlaceholder}\n />\n </div>\n\n {/* Error State */}\n {error && (\n <div className=\"p-4 text-sm text-danger bg-danger-alt\">\n Error loading {participantLabel}: {error}\n </div>\n )}\n\n {/* Participants List */}\n <div className=\"flex-1 overflow-auto\">\n {loading && availableParticipants.length === 0 ? (\n <div className=\"h-32 flex items-center justify-center\">\n <div className=\"flex items-center space-x-2\">\n <div className=\"w-4 h-4 animate-spin rounded-full border-2 border-primary border-t-transparent\"></div>\n <span className=\"text-sm text-stone\">\n Loading {participantLabel}...\n </span>\n </div>\n </div>\n ) : availableParticipants.length === 0 ? (\n <div className=\"p-6 text-center\">\n <div className=\"mx-auto mb-4 flex h-16 w-16 items-center justify-center rounded-full bg-sand\">\n <ChatCircleDotsIcon className=\"h-8 w-8 text-charcoal\" />\n </div>\n <h3 className=\"text-sm font-semibold text-charcoal mb-2\">\n {searchQuery\n ? `No ${participantLabel} found`\n : participants.length > 0\n ? `Already chatting with all ${participantLabel}`\n : `No ${participantLabel} yet`}\n </h3>\n <p className=\"text-xs text-stone\">\n {searchQuery\n ? 'Try a different search term'\n : participants.length > 0\n ? `You have existing conversations with all your ${participantLabel}`\n : `${participantLabel.charAt(0).toUpperCase() + participantLabel.slice(1)} will appear here`}\n </p>\n </div>\n ) : (\n <ul className=\"space-y-0\">\n {availableParticipants.map((participant) => {\n const displayName =\n participant.name || participant.email || participant.id\n const displaySecondary =\n participant.email && participant.name\n ? participant.email\n : participant.phone\n\n return (\n <li key={participant.id}>\n <button\n type=\"button\"\n onClick={() => handleSelectParticipant(participant)}\n onKeyDown={(e) => handleKeyDown(e, participant)}\n className=\"w-full px-4 py-3 hover:bg-sand transition-colors border-b border-sand text-left focus:outline-none focus:ring-2 focus:ring-black\"\n >\n <div className=\"flex items-center justify-between\">\n <div className=\"flex items-center space-x-3 flex-1 min-w-0\">\n {/* Avatar */}\n <Avatar\n id={participant.id}\n name={displayName}\n image={participant.image}\n size={40}\n />\n\n {/* Info */}\n <div className=\"flex-1 min-w-0\">\n <h4 className=\"text-sm font-medium text-charcoal truncate\">\n {displayName}\n </h4>\n {displaySecondary && (\n <p className=\"text-xs text-stone truncate\">\n {displaySecondary}\n </p>\n )}\n </div>\n </div>\n\n {/* Icon */}\n <div className=\"flex-shrink-0\">\n {startingChatWithId === participant.id ? (\n <SpinnerGapIcon className=\"h-5 w-5 text-primary animate-spin\" />\n ) : (\n <ChatCircleDotsIcon className=\"h-5 w-5 text-stone\" />\n )}\n </div>\n </div>\n </button>\n </li>\n )\n })}\n\n {/* Loading indicator */}\n {loading && (\n <li className=\"p-4 flex justify-center\">\n <div className=\"flex items-center space-x-2\">\n <div className=\"w-4 h-4 animate-spin rounded-full border-2 border-primary border-t-transparent\"></div>\n <span className=\"text-sm text-stone\">Loading more...</span>\n </div>\n </li>\n )}\n </ul>\n )}\n </div>\n </div>\n )\n}\n","import React from 'react'\n\nconst ChatBubblesIllustration = ({ className }: { className?: string }) => (\n <svg\n width=\"140\"\n height=\"120\"\n viewBox=\"44 -2 144 126\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n className={className}\n >\n <g clipPath=\"url(#clip0_empty_state)\">\n <path\n d=\"M123.68 82.1932C123.383 103.675 105.839 121 84.2417 121C77.4724 121 71.0986 119.297 65.5327 116.299L52.5873 119.687L53.8036 106.673C48.1776 99.8701 44.7994 91.1453 44.7994 81.6356C44.7994 59.8965 62.4554 42.2754 84.2374 42.2754C89.1328 42.2754 93.8175 43.1633 98.1413 44.789\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M84.2458 86.0364C82.2851 86.0364 80.6957 84.4501 80.6957 82.4933C80.6957 80.5365 82.2851 78.9502 84.2458 78.9502C86.2065 78.9502 87.7959 80.5365 87.7959 82.4933C87.7959 84.4501 86.2065 86.0364 84.2458 86.0364Z\"\n fill=\"white\"\n />\n <path\n d=\"M68.3044 86.0364C66.3437 86.0364 64.7543 84.4501 64.7543 82.4933C64.7543 80.5365 66.3437 78.9502 68.3044 78.9502C70.2651 78.9502 71.8545 80.5365 71.8545 82.4933C71.8545 84.4501 70.2651 86.0364 68.3044 86.0364Z\"\n fill=\"white\"\n />\n <path\n d=\"M100.183 86.0364C98.2226 86.0364 96.6332 84.4501 96.6332 82.4933C96.6332 80.5365 98.2226 78.9502 100.183 78.9502C102.144 78.9502 103.733 80.5365 103.733 82.4933C103.733 84.4501 102.144 86.0364 100.183 86.0364Z\"\n fill=\"white\"\n />\n <g filter=\"url(#filter0_empty_state)\">\n <path\n d=\"M171.522 68.7154C177.443 61.4539 181 52.1488 181 42C181 18.8027 162.421 0 139.5 0C116.579 0 98 18.8027 98 42C98 65.1973 116.579 84 139.5 84C146.622 84 153.328 82.1857 159.184 78.9829L172.801 82.5993L171.522 68.7154Z\"\n fill=\"white\"\n />\n <path\n d=\"M171.522 68.7154C177.443 61.4539 181 52.1488 181 42C181 18.8027 162.421 0 139.5 0C116.579 0 98 18.8027 98 42C98 65.1973 116.579 84 139.5 84C146.622 84 153.328 82.1857 159.184 78.9829L172.801 82.5993L171.522 68.7154Z\"\n stroke=\"#D7D4CE\"\n strokeWidth=\"2\"\n strokeMiterlimit=\"10\"\n />\n </g>\n <path\n d=\"M139.502 45.5431C137.541 45.5431 135.952 43.9568 135.952 42C135.952 40.0432 137.541 38.4569 139.502 38.4569C141.462 38.4569 143.052 40.0432 143.052 42C143.052 43.9568 141.462 45.5431 139.502 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M123.561 45.5431C121.601 45.5431 120.011 43.9568 120.011 42C120.011 40.0432 121.601 38.4569 123.561 38.4569C125.522 38.4569 127.111 40.0432 127.111 42C127.111 43.9568 125.522 45.5431 123.561 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M155.439 45.5431C153.478 45.5431 151.889 43.9568 151.889 42C151.889 40.0432 153.478 38.4569 155.439 38.4569C157.4 38.4569 158.989 40.0432 158.989 42C158.989 43.9568 157.4 45.5431 155.439 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n </g>\n <defs>\n <filter\n id=\"filter0_empty_state\"\n x=\"97\"\n y=\"-1\"\n width=\"89\"\n height=\"90\"\n filterUnits=\"userSpaceOnUse\"\n colorInterpolationFilters=\"sRGB\"\n >\n <feFlood floodOpacity=\"0\" result=\"BackgroundImageFix\" />\n <feColorMatrix\n in=\"SourceAlpha\"\n type=\"matrix\"\n values=\"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0\"\n result=\"hardAlpha\"\n />\n <feOffset dx=\"4\" dy=\"4\" />\n <feComposite in2=\"hardAlpha\" operator=\"out\" />\n <feColorMatrix\n type=\"matrix\"\n values=\"0 0 0 0 0.8428 0 0 0 0 0.830064 0 0 0 0 0.8095 0 0 0 1 0\"\n />\n <feBlend\n mode=\"normal\"\n in2=\"BackgroundImageFix\"\n result=\"effect1_dropShadow\"\n />\n <feBlend\n mode=\"normal\"\n in=\"SourceGraphic\"\n in2=\"effect1_dropShadow\"\n result=\"shape\"\n />\n </filter>\n <clipPath id=\"clip0_empty_state\">\n <rect width=\"233\" height=\"233\" fill=\"white\" />\n </clipPath>\n </defs>\n </svg>\n)\n\n/**\n * Empty state component shown when no channel is selected\n */\nexport const EmptyState = React.memo<{ hasChannels: boolean; channelsLoaded: boolean }>(\n ({ hasChannels, channelsLoaded }) => (\n <div className=\"messaging-empty-state flex items-center justify-center h-full p-8 text-balance\">\n <div className=\"flex flex-col items-center max-w-sm text-center\">\n <ChatBubblesIllustration />\n {channelsLoaded && !hasChannels && (\n <div className=\"mt-8\">\n <h2 className=\"font-medium text-black text-[18px] mb-2\">\n Your inbox is empty\n </h2>\n <p className=\"text-[#676B5F] text-sm mb-6\">\n Share with your followers to start receiving messages\n </p>\n </div>\n )}\n </div>\n </div>\n))\nEmptyState.displayName = 'EmptyState'\n","import React from 'react'\n\ntype ErrorStateProps = {\n message: string\n onBack?: () => void\n}\n\n/**\n * Error state component shown when something goes wrong\n */\nexport const ErrorState = React.memo<ErrorStateProps>(({ message, onBack }) => (\n <div className=\"messaging-error-state flex items-center justify-center h-full p-8\">\n <div className=\"text-center max-w-sm\">\n <div className=\"w-24 h-24 bg-danger-alt/20 rounded-full flex items-center justify-center mx-auto mb-6\">\n <span className=\"text-4xl\">โš ๏ธ</span>\n </div>\n\n <h2 className=\"font-semibold text-charcoal mb-2\">Oops!</h2>\n\n <p className=\"text-stone text-sm mb-6\">{message}</p>\n\n {onBack && (\n <button\n type=\"button\"\n onClick={onBack}\n className=\"inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-white bg-[#7f22fe] hover:bg-primary-alt rounded-lg focus:outline-none focus:ring-2 focus:ring-primary transition-colors\"\n >\n Go Back\n </button>\n )}\n </div>\n </div>\n))\nErrorState.displayName = 'ErrorState'\n","import classNames from 'classnames'\nimport React, { useState, useCallback, useRef, useEffect } from 'react'\nimport type { Channel } from 'stream-chat'\n\nimport { useMessaging } from '../../hooks/useMessaging'\nimport type { MessagingShellProps, Participant } from '../../types'\nimport { ChannelList } from '../ChannelList'\nimport { ChannelView } from '../ChannelView'\nimport { ParticipantPicker } from '../ParticipantPicker'\n\nimport { EmptyState } from './EmptyState'\nimport { ErrorState } from './ErrorState'\nimport { LoadingState } from './LoadingState'\n\n/**\n * Main messaging interface component that combines channel list and channel view\n */\nexport const MessagingShell: React.FC<MessagingShellProps> = ({\n capabilities = {},\n className,\n renderMessageInputActions,\n onChannelSelect,\n onParticipantSelect,\n initialParticipantFilter,\n initialParticipantData,\n CustomChannelEmptyState,\n showChannelList = true,\n filters,\n channelListCustomEmptyStateIndicator,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n dmAgentEnabled,\n messageMetadata,\n onMessageSent,\n showStarButton = false,\n renderMessagePreview,\n}) => {\n const {\n service,\n client,\n isConnected,\n isLoading,\n error,\n refreshConnection,\n debug,\n } = useMessaging()\n\n const [selectedChannel, setSelectedChannel] = useState<Channel | null>(null)\n const [hasChannels, setHasChannels] = useState(false)\n const [channelsLoaded, setChannelsLoaded] = useState(false)\n const [_showParticipantPicker, setShowParticipantPicker] = useState(false)\n const [existingParticipantIds, setExistingParticipantIds] = useState<\n Set<string>\n >(new Set())\n const [pickerKey, _setPickerKey] = useState(0) // Key to force remount of ParticipantPicker\n const [directConversationMode, setDirectConversationMode] = useState(false)\n const [directConversationError, setDirectConversationError] = useState<\n string | null\n >(null)\n\n const participantPickerRef = useRef<HTMLDialogElement>(null)\n\n const {\n participantSource,\n participantLabel = 'participants',\n showDeleteConversation = true,\n } = capabilities\n\n // Create default filters and merge with provided filters\n const channelFilters = React.useMemo(() => {\n const userId = client?.userID\n\n // Base filters that should always be present\n const baseFilters = {\n type: 'messaging',\n last_message_at: { $exists: true },\n ...(userId && {\n members: { $in: [userId] },\n hidden: false,\n }),\n }\n\n // Merge provided filters with base filters\n // Provided filters can override base filters if needed\n return {\n ...baseFilters,\n ...filters,\n }\n }, [filters, client?.userID])\n\n // Track if we've already synced channels to prevent repeated API calls\n const syncedRef = useRef<string | null>(null)\n\n // Function to sync channels (extracted for reuse)\n const syncChannels = useCallback(async () => {\n if (!client || !isConnected) return\n\n const userId = client.userID\n if (!userId) return\n\n try {\n if (debug) {\n console.log('[MessagingShell] Syncing channels for user:', userId)\n }\n\n const channels = await client.queryChannels(\n {\n type: 'messaging',\n members: { $in: [userId] },\n },\n {},\n { limit: 100 }\n )\n\n const memberIds = new Set<string>()\n channels.forEach((channel: Channel) => {\n const members = channel.state.members\n Object.values(members).forEach((member) => {\n const memberId = member.user?.id\n if (memberId && memberId !== userId) {\n memberIds.add(memberId)\n }\n })\n })\n\n // Only update if the set contents have changed to prevent re-renders\n setExistingParticipantIds((prev) => {\n if (\n prev.size === memberIds.size &&\n [...prev].every((id) => memberIds.has(id))\n ) {\n return prev\n }\n return memberIds\n })\n setHasChannels(channels.length > 0)\n setChannelsLoaded(true)\n syncedRef.current = userId // Mark as synced for this user\n\n if (debug) {\n console.log('[MessagingShell] Channels synced successfully:', {\n channelCount: channels.length,\n memberCount: memberIds.size,\n })\n }\n } catch (error) {\n console.error('[MessagingShell] Failed to sync channels:', error)\n // Don't mark as synced on error, allow retry\n }\n }, [client, isConnected, debug])\n\n // Sync existing channels to track which participants we can already message\n useEffect(() => {\n if (!client || !isConnected) return\n\n const userId = client.userID\n if (!userId) return\n\n // Prevent repeated sync for the same user\n if (syncedRef.current === userId) return\n\n syncChannels()\n }, [client, isConnected, syncChannels])\n\n // Load initial channel for direct conversation mode\n useEffect(() => {\n if (!initialParticipantFilter || !client || !isConnected) return\n\n const loadInitialChannel = async () => {\n const userId = client.userID\n if (!userId) return\n\n try {\n if (debug) {\n console.log(\n '[MessagingShell] Loading initial conversation with:',\n initialParticipantFilter\n )\n }\n\n const channels = await client.queryChannels(\n {\n type: 'messaging',\n members: { $eq: [userId, initialParticipantFilter] },\n },\n {},\n { limit: 1 }\n )\n\n if (channels.length > 0) {\n setSelectedChannel(channels[0])\n setDirectConversationMode(true)\n setDirectConversationError(null)\n\n // Notify parent component of channel selection\n if (onChannelSelect) {\n onChannelSelect(channels[0])\n }\n\n if (debug) {\n console.log(\n '[MessagingShell] Initial conversation loaded:',\n channels[0].id\n )\n }\n } else {\n // No channel found - try to create one if participant data is provided\n if (initialParticipantData && service) {\n if (debug) {\n console.log(\n '[MessagingShell] No conversation found, creating one for:',\n initialParticipantData\n )\n }\n\n try {\n // Use the existing service method to create the channel\n const channel = await service.startChannelWithParticipant({\n id: initialParticipantData.id,\n name: initialParticipantData.name,\n email: initialParticipantData.email,\n phone: initialParticipantData.phone,\n })\n\n setSelectedChannel(channel)\n setDirectConversationMode(true)\n setDirectConversationError(null)\n\n // Notify parent component of channel selection\n if (onChannelSelect) {\n onChannelSelect(channel)\n }\n\n if (debug) {\n console.log(\n '[MessagingShell] Channel created and loaded:',\n channel.id\n )\n }\n } catch (createErr) {\n console.error(\n '[MessagingShell] Failed to create conversation:',\n createErr\n )\n setDirectConversationError('Failed to create conversation')\n }\n } else {\n // No participant data provided, show error\n setDirectConversationError(\n 'No conversation found with this account'\n )\n\n if (debug) {\n console.log(\n '[MessagingShell] No conversation found for:',\n initialParticipantFilter\n )\n }\n }\n }\n } catch (err) {\n console.error(\n '[MessagingShell] Failed to load initial conversation:',\n err\n )\n setDirectConversationError('Failed to load conversation')\n }\n }\n\n loadInitialChannel()\n }, [\n initialParticipantFilter,\n initialParticipantData,\n client,\n isConnected,\n service,\n debug,\n onChannelSelect,\n ])\n\n const handleChannelSelect = useCallback(\n (channel: Channel) => {\n setSelectedChannel(channel)\n onChannelSelect?.(channel)\n },\n [onChannelSelect]\n )\n\n const handleBackToChannelList = useCallback(() => {\n // In direct conversation mode, don't allow going back to channel list\n // The parent component should handle navigation\n if (directConversationMode) return\n\n setSelectedChannel(null)\n }, [directConversationMode])\n\n const handleSelectParticipant = useCallback(\n async (participant: Participant) => {\n if (!service) return\n\n try {\n if (debug) {\n console.log(\n '[MessagingShell] Starting conversation with:',\n participant.id\n )\n }\n\n const channel = await service.startChannelWithParticipant({\n id: participant.id,\n name: participant.name,\n email: participant.email,\n phone: participant.phone,\n })\n\n // Show the channel\n try {\n await channel.show()\n } catch (error) {\n console.warn('[MessagingShell] Failed to unhide channel:', error)\n }\n\n setSelectedChannel(channel)\n setShowParticipantPicker(false)\n participantPickerRef.current?.close()\n\n onParticipantSelect?.(participant)\n } catch (error) {\n console.error('[MessagingShell] Failed to start conversation:', error)\n }\n },\n [service, onParticipantSelect, debug]\n )\n\n const handleCloseParticipantPicker = useCallback(() => {\n setShowParticipantPicker(false)\n participantPickerRef.current?.close()\n }, [])\n\n const handleDialogBackdropClick = useCallback(\n (e: React.MouseEvent<HTMLDialogElement>) => {\n if (e.target === participantPickerRef.current) {\n handleCloseParticipantPicker()\n }\n },\n [handleCloseParticipantPicker]\n )\n\n const handleLeaveConversation = useCallback(\n async (channel: Channel) => {\n if (debug) {\n console.log('[MessagingShell] Leaving conversation:', channel.id)\n }\n setSelectedChannel(null)\n setDirectConversationMode(false) // Exit direct conversation mode\n\n // Force re-sync to update the existing participants list\n syncedRef.current = null\n await syncChannels()\n },\n [syncChannels, debug]\n )\n\n const handleBlockParticipant = useCallback(\n async (participantId?: string) => {\n if (debug) {\n console.log('[MessagingShell] Blocking participant:', participantId)\n }\n setSelectedChannel(null)\n setDirectConversationMode(false) // Exit direct conversation mode\n\n // Force re-sync to update the existing participants list\n syncedRef.current = null\n await syncChannels()\n },\n [syncChannels, debug]\n )\n\n const isChannelSelected = Boolean(selectedChannel)\n\n // Show loading state\n if (isLoading) {\n return (\n <div className={classNames('h-full', className)}>\n <LoadingState />\n </div>\n )\n }\n\n // Show error state\n if (error) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState message={error} onBack={refreshConnection} />\n </div>\n )\n }\n\n // Show not connected state\n if (!isConnected || !client) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState\n message=\"Not connected to messaging service\"\n onBack={refreshConnection}\n />\n </div>\n )\n }\n\n // Show direct conversation error state\n if (directConversationError) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState message={directConversationError} />\n </div>\n )\n }\n\n return (\n <div\n className={classNames(\n 'messaging-shell h-full bg-background-primary overflow-hidden',\n className\n )}\n >\n <div className=\"flex h-full min-h-0\">\n {/* Channel List Sidebar */}\n <div\n className={classNames(\n 'messaging-channel-list-sidebar min-h-0 min-w-0 lg:flex lg:flex-col',\n {\n '!hidden': showChannelList === false || directConversationMode,\n // Hide on mobile when channel selected, show on desktop with consistent wide width\n 'hidden lg:flex lg:flex-1 lg:max-w-2xl':\n showChannelList !== false &&\n !directConversationMode &&\n isChannelSelected,\n // Show on mobile when no channel selected, use same wide width on desktop\n 'flex flex-col w-full lg:flex-1 lg:max-w-2xl':\n showChannelList !== false &&\n !directConversationMode &&\n !isChannelSelected,\n }\n )}\n >\n <ChannelList\n onChannelSelect={handleChannelSelect}\n selectedChannel={selectedChannel || undefined}\n filters={channelFilters}\n customEmptyStateIndicator={channelListCustomEmptyStateIndicator}\n renderMessagePreview={renderMessagePreview}\n />\n </div>\n\n {/* Channel View */}\n <div\n className={classNames(\n 'messaging-conversation-view flex-1 flex-col min-w-0 min-h-0',\n {\n // In direct conversation mode (or waiting for it), always show (full width)\n flex:\n directConversationMode ||\n isChannelSelected ||\n initialParticipantFilter,\n // Normal mode: hide on mobile when no channel selected\n 'hidden lg:flex':\n !directConversationMode &&\n !isChannelSelected &&\n !initialParticipantFilter,\n }\n )}\n >\n {selectedChannel ? (\n <div className=\"flex-1 min-h-0 flex flex-col\">\n <ChannelView\n channel={selectedChannel}\n key={selectedChannel.id}\n onBack={handleBackToChannelList}\n showBackButton={!directConversationMode}\n renderMessageInputActions={renderMessageInputActions}\n onLeaveConversation={handleLeaveConversation}\n onBlockParticipant={handleBlockParticipant}\n CustomChannelEmptyState={CustomChannelEmptyState}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n dmAgentEnabled={dmAgentEnabled}\n messageMetadata={messageMetadata}\n onMessageSent={onMessageSent}\n showStarButton={showStarButton}\n />\n </div>\n ) : initialParticipantFilter ? (\n // Show loading while creating/loading direct conversation channel\n <LoadingState />\n ) : (\n <EmptyState\n hasChannels={hasChannels}\n channelsLoaded={channelsLoaded}\n />\n )}\n </div>\n </div>\n\n {/* Participant Picker Dialog */}\n {participantSource && (\n // eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-noninteractive-element-interactions\n <dialog\n ref={participantPickerRef}\n className=\"mes-dialog\"\n onClick={handleDialogBackdropClick}\n onClose={handleCloseParticipantPicker}\n >\n <div className=\"h-full w-full bg-white shadow-max-elevation-light\">\n <ParticipantPicker\n key={pickerKey}\n participantSource={participantSource}\n onSelectParticipant={handleSelectParticipant}\n onClose={handleCloseParticipantPicker}\n existingParticipantIds={existingParticipantIds}\n participantLabel={participantLabel}\n searchPlaceholder={`Search ${participantLabel}...`}\n />\n </div>\n </dialog>\n )}\n </div>\n )\n}\n","import classNames from 'classnames'\nimport React from 'react'\nexport interface FaqListItemProps {\n question: string\n onClick: () => void\n loading?: boolean\n className?: string\n}\n\nexport const FaqListItem: React.FC<FaqListItemProps> = ({\n question,\n onClick,\n loading = false,\n className,\n}) => {\n return (\n <button\n type=\"button\"\n onClick={onClick}\n disabled={loading}\n style={{ backgroundColor: '#E6E5E3' }}\n className={classNames(\n 'w-full text-center p-4 rounded-xl text-charcoal font-medium transition-colors',\n {\n 'hover:brightness-95 active:brightness-90': !loading,\n 'opacity-50 cursor-not-allowed': loading,\n },\n className\n )}\n >\n {question}\n </button>\n )\n}\n","import React from 'react'\n\nimport { Avatar } from '../Avatar'\n\nimport { FaqListItem } from './FaqListItem'\n\nexport interface Faq {\n id: string\n question: string\n answer: string\n enabled: boolean\n order?: number | null\n}\n\nexport interface FaqListProps {\n faqs: Faq[]\n onFaqClick: (faqId: string) => void\n loadingFaqId?: string | null\n headerText?: string\n className?: string\n avatarImage?: string\n avatarName?: string\n}\n\nexport const FaqList: React.FC<FaqListProps> = ({\n faqs,\n onFaqClick,\n loadingFaqId,\n headerText,\n className,\n avatarImage,\n avatarName,\n}) => {\n const enabledFaqs = faqs\n .filter((faq) => faq.enabled)\n .sort((a, b) => (a.order ?? 0) - (b.order ?? 0))\n\n if (enabledFaqs.length === 0) {\n return null\n }\n\n return (\n <div className={className}>\n <div className=\"flex gap-3 items-end\">\n {/* Avatar at bottom-left, outside grey background */}\n {(avatarImage || avatarName) && (\n <div className=\"flex-none\">\n <Avatar\n id={avatarName || 'account'}\n name={avatarName || 'Account'}\n image={avatarImage}\n size={24}\n shape=\"circle\"\n />\n </div>\n )}\n\n {/* FAQs with grey background */}\n <div\n className=\"flex-1 flex flex-col gap-3 rounded-lg p-4\"\n style={{ backgroundColor: '#F1F0EE' }}\n >\n {headerText && (\n <p className=\"text-md text-charcoal mb-4\">{headerText}</p>\n )}\n {enabledFaqs.map((faq) => (\n <FaqListItem\n key={faq.id}\n question={faq.question}\n onClick={() => onFaqClick(faq.id)}\n loading={loadingFaqId === faq.id}\n />\n ))}\n </div>\n </div>\n </div>\n )\n}\n","import { useState, useEffect, useCallback } from 'react';\n\nimport type { ParticipantSource, Participant } from '../types';\n\n/**\n * Hook for managing participant loading with search and pagination\n */\nexport const useParticipants = (\n participantSource: ParticipantSource,\n options: {\n initialSearch?: string;\n pageSize?: number;\n } = {}\n) => {\n const { initialSearch = '', pageSize = 20 } = options;\n \n const [participants, setParticipants] = useState<Participant[]>([]);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n const [searchQuery, setSearchQuery] = useState(initialSearch);\n const [hasMore, setHasMore] = useState(true);\n const [cursor, setCursor] = useState<string | undefined>();\n\n // Load participants with current search query\n const loadParticipants = useCallback(async (\n reset = false,\n customSearch?: string\n ) => {\n if (loading) return;\n \n const search = customSearch !== undefined ? customSearch : searchQuery;\n \n setLoading(true);\n setError(null);\n\n try {\n const result = await participantSource.loadParticipants({\n search: search || undefined,\n limit: pageSize,\n cursor: reset ? undefined : cursor,\n });\n\n setParticipants(prev => \n reset ? result.participants : [...prev, ...result.participants]\n );\n setHasMore(result.hasMore);\n setCursor(result.nextCursor);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Failed to load participants';\n setError(errorMessage);\n console.error('[useParticipants] Load error:', err);\n } finally {\n setLoading(false);\n }\n }, [participantSource, searchQuery, cursor, pageSize, loading]);\n\n // Load more participants (pagination)\n const loadMore = useCallback(() => {\n if (hasMore && !loading) {\n loadParticipants(false);\n }\n }, [hasMore, loading, loadParticipants]);\n\n // Search participants\n const search = useCallback((query: string) => {\n setSearchQuery(query);\n setCursor(undefined);\n loadParticipants(true, query);\n }, [loadParticipants]);\n\n // Refresh participants\n const refresh = useCallback(() => {\n setCursor(undefined);\n loadParticipants(true);\n }, [loadParticipants]);\n\n // Initial load - only run once when participantSource changes\n useEffect(() => {\n loadParticipants(true);\n }, [participantSource.loadParticipants]); // Only depend on the function to avoid loops\n\n return {\n participants,\n loading,\n error,\n searchQuery,\n hasMore,\n totalCount: participantSource.totalCount,\n loadMore,\n search,\n refresh,\n };\n};\n"],"names":["MessagingContext","createContext","useMessagingContext","useContext","MessagingProvider","children","user","serviceConfig","apiKey","capabilities","debug","debugLog","useCallback","message","args","service","setService","useState","client","setClient","isConnected","setIsConnected","isLoading","setIsLoading","error","setError","connectingRef","useRef","prevPropsRef","renderCountRef","useEffect","currentRender","newService","StreamChatService","connectedUserRef","_a","_b","streamClient","err","errorMessage","refreshConnection","contextValue","React","jsx","Chat","useMessaging","getDaysDifference","date1","date2","d1","diffTime","formatRelativeTime","date","now","daysDiff","EMOJIS","hashString","str","hash","i","char","getAvatarEmoji","id","index","Avatar","image","size","className","shape","emoji","fontSizeClass","borderStyle","classNames","SparkleIcon","isTipMessage","isPaidMessage","isChatbotMessage","isTipOrPaidMessage","isTipOnlyMessage","MessageTag","standalone","isTipOrPaid","isChatbot","amountText","label","jsxs","GiftIcon","CustomChannelPreview","channel","selectedChannel","onChannelSelect","unread","renderMessagePreview","isSelected","handleClick","participant","member","participantName","participantImage","_c","lastMessage","_e","_d","lastMessageText","attachment","lastMessageTime","isLastMessageFromChatbot","messagePreview","unreadCount","_f","ChannelList","filters","customEmptyStateIndicator","PreviewComponent","props","StreamChannelList","ActionButton","variant","rest","IconButton","CloseButton","onClick","XIcon","CustomDateSeparator","DateSeparator","CustomMessageWithContext","additionalMessageInputProps","editing","endOfGroup","firstOfGroup","groupedByUser","handleAction","handleOpenThread","handleRetry","highlighted","isMessageAIGenerated","isMyMessage","renderText","threadList","useChatContext","isBounceDialogOpen","setIsBounceDialogOpen","reminder","useMessageReminder","Attachment","DefaultAttachment","EditMessageModal","DefaultEditMessageModal","MessageBlocked","DefaultMessageBlocked","MessageBouncePrompt","DefaultMessageBouncePrompt","MessageDeleted","DefaultMessageDeleted","MessageIsThreadReplyInChannelButtonIndicator","DefaultMessageIsThreadReplyInChannelButtonIndicator","MessageRepliesCountButton","DefaultMessageRepliesCountButton","ReminderNotification","DefaultReminderNotification","StreamedMessageText","DefaultStreamedMessageText","PinIndicator","useComponentContext","hasAttachment","messageHasAttachments","hasReactions","messageHasReactions","isAIGenerated","useMemo","finalAttachments","isDateSeparatorMessage","isMessageBlocked","showReplyCountButton","showIsReplyInChannel","allowRetry","isBounced","isMessageBounced","rootClassName","poll","isTipOnly","Fragment","MessageBounceModal","Poll","MessageText","MessageErrorIcon","MemoizedCustomMessage","areMessageUIPropsEqual","CustomMessage","messageContext","useMessageContext","linkPreviewsManagerStateSelector","state","preview","LinkPreviewsManager","CustomLinkPreviewCard","link","onDismiss","og_scrape_url","title","image_url","e","CustomLinkPreviewList","linkPreviewsManager","useMessageComposer","stateLinkPreviews","useStateStore","handleDismiss","url","linkPreview","CustomMessageInputInner","handleSubmit","useMessageInputContext","hasSendableData","useMessageComposerHasSendableData","SimpleAttachmentSelector","QuotedMessagePreview","AttachmentPreviewList","TextareaComposer","ArrowUpIcon","CustomMessageInput","renderActions","MessageInput","CustomSystemMessage","isDateHidden","MessageTimestamp","ChannelEmptyState","Loading","LoadingState","CustomChannelHeader","onBack","showBackButton","onShowInfo","canShowInfo","showStarButton","useChannelStateContext","isStarred","setIsStarred","handleMemberUpdate","event","handleStarClick","ArrowLeftIcon","StarIcon","DotsThreeIcon","ChannelInfoDialog","dialogRef","onClose","followerStatusLabel","onLeaveConversation","onBlockParticipant","showDeleteConversation","onDeleteConversationClick","onBlockParticipantClick","onReportParticipantClick","isParticipantBlocked","setIsParticipantBlocked","isLeaving","setIsLeaving","isUpdatingBlockStatus","setIsUpdatingBlockStatus","checkIsParticipantBlocked","isBlocked","handleLeaveConversation","actingUserId","handleBlockUser","handleUnblockUser","handleReportUser","participantEmail","participantUsername","participantSecondary","participantId","_g","SpinnerGapIcon","SignOutIcon","ProhibitInsetIcon","FlagIcon","ChannelViewInner","renderMessageInputActions","infoDialogRef","channelExtraData","handleShowInfo","handleCloseInfo","WithComponents","Window","MessageList","ChannelView","CustomChannelEmptyState","dmAgentEnabled","messageMetadata","onMessageSent","doSendMessageRequest","_channel","options","agentPaused","shouldSuppressNotifications","finalMessage","finalOptions","response","Channel","SearchInput","searchQuery","setSearchQuery","placeholder","searchInputRef","MagnifyingGlassIcon","ParticipantPicker","participantSource","onSelectParticipant","existingParticipantIds","participantLabel","searchPlaceholder","participants","setParticipants","loading","setLoading","startingChatWithId","setStartingChatWithId","loadedRef","result","availableParticipants","searchLower","handleSelectParticipant","handleKeyDown","ChatCircleDotsIcon","displayName","displaySecondary","ChatBubblesIllustration","EmptyState","hasChannels","channelsLoaded","ErrorState","MessagingShell","onParticipantSelect","initialParticipantFilter","initialParticipantData","showChannelList","channelListCustomEmptyStateIndicator","setSelectedChannel","setHasChannels","setChannelsLoaded","_showParticipantPicker","setShowParticipantPicker","setExistingParticipantIds","pickerKey","_setPickerKey","directConversationMode","setDirectConversationMode","directConversationError","setDirectConversationError","participantPickerRef","channelFilters","userId","syncedRef","syncChannels","channels","memberIds","members","memberId","prev","createErr","handleChannelSelect","handleBackToChannelList","handleCloseParticipantPicker","handleDialogBackdropClick","handleBlockParticipant","isChannelSelected","FaqListItem","question","FaqList","faqs","onFaqClick","loadingFaqId","headerText","avatarImage","avatarName","enabledFaqs","faq","a","b","useParticipants","initialSearch","pageSize","hasMore","setHasMore","cursor","setCursor","loadParticipants","reset","customSearch","search","loadMore","query","refresh"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/providers/MessagingProvider.tsx","../src/hooks/useMessaging.ts","../src/utils/formatRelativeTime.ts","../src/components/Avatar/getAvatarEmoji.ts","../src/components/Avatar/index.tsx","../src/components/CustomMessage/MessageTag.tsx","../src/components/ChannelList/CustomChannelPreview.tsx","../src/components/ChannelList/index.tsx","../src/components/ActionButton/index.tsx","../src/components/IconButton/index.tsx","../src/components/CloseButton/index.tsx","../src/components/CustomDateSeparator/index.tsx","../src/components/CustomMessage/index.tsx","../src/components/CustomLinkPreviewList/index.tsx","../src/components/CustomMessageInput/index.tsx","../src/components/CustomSystemMessage/index.tsx","../src/components/MessagingShell/ChannelEmptyState.tsx","../src/components/Loading/index.tsx","../src/components/MessagingShell/LoadingState.tsx","../src/components/ChannelView.tsx","../src/components/SearchInput/index.tsx","../src/components/ParticipantPicker/index.tsx","../src/components/MessagingShell/EmptyState.tsx","../src/components/MessagingShell/ErrorState.tsx","../src/components/MessagingShell/index.tsx","../src/components/FaqList/FaqListItem.tsx","../src/components/FaqList/index.tsx","../src/hooks/useParticipants.ts"],"sourcesContent":["import { StreamChatService } from '@linktr.ee/messaging-core'\nimport React, {\n createContext,\n useContext,\n useEffect,\n useState,\n useRef,\n useCallback,\n} from 'react'\nimport type { StreamChat } from 'stream-chat'\nimport { Chat } from 'stream-chat-react'\n\nimport type { MessagingProviderProps, MessagingCapabilities } from '../types'\n\n/**\n * Context value for messaging state and service\n */\nexport interface MessagingContextValue {\n service: StreamChatService | null\n client: StreamChat | null // Stream Chat client\n isConnected: boolean\n isLoading: boolean\n error: string | null\n capabilities: MessagingCapabilities\n refreshConnection: () => Promise<void>\n debug: boolean\n}\n\nconst MessagingContext = createContext<MessagingContextValue>({\n service: null,\n client: null,\n isConnected: false,\n isLoading: false,\n error: null,\n capabilities: {},\n refreshConnection: async () => {},\n debug: false,\n})\n\n/**\n * Hook to access messaging context\n */\nexport const useMessagingContext = () => useContext(MessagingContext)\n\n/**\n * Provider component that wraps messaging-core with React state management\n */\nexport const MessagingProvider: React.FC<MessagingProviderProps> = ({\n children,\n user,\n serviceConfig,\n apiKey,\n capabilities = {},\n debug = false,\n}) => {\n // Create debug logger that respects the debug prop\n const debugLog = useCallback(\n (message: string, ...args: unknown[]) => {\n if (debug) {\n console.log(`๐Ÿ”ฅ [MessagingProvider] ${message}`, ...args)\n }\n },\n [debug]\n )\n\n debugLog('๐Ÿ”„ RENDER START', {\n userId: user?.id,\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfig: !!serviceConfig,\n capabilities: Object.keys(capabilities),\n })\n\n const [service, setService] = useState<StreamChatService | null>(null)\n const [client, setClient] = useState<StreamChat | null>(null)\n const [isConnected, setIsConnected] = useState(false)\n const [isLoading, setIsLoading] = useState(false)\n const [error, setError] = useState<string | null>(null)\n\n // Prevent multiple concurrent connection attempts\n const connectingRef = useRef(false)\n\n // Track renders and prop changes\n const prevPropsRef = useRef({\n userId: user?.id,\n apiKey,\n serviceConfig,\n capabilities,\n })\n const renderCountRef = useRef(0)\n renderCountRef.current++\n\n debugLog('๐Ÿ“Š RENDER INFO', {\n renderCount: renderCountRef.current,\n currentProps: { userId: user?.id, apiKey: apiKey?.substring(0, 8) + '...' },\n propChanges: {\n userChanged: prevPropsRef.current.userId !== user?.id,\n apiKeyChanged: prevPropsRef.current.apiKey !== apiKey,\n serviceConfigChanged:\n prevPropsRef.current.serviceConfig !== serviceConfig,\n capabilitiesChanged: prevPropsRef.current.capabilities !== capabilities,\n },\n })\n\n prevPropsRef.current = {\n userId: user?.id,\n apiKey,\n serviceConfig,\n capabilities,\n }\n\n // Initialize service when config changes\n useEffect(() => {\n const currentRender = renderCountRef.current\n debugLog('๐Ÿ”ง SERVICE INIT EFFECT TRIGGERED', {\n renderCount: currentRender,\n apiKey: !!apiKey,\n serviceConfig: !!serviceConfig,\n dependencies: {\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfigRef: serviceConfig,\n serviceConfigStable:\n prevPropsRef.current.serviceConfig === serviceConfig,\n apiKeyStable: prevPropsRef.current.apiKey === apiKey,\n },\n })\n\n if (!apiKey || !serviceConfig) {\n debugLog('โš ๏ธ SERVICE INIT SKIPPED', {\n renderCount: currentRender,\n reason: 'Missing apiKey or serviceConfig',\n })\n return\n }\n\n debugLog('๐Ÿš€ CREATING NEW SERVICE', {\n renderCount: currentRender,\n apiKey: apiKey?.substring(0, 8) + '...',\n serviceConfigChanged:\n prevPropsRef.current.serviceConfig !== serviceConfig,\n })\n\n const newService = new StreamChatService({\n ...serviceConfig,\n apiKey,\n debug,\n })\n\n setService(newService)\n debugLog('โœ… SERVICE SET', {\n renderCount: currentRender,\n serviceInstance: !!newService,\n })\n\n return () => {\n debugLog('๐Ÿงน SERVICE CLEANUP', {\n renderCount: currentRender,\n reason: 'Effect cleanup',\n })\n newService.disconnectUser().catch(console.error)\n }\n }, [apiKey, serviceConfig, debug, debugLog]) // Use serviceConfig object directly, not individual properties\n\n // Track if we've already connected this user with this service to prevent duplicate connections\n const connectedUserRef = useRef<{\n serviceId: StreamChatService\n userId: string\n } | null>(null)\n\n // Connect user when service and user are available\n useEffect(() => {\n debugLog('๐Ÿ”— USER CONNECTION EFFECT TRIGGERED', {\n hasService: !!service,\n hasUser: !!user,\n userId: user?.id,\n isConnecting: connectingRef.current,\n isConnected: isConnected,\n dependencies: { service: !!service, userId: user?.id },\n })\n\n if (!service || !user) {\n debugLog('โš ๏ธ USER CONNECTION SKIPPED', 'Missing service or user')\n return\n }\n\n if (connectingRef.current) {\n debugLog('โš ๏ธ USER CONNECTION SKIPPED', 'Already connecting')\n return\n }\n\n // Check if we've already connected this exact user with this exact service instance\n if (\n connectedUserRef.current?.serviceId === service &&\n connectedUserRef.current?.userId === user.id\n ) {\n debugLog(\n 'โš ๏ธ USER CONNECTION SKIPPED',\n 'Already connected this user with this service'\n )\n return\n }\n\n const connectUser = async () => {\n debugLog('๐Ÿš€ STARTING USER CONNECTION', { userId: user.id })\n connectingRef.current = true\n setIsLoading(true)\n setError(null)\n\n try {\n debugLog('๐Ÿ“ž CALLING SERVICE.CONNECTUSER', { userId: user.id })\n const streamClient = await service.connectUser(user)\n setClient(streamClient)\n setIsConnected(true)\n connectedUserRef.current = { serviceId: service, userId: user.id } // Mark as connected\n debugLog('โœ… USER CONNECTION SUCCESS', {\n userId: user.id,\n clientId: streamClient.userID,\n })\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'Connection failed'\n setError(errorMessage)\n debugLog('โŒ USER CONNECTION ERROR', {\n userId: user.id,\n error: errorMessage,\n })\n } finally {\n setIsLoading(false)\n connectingRef.current = false\n debugLog('๐Ÿ”„ USER CONNECTION FINISHED', {\n userId: user.id,\n isConnected,\n })\n }\n }\n\n connectUser()\n }, [service, user, debugLog, isConnected]) // Remove isConnected to prevent circular dependency\n\n // Disconnect when user is removed (cleanup effect)\n useEffect(() => {\n debugLog('๐Ÿ”Œ CLEANUP EFFECT REGISTERED', {\n hasService: !!service,\n isConnected,\n })\n return () => {\n if (service && isConnected) {\n debugLog(\n '๐Ÿงน CLEANUP EFFECT TRIGGERED',\n 'Cleaning up connection on unmount'\n )\n connectedUserRef.current = null // Reset connection tracking\n service.disconnectUser().catch(console.error)\n } else {\n debugLog('๐Ÿ”‡ CLEANUP EFFECT SKIPPED', {\n hasService: !!service,\n isConnected,\n })\n }\n }\n }, [service, isConnected, debugLog])\n\n const refreshConnection = useCallback(async () => {\n debugLog('๐Ÿ”„ REFRESH CONNECTION CALLED', {\n hasService: !!service,\n hasUser: !!user,\n })\n\n if (!service || !user) {\n debugLog('โš ๏ธ REFRESH CONNECTION SKIPPED', 'Missing service or user')\n return\n }\n\n debugLog('๐Ÿš€ STARTING CONNECTION REFRESH', { userId: user.id })\n setIsLoading(true)\n try {\n debugLog('๐Ÿ”Œ DISCONNECTING FOR REFRESH')\n await service.disconnectUser()\n debugLog('๐Ÿ“ž RECONNECTING FOR REFRESH')\n const streamClient = await service.connectUser(user)\n setClient(streamClient)\n setIsConnected(true)\n setError(null)\n debugLog('โœ… CONNECTION REFRESH SUCCESS', { userId: user.id })\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Refresh failed'\n setError(errorMessage)\n debugLog('โŒ CONNECTION REFRESH ERROR', {\n userId: user.id,\n error: errorMessage,\n })\n } finally {\n setIsLoading(false)\n debugLog('๐Ÿ”„ CONNECTION REFRESH FINISHED', { userId: user.id })\n }\n }, [service, user, debugLog])\n\n // Memoize context value to prevent unnecessary re-renders\n const contextValue: MessagingContextValue = React.useMemo(() => {\n debugLog('๐Ÿ’ซ CONTEXT VALUE MEMOIZATION', {\n hasService: !!service,\n hasClient: !!client,\n isConnected,\n isLoading,\n hasError: !!error,\n capabilitiesKeys: Object.keys(capabilities),\n })\n\n return {\n service,\n client,\n isConnected,\n isLoading,\n error,\n capabilities,\n refreshConnection,\n debug,\n }\n }, [\n service,\n client,\n isConnected,\n isLoading,\n error,\n capabilities,\n refreshConnection,\n debug,\n debugLog,\n ])\n\n debugLog('๐Ÿ”„ RENDER END', {\n renderCount: renderCountRef.current,\n willRenderChat: !!(client && isConnected),\n contextValueReady: !!contextValue,\n })\n\n return (\n <MessagingContext.Provider value={contextValue}>\n {client && isConnected ? (\n <Chat\n client={client}\n customClasses={{\n channelList:\n 'str-chat__channel-list str-chat__channel-list-react bg-transparent lg:border-r-2 border-r-0 border-[#0000000A]',\n }}\n >\n {children}\n </Chat>\n ) : (\n children\n )}\n </MessagingContext.Provider>\n )\n}\n","import { useMessagingContext } from '../providers/MessagingProvider';\nimport type { MessagingContextValue } from '../providers/MessagingProvider';\n\n/**\n * Hook to access messaging service and state\n */\nexport const useMessaging = (): MessagingContextValue => {\n return useMessagingContext();\n};\n","/**\n * Get the number of days between two dates (calendar days in UTC, not 24-hour periods)\n * Uses UTC to ensure consistent day calculations globally since messages are stored in UTC\n */\nconst getDaysDifference = (date1: Date, date2: Date): number => {\n const d1 = new Date(\n Date.UTC(date1.getUTCFullYear(), date1.getUTCMonth(), date1.getUTCDate())\n )\n const d2 = new Date(\n Date.UTC(date2.getUTCFullYear(), date2.getUTCMonth(), date2.getUTCDate())\n )\n const diffTime = d2.getTime() - d1.getTime()\n return Math.floor(diffTime / (1000 * 60 * 60 * 24))\n}\n\n/**\n * Format a date - shows time for today, relative time for older messages\n * (e.g., \"Just now\", \"2:08 PM\" for today, \"Yesterday\" for yesterday, \"2d\" for 2 days ago)\n */\nexport const formatRelativeTime = (date: Date): string => {\n const now = new Date()\n const diffInSeconds = Math.floor((now.getTime() - date.getTime()) / 1000)\n\n // Less than 1 minute\n if (diffInSeconds < 60) {\n return 'Just now'\n }\n\n // Check if it's today (same calendar day)\n const daysDiff = getDaysDifference(date, now)\n\n // If today, show time in 12-hour format (e.g., \"8:40 PM\")\n if (daysDiff === 0) {\n return date.toLocaleTimeString([], {\n hour: 'numeric',\n minute: '2-digit',\n hour12: true,\n })\n }\n\n // Yesterday\n if (daysDiff === 1) {\n return 'Yesterday'\n }\n\n // Less than 7 days - show days\n if (daysDiff < 7) {\n return `${daysDiff}d`\n }\n\n // Less than 4 weeks - show weeks\n if (daysDiff < 28) {\n const weeks = Math.floor(daysDiff / 7)\n return `${weeks}w`\n }\n\n // More than 4 weeks - show date as MM/DD/YY\n return date.toLocaleDateString('en-US', {\n month: 'numeric',\n day: 'numeric',\n year: '2-digit',\n })\n}\n","/**\n * Generate a fruit emoji based on a string id\n * Returns a consistent fruit emoji for the same id\n */\nconst EMOJIS = [\n '๐ŸŽ', // Apple\n '๐ŸŒ', // Banana\n '๐Ÿ‡', // Grape\n '๐ŸŠ', // Orange\n '๐Ÿ“', // Strawberry\n '๐Ÿฅฅ', // Coconut\n '๐Ÿ’', // Cherry\n '๐Ÿฅญ', // Mango\n '๐Ÿ‰', // Watermelon\n '๐Ÿ‹', // Lemon\n '๐Ÿฅ', // Kiwi\n '๐Ÿซ’', // Olive\n '๐Ÿˆ', // Melon\n]\n\n/**\n * Simple hash function to convert string to number\n */\nfunction hashString(str: string): number {\n let hash = 0\n for (let i = 0; i < str.length; i++) {\n const char = str.charCodeAt(i)\n hash = (hash << 5) - hash + char\n hash = hash & hash // Convert to 32-bit integer\n }\n return Math.abs(hash)\n}\n\n/**\n * Get a fruit emoji based on an id string\n * @param id - The string id to generate emoji from\n * @returns A fruit emoji string\n */\nexport function getAvatarEmoji(id: string): string {\n const hash = hashString(id)\n const index = hash % EMOJIS.length\n return EMOJIS[index]\n}\n\n","import classNames from 'classnames'\nimport React from 'react'\n\nimport { getAvatarEmoji } from './getAvatarEmoji'\n\nexport interface AvatarProps {\n id: string\n name: string\n image?: string\n size?: number\n className?: string\n shape?: 'squircle' | 'circle'\n}\n\n/**\n * Avatar component that displays a user image or colored initial fallback\n */\nexport const Avatar: React.FC<AvatarProps> = ({\n id,\n image,\n size = 40,\n className,\n shape = 'squircle',\n}) => {\n const emoji = getAvatarEmoji(id)\n\n // Determine font size based on avatar size\n const getFontSizeClass = () => {\n if (size < 32) return 'text-xs'\n if (size < 56) return 'text-sm'\n return 'text-lg'\n }\n\n const fontSizeClass = getFontSizeClass()\n\n const borderStyle =\n shape === 'circle'\n ? { borderRadius: '50%' }\n : {\n borderRadius: '33%',\n 'corner-shape': 'superellipse(1.3)',\n }\n\n return (\n <div\n className={classNames('flex-shrink-0 overflow-hidden', className)}\n style={{\n width: `${size}px`,\n height: `${size}px`,\n ...borderStyle,\n }}\n >\n {image ? (\n <img\n src={image}\n alt=\"\"\n className=\"h-full w-full object-cover aspect-square\"\n />\n ) : (\n <div\n aria-hidden=\"true\"\n className={classNames(\n 'avatar-fallback flex h-full w-full items-center justify-center font-semibold bg-[#E6E5E3] select-none transition-colors',\n fontSizeClass\n )}\n >\n {emoji}\n </div>\n )}\n </div>\n )\n}\n","import { GiftIcon } from '@phosphor-icons/react'\nimport { LocalMessage } from 'stream-chat'\n\ninterface MessageTagProps {\n message: LocalMessage\n /** When true, renders as a standalone bubble instead of a small tag */\n standalone?: boolean\n}\n\nconst SparkleIcon = () => (\n <svg width=\"12\" height=\"12\" viewBox=\"0 0 10 10\" fill=\"none\">\n <path\n d=\"M10.003 5a.705.705 0 0 1-.469.67L6.7 6.7 5.67 9.535a.715.715 0 0 1-1.34 0L3.3 6.7.466 5.67a.715.715 0 0 1 0-1.34L3.3 3.3 4.33.466a.715.715 0 0 1 1.34 0L6.7 3.3l2.834 1.03a.705.705 0 0 1 .469.67\"\n fill=\"currentColor\"\n />\n </svg>\n)\n\n/** Check if a message is a tip based on metadata */\nexport const isTipMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_TIP'\n}\n\n/** Check if a message is a paid message based on metadata */\nexport const isPaidMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_PAID'\n}\n\n/** Check if a message is a chatbot message based on metadata */\nexport const isChatbotMessage = (message: LocalMessage): boolean => {\n return message.metadata?.custom_type === 'MESSAGE_CHATBOT'\n}\n\n/** Check if a message has a tip/paid tag (both render the same) */\nexport const isTipOrPaidMessage = (message: LocalMessage): boolean => {\n return isTipMessage(message) || isPaidMessage(message)\n}\n\n/** Check if a message is a tip/paid-only message (no text) */\nexport const isTipOnlyMessage = (message: LocalMessage): boolean => {\n return isTipOrPaidMessage(message) && !message.text?.trim()\n}\n\nexport const MessageTag = ({\n message,\n standalone = false,\n}: MessageTagProps) => {\n const isTipOrPaid = isTipOrPaidMessage(message)\n const isChatbot = isChatbotMessage(message)\n\n if (!isTipOrPaid && !isChatbot) {\n return null\n }\n\n if (isTipOrPaid) {\n const amountText = message.metadata?.amount_text\n if (!amountText) return null\n\n const className = standalone\n ? 'message-tip-standalone'\n : 'message-tag message-tag--tip'\n\n const label = standalone\n ? `${amountText} tip`\n : `Delivered with ${amountText} tip`\n\n return (\n <div className={className}>\n <GiftIcon size={standalone ? 14 : 12} />\n <span>{label}</span>\n </div>\n )\n }\n\n // Chatbot tag\n return (\n <div className=\"message-tag message-tag--chatbot\">\n <span className=\"message-tag__icon\" style={{ marginTop: -1 }}>\n <SparkleIcon />\n </span>\n <span className=\"message-tag__label\">Chatbot</span>\n </div>\n )\n}\n","import classNames from 'classnames'\nimport React from 'react'\nimport { Channel, LocalMessage } from 'stream-chat'\nimport { ChannelPreviewUIComponentProps } from 'stream-chat-react'\n\nimport { formatRelativeTime } from '../../utils/formatRelativeTime'\nimport { Avatar } from '../Avatar'\nimport { isChatbotMessage } from '../CustomMessage/MessageTag'\n\ntype CustomChannelPreviewProps = ChannelPreviewUIComponentProps & {\n selectedChannel?: Channel | null\n onChannelSelect: (channel: Channel) => void\n debug?: boolean\n renderMessagePreview?: (\n message: LocalMessage | undefined,\n defaultPreview?: string\n ) => React.ReactNode\n}\n\n/**\n * Custom channel preview that handles selection\n */\nconst CustomChannelPreview = React.memo<CustomChannelPreviewProps>(\n ({\n channel,\n selectedChannel,\n onChannelSelect,\n debug = false,\n unread,\n renderMessagePreview,\n }) => {\n const isSelected = selectedChannel?.id === channel?.id\n\n const handleClick = () => {\n if (channel) {\n onChannelSelect(channel)\n }\n }\n\n // Get participant info\n const members = Object.values(channel?.state?.members || {})\n const participant = members.find(\n (member) => member.user?.id && member.user.id !== channel?._client?.userID\n )\n const participantName = participant?.user?.name || 'Conversation'\n const participantImage = participant?.user?.image\n\n // Get last message and format timestamp\n const lastMessage =\n channel?.state?.messages?.[channel.state.messages.length - 1]\n\n const getLastMessageText = () => {\n if (lastMessage?.text) return lastMessage.text\n\n const attachment = lastMessage?.attachments?.[0]\n if (attachment) {\n // Link previews - show the actual URL\n if (attachment.og_scrape_url) return attachment.og_scrape_url\n\n // Type-based detection\n if (attachment.type === 'image') return '๐Ÿ“ท Sent an image'\n if (attachment.type === 'video') return '๐ŸŽฅ Sent a video'\n if (attachment.type === 'audio') return '๐ŸŽต Sent audio'\n if (attachment.type === 'file') return '๐Ÿ“Ž Sent a file'\n\n // Fallback\n return '๐Ÿ“Ž Sent an attachment'\n }\n\n return 'No messages yet'\n }\n\n const lastMessageText = getLastMessageText()\n const lastMessageTime = lastMessage?.created_at\n ? formatRelativeTime(new Date(lastMessage.created_at))\n : ''\n const isLastMessageFromChatbot = lastMessage\n ? isChatbotMessage(lastMessage)\n : false\n\n const messagePreview = renderMessagePreview\n ? renderMessagePreview(lastMessage, lastMessageText)\n : `${isLastMessageFromChatbot ? 'โœจ ' : ''}${lastMessageText}`\n\n // Use the unread prop passed by Stream Chat (reactive and updates automatically)\n const unreadCount = unread ?? 0\n\n if (debug) {\n console.log('๐Ÿ“บ [ChannelList] ๐Ÿ“‹ CHANNEL PREVIEW RENDER', {\n channelId: channel?.id,\n isSelected,\n participantName,\n unreadCount,\n hasTimestamp: !!lastMessageTime,\n })\n }\n\n return (\n <button\n type=\"button\"\n onClick={handleClick}\n className={classNames(\n 'group w-full px-4 py-3 transition-colors text-left max-w-full overflow-hidden focus-ring',\n {\n 'bg-primary-alt/10 border-l-4 border-l-primary': isSelected,\n 'hover:bg-sand': !isSelected,\n }\n )}\n >\n <div className=\"flex items-start gap-3\">\n {/* Avatar */}\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={44}\n className=\"[&_.avatar-fallback]:group-hover:bg-[#eeeeee]\"\n />\n\n {/* Content column */}\n <div className=\"flex-1 min-w-0 flex flex-col gap-1\">\n {/* Name and timestamp row */}\n <div className=\"flex items-center justify-between gap-2\">\n <h3\n className={classNames(\n 'text-sm font-medium truncate',\n isSelected ? 'text-primary' : 'text-charcoal'\n )}\n >\n {participantName}\n </h3>\n {lastMessageTime && (\n <span className=\"text-xs text-stone flex-shrink-0\">\n {lastMessageTime}\n </span>\n )}\n </div>\n\n {/* Message and unread badge row */}\n <div className=\"flex items-center justify-between gap-2 min-w-0\">\n <p className=\"text-xs text-stone flex-1 line-clamp-1\">\n {messagePreview}\n </p>\n {unreadCount > 0 && (\n <span className=\"bg-[#7f22fe] text-white text-xs px-2 py-0.5 rounded-full min-w-[20px] text-center flex-shrink-0\">\n {unreadCount > 99 ? '99+' : unreadCount}\n </span>\n )}\n </div>\n </div>\n </div>\n </button>\n )\n }\n)\n\nexport default CustomChannelPreview\nCustomChannelPreview.displayName = 'CustomChannelPreview'\n","import classNames from 'classnames'\nimport React from 'react'\nimport type { ChannelPreviewUIComponentProps } from 'stream-chat-react'\nimport { ChannelList as StreamChannelList } from 'stream-chat-react'\n\nimport { useMessagingContext } from '../../providers/MessagingProvider'\nimport type { ChannelListProps } from '../../types'\n\nimport CustomChannelPreview from './CustomChannelPreview'\n\n/**\n * Channel list component with customizable header and actions\n */\nexport const ChannelList = React.memo<ChannelListProps>(\n ({\n onChannelSelect,\n selectedChannel,\n filters,\n className,\n customEmptyStateIndicator,\n renderMessagePreview,\n }) => {\n // Track renders\n const renderCountRef = React.useRef(0)\n renderCountRef.current++\n\n // Get debug flag from context\n const { debug = false } = useMessagingContext()\n\n if (debug) {\n console.log('๐Ÿ“บ [ChannelList] ๐Ÿ”„ RENDER START', {\n renderCount: renderCountRef.current,\n selectedChannelId: selectedChannel?.id,\n filters,\n })\n }\n\n // Memoize Preview component to prevent re-renders\n const PreviewComponent = React.useMemo(() => {\n const Preview = (props: ChannelPreviewUIComponentProps) => (\n <CustomChannelPreview\n {...props}\n selectedChannel={selectedChannel}\n onChannelSelect={onChannelSelect}\n debug={debug}\n renderMessagePreview={renderMessagePreview}\n />\n )\n return Preview\n }, [selectedChannel, onChannelSelect, debug, renderMessagePreview])\n\n return (\n <div\n className={classNames(\n 'messaging-channel-list h-full flex flex-col min-w-0 overflow-hidden',\n className\n )}\n >\n {/* Channel List */}\n <div className=\"flex-1 overflow-hidden min-w-0\">\n <StreamChannelList\n key={JSON.stringify(filters)}\n filters={filters}\n sort={{ last_message_at: -1 }}\n options={{ limit: 30 }}\n Preview={PreviewComponent}\n EmptyStateIndicator={customEmptyStateIndicator}\n />\n </div>\n </div>\n )\n }\n)\nChannelList.displayName = 'ChannelList'\n","import classNames from \"classnames\";\nimport React from \"react\";\n\ninterface ActionButtonProps\n extends React.ButtonHTMLAttributes<HTMLButtonElement> {\n variant?: \"default\" | \"danger\";\n}\n\nconst ActionButton = ({\n variant = \"default\",\n className,\n children,\n ...rest\n}: ActionButtonProps) => {\n const isDanger = variant === \"danger\";\n return (\n <button\n type=\"button\"\n className={classNames(\n \"flex w-full items-center gap-3 rounded-lg px-4 py-3 text-left text-sm transition-colors focus-ring disabled:cursor-not-allowed disabled:opacity-60\",\n isDanger\n ? \"text-danger hover:bg-danger/50\"\n : \"text-charcoal hover:bg-sand\",\n className,\n )}\n {...rest}\n >\n {children}\n </button>\n );\n};\n\nexport default ActionButton;\n","import classNames from \"classnames\";\nimport React from \"react\";\n\ninterface IconButtonProps\n extends Omit<\n React.ButtonHTMLAttributes<HTMLButtonElement>,\n \"type\" | \"children\"\n > {\n label: string;\n children: React.ReactNode;\n className?: string;\n}\n\nexport function IconButton({ label, className, children, ...rest }: IconButtonProps) {\n return (\n <button\n type=\"button\"\n className={classNames(\n \"rounded-full p-2 transition-colors focus-ring\",\n {\n \"cursor-not-allowed opacity-50\": rest.disabled,\n \"hover:bg-sand\": !rest.disabled,\n },\n className,\n )}\n {...rest}\n >\n <span className=\"sr-only\">{label}</span>\n {children}\n </button>\n );\n}","import { XIcon } from '@phosphor-icons/react'\n\nimport { IconButton } from '../IconButton'\n\ninterface CloseButtonProps {\n onClick: () => void\n}\n\nexport function CloseButton({ onClick }: CloseButtonProps) {\n return (\n <IconButton label=\"Close\" onClick={onClick} className=\"p-1\">\n <XIcon className=\"h-5 w-5 text-stone\" weight=\"bold\" />\n </IconButton>\n )\n}\n","import { DateSeparator, type DateSeparatorProps } from 'stream-chat-react'\n\nexport const CustomDateSeparator = (props: DateSeparatorProps) => (\n <DateSeparator {...props} position=\"center\" />\n)\n","import classNames from 'classnames'\nimport React, { useMemo, useState } from 'react'\nimport {\n Attachment as DefaultAttachment,\n EditMessageModal as DefaultEditMessageModal,\n MessageBounceModal,\n MessageBouncePrompt as DefaultMessageBouncePrompt,\n MessageBlocked as DefaultMessageBlocked,\n MessageDeleted as DefaultMessageDeleted,\n MessageErrorIcon,\n MessageIsThreadReplyInChannelButtonIndicator as DefaultMessageIsThreadReplyInChannelButtonIndicator,\n MessageRepliesCountButton as DefaultMessageRepliesCountButton,\n MessageText,\n Poll,\n ReminderNotification as DefaultReminderNotification,\n StreamedMessageText as DefaultStreamedMessageText,\n areMessageUIPropsEqual,\n isDateSeparatorMessage,\n isMessageBlocked,\n isMessageBounced,\n messageHasAttachments,\n messageHasReactions,\n useComponentContext,\n useChatContext,\n useMessageContext,\n useMessageReminder,\n type MessageContextValue,\n type MessageUIComponentProps,\n} from 'stream-chat-react'\n\nimport { Avatar } from '../Avatar'\n\nimport { MessageTag, isTipOnlyMessage } from './MessageTag'\n\ntype CustomMessageWithContextProps = MessageContextValue\n\nconst CustomMessageWithContext = (props: CustomMessageWithContextProps) => {\n const {\n additionalMessageInputProps,\n editing,\n endOfGroup,\n firstOfGroup,\n groupedByUser,\n handleAction,\n handleOpenThread,\n handleRetry,\n highlighted,\n isMessageAIGenerated,\n isMyMessage,\n message,\n renderText,\n threadList,\n } = props\n\n const { client } = useChatContext('CustomMessage')\n const [isBounceDialogOpen, setIsBounceDialogOpen] = useState(false)\n const reminder = useMessageReminder(message.id)\n\n const {\n Attachment = DefaultAttachment,\n EditMessageModal = DefaultEditMessageModal,\n MessageBlocked = DefaultMessageBlocked,\n MessageBouncePrompt = DefaultMessageBouncePrompt,\n MessageDeleted = DefaultMessageDeleted,\n MessageIsThreadReplyInChannelButtonIndicator = DefaultMessageIsThreadReplyInChannelButtonIndicator,\n MessageRepliesCountButton = DefaultMessageRepliesCountButton,\n ReminderNotification = DefaultReminderNotification,\n StreamedMessageText = DefaultStreamedMessageText,\n PinIndicator,\n } = useComponentContext('CustomMessage')\n\n const hasAttachment = messageHasAttachments(message)\n const hasReactions = messageHasReactions(message)\n const isAIGenerated = useMemo(\n () => isMessageAIGenerated?.(message),\n [isMessageAIGenerated, message]\n )\n const finalAttachments = useMemo(\n () =>\n !message.shared_location && !message.attachments\n ? []\n : !message.shared_location\n ? message.attachments\n : [message.shared_location, ...(message.attachments ?? [])],\n [message]\n )\n\n if (isDateSeparatorMessage(message)) {\n return null\n }\n\n if (message.deleted_at || message.type === 'deleted') {\n return <MessageDeleted message={message} />\n }\n\n if (isMessageBlocked(message)) {\n return <MessageBlocked />\n }\n\n const showReplyCountButton = !threadList && !!message.reply_count\n const showIsReplyInChannel =\n !threadList && message.show_in_channel && message.parent_id\n const allowRetry =\n message.status === 'failed' && message.error?.status !== 403\n const isBounced = isMessageBounced(message)\n let handleClick: (() => void) | undefined = undefined\n\n if (allowRetry) {\n handleClick = () => handleRetry(message)\n } else if (isBounced) {\n handleClick = () => setIsBounceDialogOpen(true)\n }\n\n const rootClassName = classNames(\n 'str-chat__message str-chat__message-simple',\n `str-chat__message--${message.type}`,\n `str-chat__message--${message.status}`,\n isMyMessage()\n ? 'str-chat__message--me str-chat__message-simple--me'\n : 'str-chat__message--other',\n message.text ? 'str-chat__message--has-text' : 'has-no-text',\n {\n 'str-chat__message--has-attachment': hasAttachment,\n 'str-chat__message--highlighted': highlighted,\n 'str-chat__message--pinned pinned-message': message.pinned,\n 'str-chat__message--with-reactions': hasReactions,\n 'str-chat__message-send-can-be-retried':\n message?.status === 'failed' && message?.error?.status !== 403,\n 'str-chat__message-with-thread-link':\n showReplyCountButton || showIsReplyInChannel,\n 'str-chat__virtual-message__wrapper--end': endOfGroup,\n 'str-chat__virtual-message__wrapper--first': firstOfGroup,\n 'str-chat__virtual-message__wrapper--group': groupedByUser,\n }\n )\n\n const poll = message.poll_id && client.polls.fromState(message.poll_id)\n const isTipOnly = isTipOnlyMessage(message)\n\n return (\n <>\n {editing && (\n <EditMessageModal\n additionalMessageInputProps={additionalMessageInputProps}\n />\n )}\n {isBounceDialogOpen && (\n <MessageBounceModal\n MessageBouncePrompt={MessageBouncePrompt}\n onClose={() => setIsBounceDialogOpen(false)}\n open={isBounceDialogOpen}\n />\n )}\n <div className={rootClassName} key={message.id}>\n {PinIndicator && <PinIndicator />}\n {!!reminder && <ReminderNotification reminder={reminder} />}\n {message.user && (\n <Avatar\n className=\"str-chat__avatar str-chat__message-sender-avatar\"\n id={message.user.id}\n image={message.user.image}\n name={message.user.name || message.user.id}\n />\n )}\n {/* eslint-disable-next-line jsx-a11y/no-static-element-interactions */}\n <div\n className={classNames('str-chat__message-inner', {\n 'str-chat__simple-message--error-failed': allowRetry || isBounced,\n })}\n data-testid=\"message-inner\"\n onClick={handleClick}\n onKeyDown={handleClick}\n role={handleClick ? 'button' : undefined}\n tabIndex={handleClick ? 0 : undefined}\n style={{\n // Force margins to 0 to prevent hover layout shift\n // Stream Chat CSS sets margin-inline-end/start to 78px, then 0 on hover\n marginInlineEnd: 0,\n marginInlineStart: 0,\n }}\n >\n {isTipOnly ? (\n /* Tip-only messages render as a standalone bubble */\n <MessageTag message={message} standalone />\n ) : (\n <div className=\"str-chat__message-bubble-wrapper\">\n <div className=\"str-chat__message-bubble\">\n {poll && <Poll poll={poll} />}\n {finalAttachments?.length && !message.quoted_message ? (\n <Attachment\n actionHandler={handleAction}\n attachments={finalAttachments}\n />\n ) : null}\n {isAIGenerated ? (\n <StreamedMessageText\n message={message}\n renderText={renderText}\n />\n ) : (\n <MessageText message={message} renderText={renderText} />\n )}\n <MessageErrorIcon />\n </div>\n {/* Tag positioned outside and below the bubble */}\n <MessageTag message={message} />\n </div>\n )}\n </div>\n {showReplyCountButton && (\n <MessageRepliesCountButton\n onClick={handleOpenThread}\n reply_count={message.reply_count}\n />\n )}\n {showIsReplyInChannel && (\n <MessageIsThreadReplyInChannelButtonIndicator />\n )}\n </div>\n </>\n )\n}\n\nconst MemoizedCustomMessage = React.memo(\n CustomMessageWithContext,\n areMessageUIPropsEqual\n) as typeof CustomMessageWithContext\n\nexport const CustomMessage = (props: MessageUIComponentProps) => {\n const messageContext = useMessageContext('CustomMessage')\n return <MemoizedCustomMessage {...messageContext} {...props} />\n}\n","import { XIcon } from '@phosphor-icons/react'\nimport React from 'react'\nimport {\n LinkPreview,\n LinkPreviewsManager,\n LinkPreviewsManagerState,\n} from 'stream-chat'\nimport { useMessageComposer, useStateStore } from 'stream-chat-react'\n\nconst linkPreviewsManagerStateSelector = (state: LinkPreviewsManagerState) => ({\n linkPreviews: Array.from(state.previews.values()).filter(\n (preview) =>\n LinkPreviewsManager.previewIsLoaded(preview) ||\n LinkPreviewsManager.previewIsLoading(preview)\n ),\n})\n\ninterface CustomLinkPreviewCardProps {\n link: LinkPreview\n onDismiss: (url: string) => void\n}\n\nconst CustomLinkPreviewCard: React.FC<CustomLinkPreviewCardProps> = ({\n link,\n onDismiss,\n}) => {\n const { og_scrape_url, title, image_url } = link\n\n const handleDismissLink = (e: React.MouseEvent) => {\n e.preventDefault()\n onDismiss(og_scrape_url)\n }\n\n return (\n <a\n href={og_scrape_url}\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n className=\"relative w-full block rounded-[24px] bg-[#121110] p-2 no-underline transition-opacity hover:opacity-90\"\n >\n {image_url && (\n <img\n src={image_url}\n alt={title || ''}\n className=\"h-[148px] w-full rounded-[20px] object-cover\"\n />\n )}\n <button\n type=\"button\"\n onClick={handleDismissLink}\n className=\"absolute right-4 top-4 flex size-6 items-center justify-center rounded-full border border-white/40 bg-white/70 backdrop-blur-2xl\"\n aria-label=\"Close link preview\"\n >\n <XIcon className=\"size-4 text-black/90\" />\n </button>\n <div className=\"p-2\">\n {title && (\n <div className=\"text-[14px] font-medium leading-5 text-white\">\n {title}\n </div>\n )}\n <div className=\"text-[12px] leading-4 text-white/55\">\n {og_scrape_url}\n </div>\n </div>\n </a>\n )\n}\n\nexport const CustomLinkPreviewList = () => {\n const { linkPreviewsManager } = useMessageComposer()\n\n const { linkPreviews: stateLinkPreviews } = useStateStore(\n linkPreviewsManager.state,\n linkPreviewsManagerStateSelector\n )\n\n const handleDismiss = (url: string) => {\n linkPreviewsManager.dismissPreview(url)\n }\n\n const showLinkPreviews = stateLinkPreviews.length > 0\n\n if (!showLinkPreviews) return null\n\n return (\n <div className=\"flex flex-col items-center w-full gap-2 mb-4\">\n {stateLinkPreviews.map((linkPreview) => (\n <CustomLinkPreviewCard\n key={linkPreview.og_scrape_url}\n link={linkPreview}\n onDismiss={handleDismiss}\n />\n ))}\n </div>\n )\n}\n","import { ArrowUpIcon } from '@phosphor-icons/react'\nimport React from 'react'\nimport {\n AttachmentPreviewList,\n MessageInput,\n QuotedMessagePreview,\n SimpleAttachmentSelector,\n TextareaComposer,\n useMessageComposerHasSendableData,\n useMessageInputContext,\n} from 'stream-chat-react'\n\nimport { CustomLinkPreviewList } from '../CustomLinkPreviewList'\n\nconst CustomMessageInputInner: React.FC = () => {\n const { handleSubmit } = useMessageInputContext()\n const hasSendableData = useMessageComposerHasSendableData()\n\n return (\n <>\n <div className=\"left-container\">\n <SimpleAttachmentSelector />\n </div>\n <div className=\"central-container min-w-0 w-full p-2 bg-white rounded-[1.5rem] shadow-[0_4px_16px_0_rgba(0,0,0,0.08),0_1px_2px_0_rgba(0,0,0,0.04),0_0_0_1px_rgba(0,0,0,0.04)]\">\n <QuotedMessagePreview />\n <CustomLinkPreviewList />\n <AttachmentPreviewList />\n <div className=\"flex\">\n <div className=\"w-full ml-2 mr-4 self-center leading-[0]\">\n <TextareaComposer\n className=\"w-full resize-none outline-none leading-6\"\n // While this might usually be considered an anti-pattern, in most\n // cases, when a message thread is rendered, we want the input to\n // gain focus automatically.\n // eslint-disable-next-line jsx-a11y/no-autofocus\n autoFocus\n maxRows={4}\n />\n </div>\n <button\n aria-label=\"Send\"\n className=\"str-chat__send-button mt-auto flex justify-center items-center flex-shrink-0 rounded-full size-8 bg-[#121110] disabled:bg-[#F1F0EE] disabled:text-black/20 text-white\"\n data-testid=\"send-button\"\n disabled={!hasSendableData}\n onClick={handleSubmit}\n type=\"button\"\n >\n <ArrowUpIcon className=\"size-4\" />\n </button>\n </div>\n </div>\n </>\n )\n}\n\nexport interface CustomMessageInputProps {\n renderActions?: () => React.ReactNode\n}\n\nexport const CustomMessageInput: React.FC<CustomMessageInputProps> = ({\n renderActions,\n}) => (\n <div className=\"message-input flex items-center gap-2 p-4\">\n {renderActions && renderActions?.()}\n <MessageInput Input={CustomMessageInputInner} />\n </div>\n)\n","import { MessageTimestamp, type EventComponentProps } from 'stream-chat-react'\n\nexport const CustomSystemMessage: React.FC<EventComponentProps> = (props) => {\n const isDateHidden = props.message.hide_date === true\n\n return (\n <div className=\"str-chat__message--system\" data-testid=\"message-system\">\n <div className=\"str-chat__message--system__text\">\n <div className=\"str-chat__message--system__line\"></div>\n <p>{props.message.text}</p>\n <div className=\"str-chat__message--system__line\"></div>\n </div>\n {!isDateHidden && <MessageTimestamp message={props.message} />}\n </div>\n )\n}\n\n","import React from 'react'\n\n/**\n * Empty state component shown when a channel has no messages\n * Returns null to show nothing - the LoadingIndicator handles the loading state\n */\nexport const ChannelEmptyState: React.FC = () => null\n","import classNames from 'classnames'\n\ntype LoadingProps = {\n className?: string\n message?: string\n}\n\nconst Loading = ({ className, message }: LoadingProps) => (\n <div\n className={classNames('flex items-center justify-center h-full', className)}\n >\n <svg viewBox=\"0 0 100 100\" className=\"size-8 fill-pebble\" stroke=\"none\">\n <circle cx=\"6\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 15 ; 0 -15; 0 15\"\n repeatCount=\"indefinite\"\n begin=\"0.1\"\n />\n </circle>\n <circle cx=\"30\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 10 ; 0 -10; 0 10\"\n repeatCount=\"indefinite\"\n begin=\"0.2\"\n />\n </circle>\n <circle cx=\"54\" cy=\"50\" r=\"6\">\n <animateTransform\n attributeName=\"transform\"\n dur=\"1s\"\n type=\"translate\"\n values=\"0 5 ; 0 -5; 0 5\"\n repeatCount=\"indefinite\"\n begin=\"0.3\"\n />\n </circle>\n </svg>\n {message && <span className=\"text-stone\">{message}</span>}\n </div>\n)\n\nexport default Loading\n","import React from 'react'\n\nimport Loading from '../Loading'\n\n/**\n * Loading state component\n */\nexport const LoadingState = React.memo(() => (\n <div className=\"messaging-loading-state flex items-center justify-center h-full\">\n <div className=\"flex items-center\">\n <Loading className=\"w-6 h-6\" />\n <span className=\"text-sm text-stone\">Loading messages</span>\n </div>\n </div>\n))\nLoadingState.displayName = 'LoadingState'\n","import {\n ArrowLeftIcon,\n DotsThreeIcon,\n FlagIcon,\n ProhibitInsetIcon,\n SignOutIcon,\n SpinnerGapIcon,\n StarIcon,\n} from '@phosphor-icons/react'\nimport classNames from 'classnames'\nimport React, { useState, useCallback, useRef, useEffect } from 'react'\nimport {\n Channel as ChannelType,\n ChannelMemberResponse,\n Event,\n} from 'stream-chat'\nimport {\n Channel,\n Window,\n MessageList,\n useChannelStateContext,\n WithComponents,\n MessageUIComponentProps,\n} from 'stream-chat-react'\n\nimport { useMessagingContext } from '../providers/MessagingProvider'\nimport type { ChannelViewProps } from '../types'\n\nimport ActionButton from './ActionButton'\nimport { Avatar } from './Avatar'\nimport { CloseButton } from './CloseButton'\nimport { CustomDateSeparator } from './CustomDateSeparator'\nimport { CustomMessage } from './CustomMessage'\nimport { CustomMessageInput } from './CustomMessageInput'\nimport { CustomSystemMessage } from './CustomSystemMessage'\nimport { ChannelEmptyState } from './MessagingShell/ChannelEmptyState'\nimport { LoadingState } from './MessagingShell/LoadingState'\n\n// Custom user type with email and username\ntype CustomUser = {\n email?: string\n username?: string\n}\n\n// Blocked user from Stream Chat API\ntype BlockedUser = {\n blocked_user_id: string\n}\n\n/**\n * Custom channel header component\n */\nconst CustomChannelHeader: React.FC<{\n onBack?: () => void\n showBackButton: boolean\n onShowInfo: () => void\n canShowInfo: boolean\n showStarButton?: boolean\n}> = ({\n onBack,\n showBackButton,\n onShowInfo,\n canShowInfo,\n showStarButton = false,\n}) => {\n const { channel } = useChannelStateContext()\n\n // Get participant info (excluding current user)\n const participant = React.useMemo(() => {\n const members = Object.values(channel.state.members || {})\n return members.find(\n (member) => member.user?.id && member.user.id !== channel._client.userID\n )\n }, [channel._client.userID, channel.state.members])\n\n const participantName =\n participant?.user?.name || participant?.user?.id || 'Unknown member'\n const participantImage = participant?.user?.image\n\n const [isStarred, setIsStarred] = useState(\n !!channel.state.membership?.pinned_at\n )\n\n useEffect(() => {\n const handleMemberUpdate = (event: Event) => {\n setIsStarred(\n event?.member\n ? !!event.member.pinned_at\n : !!channel.state.membership?.pinned_at\n )\n }\n\n channel.on('member.updated', handleMemberUpdate)\n\n return () => {\n channel.off('member.updated', handleMemberUpdate)\n }\n }, [channel])\n\n const handleStarClick = async () => {\n try {\n if (isStarred) {\n await channel.unpin()\n } else {\n await channel.pin()\n }\n } catch (error) {\n console.error(\n '[CustomChannelHeader] Failed to update pinned status:',\n error\n )\n }\n }\n\n return (\n <div className=\"@container\">\n <div className=\"grid grid-cols-[1fr_auto_1fr] w-full items-center @lg:hidden\">\n <div className=\"flex items-center gap-2\">\n {showBackButton && (\n <button\n className={classNames(\n 'size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center'\n )}\n onClick={onBack || (() => {})}\n type=\"button\"\n aria-label=\"Back to conversations\"\n >\n <ArrowLeftIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n </div>\n <div className=\"flex flex-col gap-1 items-center\">\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={40}\n />\n <h1 className=\"text-xs font-medium text-black/90\">\n {participantName}\n </h1>\n </div>\n <div className=\"flex justify-end items-center gap-2\">\n {showStarButton && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={handleStarClick}\n type=\"button\"\n aria-label={\n isStarred ? 'Unstar conversation' : 'Star conversation'\n }\n >\n <StarIcon\n className={classNames('size-5', {\n 'text-yellow-600': isStarred,\n 'text-black/90': !isStarred,\n })}\n weight={isStarred ? 'duotone' : 'regular'}\n />\n </button>\n )}\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={onShowInfo}\n type=\"button\"\n aria-label=\"Show info\"\n >\n <DotsThreeIcon className=\"size-5 text-black/90\" />\n </button>\n </div>\n </div>\n <div className=\"hidden @lg:flex items-center justify-between gap-3 min-h-12\">\n <div className=\"flex items-center gap-4 min-w-0\">\n {showBackButton && onBack && (\n <button\n type=\"button\"\n onClick={onBack}\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n aria-label=\"Back to conversations\"\n >\n <ArrowLeftIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n {/* Avatar */}\n <Avatar\n id={participant?.user?.id || channel.id || 'unknown'}\n name={participantName}\n image={participantImage}\n size={40}\n />\n <div className=\"min-w-0\">\n <h1 className=\"font-medium text-black/90 truncate\">\n {participantName}\n </h1>\n </div>\n </div>\n <div className=\"flex items-center gap-2\">\n {showStarButton && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={handleStarClick}\n type=\"button\"\n aria-label={\n isStarred ? 'Unstar conversation' : 'Star conversation'\n }\n >\n <StarIcon\n className={classNames('size-5', {\n 'text-yellow-600': isStarred,\n 'text-black/90': !isStarred,\n })}\n weight={isStarred ? 'duotone' : 'regular'}\n />\n </button>\n )}\n {canShowInfo && onShowInfo && (\n <button\n className=\"size-10 rounded-full bg-[#F1F0EE] flex items-center justify-center\"\n onClick={onShowInfo}\n type=\"button\"\n aria-label=\"Show info\"\n >\n <DotsThreeIcon className=\"size-5 text-black/90\" />\n </button>\n )}\n </div>\n </div>\n </div>\n )\n}\n\n/**\n * Channel info dialog (matching original implementation)\n */\nconst ChannelInfoDialog: React.FC<{\n dialogRef: React.RefObject<HTMLDialogElement>\n onClose: () => void\n participant: ChannelMemberResponse | undefined\n channel: ChannelType\n followerStatusLabel?: string\n onLeaveConversation?: (channel: ChannelType) => void\n onBlockParticipant?: (participantId?: string) => void\n showDeleteConversation?: boolean\n onDeleteConversationClick?: () => void\n onBlockParticipantClick?: () => void\n onReportParticipantClick?: () => void\n}> = ({\n dialogRef,\n onClose,\n participant,\n channel,\n followerStatusLabel,\n onLeaveConversation,\n onBlockParticipant,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n}) => {\n const { service, debug } = useMessagingContext()\n const [isParticipantBlocked, setIsParticipantBlocked] = useState(false)\n const [isLeaving, setIsLeaving] = useState(false)\n const [isUpdatingBlockStatus, setIsUpdatingBlockStatus] = useState(false)\n\n // Check if participant is blocked when participant changes\n const checkIsParticipantBlocked = useCallback(async () => {\n if (!service || !participant?.user?.id) return\n\n try {\n const blockedUsers = await service.getBlockedUsers()\n const isBlocked = blockedUsers.some(\n (user: BlockedUser) => user.blocked_user_id === participant?.user?.id\n )\n setIsParticipantBlocked(isBlocked)\n } catch (error) {\n console.error(\n '[ChannelInfoDialog] Failed to check blocked status:',\n error\n )\n }\n }, [service, participant?.user?.id])\n\n useEffect(() => {\n checkIsParticipantBlocked()\n }, [checkIsParticipantBlocked])\n\n const handleLeaveConversation = async () => {\n if (isLeaving) return\n\n // Fire analytics callback before action\n onDeleteConversationClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Leave conversation', channel.cid)\n }\n setIsLeaving(true)\n\n try {\n const actingUserId = channel._client?.userID ?? null\n await channel.hide(actingUserId, false)\n\n if (onLeaveConversation) {\n await onLeaveConversation(channel)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to leave conversation', error)\n } finally {\n setIsLeaving(false)\n }\n }\n\n const handleBlockUser = async () => {\n if (isUpdatingBlockStatus || !service) return\n\n // Fire analytics callback before action\n onBlockParticipantClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Block member', participant?.user?.id)\n }\n setIsUpdatingBlockStatus(true)\n\n try {\n await service.blockUser(participant?.user?.id)\n\n if (onBlockParticipant) {\n await onBlockParticipant(participant?.user?.id)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to block member', error)\n } finally {\n setIsUpdatingBlockStatus(false)\n }\n }\n\n const handleUnblockUser = async () => {\n if (isUpdatingBlockStatus || !service) return\n\n // Fire analytics callback before action\n onBlockParticipantClick?.()\n\n if (debug) {\n console.log('[ChannelInfoDialog] Unblock member', participant?.user?.id)\n }\n setIsUpdatingBlockStatus(true)\n\n try {\n await service.unBlockUser(participant?.user?.id)\n\n if (onBlockParticipant) {\n await onBlockParticipant(participant?.user?.id)\n }\n\n onClose()\n } catch (error) {\n console.error('[ChannelInfoDialog] Failed to unblock member', error)\n } finally {\n setIsUpdatingBlockStatus(false)\n }\n }\n\n const handleReportUser = () => {\n // Fire analytics callback before action\n onReportParticipantClick?.()\n\n onClose()\n window.open(\n 'https://linktr.ee/s/about/trust-center/report',\n '_blank',\n 'noopener,noreferrer'\n )\n }\n\n if (!participant) return null\n\n const participantName =\n participant.user?.name || participant.user?.id || 'Unknown member'\n const participantImage = participant.user?.image\n const participantEmail = (participant.user as CustomUser)?.email\n const participantUsername = (participant.user as CustomUser)?.username\n const participantSecondary = participantEmail\n ? participantEmail\n : participantUsername\n ? `linktr.ee/${participantUsername}`\n : undefined\n const participantId = participant.user?.id || 'unknown'\n\n return (\n // eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-noninteractive-element-interactions\n <dialog\n ref={dialogRef}\n className=\"mes-dialog group\"\n onClose={onClose}\n onClick={(e) => {\n if (e.target === dialogRef.current) {\n onClose()\n }\n }}\n >\n <div className=\"ml-auto flex h-full w-full flex-col bg-white shadow-none transition-shadow duration-200 group-open:shadow-max-elevation-light\">\n <div className=\"flex items-center justify-between border-b border-sand px-4 py-3\">\n <h2 className=\"text-base font-semibold text-charcoal\">Chat info</h2>\n <CloseButton onClick={onClose} />\n </div>\n\n <div className=\"flex-1 px-2 overflow-y-auto w-full\">\n <div\n className=\"flex flex-col items-center gap-3 self-stretch px-4 py-2 mt-6 rounded-lg border border-black/[0.04]\"\n style={{ backgroundColor: '#FBFAF9' }}\n >\n <div className=\"flex items-center gap-3 w-full\">\n <Avatar\n id={participantId}\n name={participantName}\n image={participantImage}\n size={88}\n shape=\"circle\"\n />\n <div className=\"flex flex-col min-w-0 flex-1\">\n <p className=\"truncate text-base font-semibold text-charcoal\">\n {participantName}\n </p>\n {participantSecondary && (\n <p className=\"truncate text-sm text-[#00000055]\">\n {participantSecondary}\n </p>\n )}\n {followerStatusLabel && (\n <span\n className=\"mt-1 rounded-full text-xs font-normal w-fit\"\n style={{\n padding: '4px 8px',\n backgroundColor:\n followerStatusLabel === 'Subscribed to you'\n ? '#DCFCE7'\n : '#F5F5F4',\n color:\n followerStatusLabel === 'Subscribed to you'\n ? '#008236'\n : '#78716C',\n lineHeight: '133.333%',\n letterSpacing: '0.21px',\n }}\n >\n {followerStatusLabel}\n </span>\n )}\n </div>\n </div>\n </div>\n\n <ul className=\"flex flex-col gap-2 mt-2\">\n {showDeleteConversation && (\n <li>\n <ActionButton\n onClick={handleLeaveConversation}\n disabled={isLeaving}\n aria-busy={isLeaving}\n >\n {isLeaving ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <SignOutIcon className=\"h-5 w-5\" />\n )}\n <span>Delete Conversation</span>\n </ActionButton>\n </li>\n )}\n <li>\n {isParticipantBlocked ? (\n <ActionButton\n onClick={handleUnblockUser}\n disabled={isUpdatingBlockStatus}\n aria-busy={isUpdatingBlockStatus}\n >\n {isUpdatingBlockStatus ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <ProhibitInsetIcon className=\"h-5 w-5\" />\n )}\n <span>Unblock</span>\n </ActionButton>\n ) : (\n <ActionButton\n onClick={handleBlockUser}\n disabled={isUpdatingBlockStatus}\n aria-busy={isUpdatingBlockStatus}\n >\n {isUpdatingBlockStatus ? (\n <SpinnerGapIcon className=\"h-5 w-5 animate-spin\" />\n ) : (\n <ProhibitInsetIcon className=\"h-5 w-5\" />\n )}\n <span>Block</span>\n </ActionButton>\n )}\n </li>\n <li>\n <ActionButton variant=\"danger\" onClick={handleReportUser}>\n <FlagIcon className=\"h-5 w-5\" />\n <span>Report</span>\n </ActionButton>\n </li>\n </ul>\n </div>\n </div>\n </dialog>\n )\n}\n\n/**\n * Inner component that has access to channel context\n */\nconst ChannelViewInner: React.FC<{\n onBack?: () => void\n showBackButton: boolean\n renderMessageInputActions?: (channel: ChannelType) => React.ReactNode\n onLeaveConversation?: (channel: ChannelType) => void\n onBlockParticipant?: (participantId?: string) => void\n CustomChannelEmptyState?: React.ComponentType\n showDeleteConversation?: boolean\n onDeleteConversationClick?: () => void\n onBlockParticipantClick?: () => void\n onReportParticipantClick?: () => void\n showStarButton?: boolean\n}> = ({\n onBack,\n showBackButton,\n renderMessageInputActions,\n onLeaveConversation,\n onBlockParticipant,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n showStarButton = false,\n}) => {\n const { channel } = useChannelStateContext()\n const infoDialogRef = useRef<HTMLDialogElement>(null)\n\n // Get participant info for info dialog\n const participant = React.useMemo(() => {\n const members = Object.values(channel.state.members || {})\n return members.find(\n (member) => member.user?.id && member.user.id !== channel._client.userID\n )\n }, [channel._client.userID, channel.state.members])\n\n // Get follower status label from channel data\n const followerStatusLabel = React.useMemo(() => {\n const channelExtraData = (channel.data ?? {}) as {\n followerStatus?: string\n isFollower?: boolean\n }\n\n // If explicit followerStatus is provided, use it\n if (channelExtraData.followerStatus) {\n return String(channelExtraData.followerStatus)\n }\n // If isFollower is explicitly defined, use it to determine status\n if (channelExtraData.isFollower !== undefined) {\n return channelExtraData.isFollower\n ? 'Subscribed to you'\n : 'Not subscribed'\n }\n // Otherwise, don't show any status\n return undefined\n }, [channel.data])\n\n const handleShowInfo = useCallback(() => {\n infoDialogRef.current?.showModal()\n }, [])\n\n const handleCloseInfo = useCallback(() => {\n infoDialogRef.current?.close()\n }, [])\n\n return (\n <>\n <WithComponents\n overrides={{\n Message: (props: MessageUIComponentProps) => (\n <CustomMessage {...props} />\n ),\n }}\n >\n <Window>\n {/* Custom Channel Header */}\n <div className=\"p-4\">\n <CustomChannelHeader\n onBack={onBack}\n showBackButton={showBackButton}\n onShowInfo={handleShowInfo}\n canShowInfo={Boolean(participant)}\n showStarButton={showStarButton}\n />\n </div>\n\n {/* Message List */}\n <div className=\"flex-1 overflow-hidden relative\">\n <MessageList\n hideDeletedMessages\n hideNewMessageSeparator={false}\n messageActions={undefined}\n />\n </div>\n\n {/* Message Input */}\n <CustomMessageInput\n renderActions={() => renderMessageInputActions?.(channel)}\n />\n </Window>\n </WithComponents>\n\n {/* Channel Info Dialog */}\n <ChannelInfoDialog\n dialogRef={infoDialogRef}\n onClose={handleCloseInfo}\n participant={participant}\n channel={channel}\n followerStatusLabel={followerStatusLabel}\n onLeaveConversation={onLeaveConversation}\n onBlockParticipant={onBlockParticipant}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n />\n </>\n )\n}\n\n/**\n * Channel view component with message list and input\n */\nexport const ChannelView = React.memo<ChannelViewProps>(\n ({\n channel,\n onBack,\n showBackButton = false,\n renderMessageInputActions,\n onLeaveConversation,\n onBlockParticipant,\n className,\n CustomChannelEmptyState = ChannelEmptyState,\n showDeleteConversation = true,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n dmAgentEnabled,\n messageMetadata,\n onMessageSent,\n showStarButton = false,\n }) => {\n // Custom send message handler that:\n // 1. Applies messageMetadata if provided\n // 2. Adds skip_push and silent when DM agent is active\n // 3. Calls onMessageSent callback with full response\n // Read chatbot_paused inside callback to get current value at send time (not stale closure)\n const doSendMessageRequest = useCallback(\n async (\n _channel: ChannelType,\n message: Parameters<ChannelType['sendMessage']>[0],\n options?: Parameters<ChannelType['sendMessage']>[1]\n ) => {\n const agentPaused =\n (channel.data as { chatbot_paused?: boolean })?.chatbot_paused ===\n true\n const shouldSuppressNotifications = dmAgentEnabled && !agentPaused\n\n // Build final message with nested metadata\n const finalMessage = {\n ...message,\n ...(shouldSuppressNotifications && { silent: true }),\n ...(messageMetadata && {\n metadata: {\n ...(message.metadata ?? {}),\n ...messageMetadata,\n },\n }),\n }\n\n // Build final options\n const finalOptions = {\n ...options,\n ...(shouldSuppressNotifications && { skip_push: true }),\n }\n\n const response = await channel.sendMessage(finalMessage, finalOptions)\n\n // Fire callback with full response (includes message.id)\n onMessageSent?.(response)\n\n return response\n },\n [channel, dmAgentEnabled, messageMetadata, onMessageSent]\n )\n\n return (\n <div\n className={classNames(\n 'messaging-channel-view h-full flex flex-col',\n className\n )}\n >\n <Channel\n channel={channel}\n MessageSystem={CustomSystemMessage}\n EmptyStateIndicator={CustomChannelEmptyState}\n LoadingIndicator={LoadingState}\n DateSeparator={CustomDateSeparator}\n doSendMessageRequest={doSendMessageRequest}\n >\n <ChannelViewInner\n onBack={onBack}\n showBackButton={showBackButton}\n renderMessageInputActions={renderMessageInputActions}\n onLeaveConversation={onLeaveConversation}\n onBlockParticipant={onBlockParticipant}\n CustomChannelEmptyState={CustomChannelEmptyState}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n showStarButton={showStarButton}\n />\n </Channel>\n </div>\n )\n }\n)\nChannelView.displayName = 'ChannelView'\n","import { MagnifyingGlassIcon, XIcon } from '@phosphor-icons/react'\nimport { useRef } from 'react'\n\nimport { IconButton } from '../IconButton'\n\ninterface SearchInputProps {\n searchQuery: string\n setSearchQuery: (value: string) => void\n placeholder: string\n}\n\nexport function SearchInput({\n searchQuery,\n setSearchQuery,\n placeholder,\n}: SearchInputProps) {\n const searchInputRef = useRef<HTMLInputElement>(null)\n\n return (\n <div className=\"relative\">\n <MagnifyingGlassIcon\n className=\"pointer-events-none absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-stone\"\n weight=\"bold\"\n />\n\n <input\n ref={searchInputRef}\n type=\"text\"\n placeholder={placeholder}\n value={searchQuery}\n onChange={(e) => setSearchQuery(e.target.value)}\n className=\"w-full pl-10 pr-10 py-3 text-sm border border-sand rounded-xl focus:outline-none focus:ring-2 focus:ring-black focus:border-transparent\"\n />\n\n {searchQuery && (\n <IconButton\n label=\"Clear search\"\n onClick={() => {\n setSearchQuery('')\n searchInputRef.current?.focus()\n }}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 p-1 text-stone hover:text-charcoal\"\n >\n <XIcon className=\"h-4 w-4\" weight=\"bold\" />\n </IconButton>\n )}\n </div>\n )\n}\n","import { ChatCircleDotsIcon, SpinnerGapIcon } from '@phosphor-icons/react'\nimport classNames from 'classnames'\nimport React, { useCallback, useEffect, useState, useRef } from 'react'\n\nimport { useMessagingContext } from '../../providers/MessagingProvider'\nimport type { ParticipantPickerProps, Participant } from '../../types'\nimport { Avatar } from '../Avatar'\nimport { CloseButton } from '../CloseButton'\nimport { SearchInput } from '../SearchInput'\n\n/**\n * Generic participant picker component for starting conversations\n */\nexport const ParticipantPicker: React.FC<ParticipantPickerProps> = ({\n participantSource,\n onSelectParticipant,\n onClose,\n existingParticipantIds = new Set(),\n participantLabel = 'participants',\n searchPlaceholder = 'Search participants...',\n className,\n}) => {\n const { debug } = useMessagingContext()\n const [searchQuery, setSearchQuery] = useState('')\n const [participants, setParticipants] = useState<Participant[]>([])\n const [loading, setLoading] = useState(false)\n const [error, setError] = useState<string | null>(null)\n const [startingChatWithId, setStartingChatWithId] = useState<string | null>(\n null\n )\n\n // Track if we've already loaded participants to prevent repeated loading\n const loadedRef = useRef(false)\n\n // Load participants initially - wait for participantSource to finish loading first\n useEffect(() => {\n // Wait for the participantSource to finish loading before we try to load participants\n if (participantSource.loading) {\n if (debug) {\n console.log(\n '[ParticipantPicker] Waiting for participant source to finish loading...'\n )\n }\n return\n }\n\n if (loadedRef.current) return // Prevent multiple loads\n\n const loadInitialParticipants = async () => {\n if (debug) {\n console.log('[ParticipantPicker] Loading initial participants...')\n }\n setLoading(true)\n setError(null)\n\n try {\n const result = await participantSource.loadParticipants({\n search: '', // Load all participants initially\n limit: 100,\n })\n setParticipants(result.participants)\n loadedRef.current = true // Mark as loaded\n if (debug) {\n console.log(\n '[ParticipantPicker] Participants loaded successfully:',\n result.participants.length\n )\n }\n } catch (err) {\n const errorMessage =\n err instanceof Error ? err.message : 'Failed to load participants'\n setError(errorMessage)\n console.error('[ParticipantPicker] Failed to load participants:', err)\n // Don't mark as loaded on error, allow retry\n } finally {\n setLoading(false)\n }\n }\n\n loadInitialParticipants()\n }, [participantSource.loading, debug]) // Re-run when loading state changes\n\n // Filter participants by search query and existing participants\n const availableParticipants = participants\n .filter((participant) => !existingParticipantIds.has(participant.id))\n .filter((participant) => {\n if (!searchQuery) return true\n const searchLower = searchQuery.toLowerCase()\n return (\n participant.name.toLowerCase().includes(searchLower) ||\n participant.email?.toLowerCase().includes(searchLower) ||\n false\n )\n })\n\n const handleSelectParticipant = useCallback(\n async (participant: Participant) => {\n if (startingChatWithId) return // Prevent multiple clicks\n\n setStartingChatWithId(participant.id)\n try {\n await onSelectParticipant(participant)\n } catch (error) {\n console.error('[ParticipantPicker] Failed to start chat:', error)\n // Reset the loading state on error\n setStartingChatWithId(null)\n }\n // Note: Don't reset startingChatWithId on success because the dialog will close\n },\n [onSelectParticipant, startingChatWithId]\n )\n\n const handleKeyDown = (\n event: React.KeyboardEvent,\n participant: Participant\n ) => {\n if (event.key === 'Enter' || event.key === ' ') {\n event.preventDefault()\n handleSelectParticipant(participant)\n }\n }\n\n return (\n <div className={classNames('flex flex-col h-full', className)}>\n {/* Header */}\n <div className=\"px-4 py-4 border-b border-sand bg-chalk\">\n <div className=\"flex items-center justify-between mb-3\">\n <h2 className=\"text-lg font-semibold text-charcoal\">\n Start a new Conversation\n </h2>\n <CloseButton onClick={onClose} />\n </div>\n\n <p className=\"text-xs text-stone mb-3\">\n Select a {participantLabel.slice(0, -1)} to start messaging (\n {availableParticipants.length} available)\n {participantSource.totalCount !== undefined &&\n ` โ€ข ${participantSource.totalCount} ${participantLabel} total`}\n </p>\n\n <SearchInput\n searchQuery={searchQuery}\n setSearchQuery={setSearchQuery}\n placeholder={searchPlaceholder}\n />\n </div>\n\n {/* Error State */}\n {error && (\n <div className=\"p-4 text-sm text-danger bg-danger-alt\">\n Error loading {participantLabel}: {error}\n </div>\n )}\n\n {/* Participants List */}\n <div className=\"flex-1 overflow-auto\">\n {loading && availableParticipants.length === 0 ? (\n <div className=\"h-32 flex items-center justify-center\">\n <div className=\"flex items-center space-x-2\">\n <div className=\"w-4 h-4 animate-spin rounded-full border-2 border-primary border-t-transparent\"></div>\n <span className=\"text-sm text-stone\">\n Loading {participantLabel}...\n </span>\n </div>\n </div>\n ) : availableParticipants.length === 0 ? (\n <div className=\"p-6 text-center\">\n <div className=\"mx-auto mb-4 flex h-16 w-16 items-center justify-center rounded-full bg-sand\">\n <ChatCircleDotsIcon className=\"h-8 w-8 text-charcoal\" />\n </div>\n <h3 className=\"text-sm font-semibold text-charcoal mb-2\">\n {searchQuery\n ? `No ${participantLabel} found`\n : participants.length > 0\n ? `Already chatting with all ${participantLabel}`\n : `No ${participantLabel} yet`}\n </h3>\n <p className=\"text-xs text-stone\">\n {searchQuery\n ? 'Try a different search term'\n : participants.length > 0\n ? `You have existing conversations with all your ${participantLabel}`\n : `${participantLabel.charAt(0).toUpperCase() + participantLabel.slice(1)} will appear here`}\n </p>\n </div>\n ) : (\n <ul className=\"space-y-0\">\n {availableParticipants.map((participant) => {\n const displayName =\n participant.name || participant.email || participant.id\n const displaySecondary =\n participant.email && participant.name\n ? participant.email\n : participant.phone\n\n return (\n <li key={participant.id}>\n <button\n type=\"button\"\n onClick={() => handleSelectParticipant(participant)}\n onKeyDown={(e) => handleKeyDown(e, participant)}\n className=\"w-full px-4 py-3 hover:bg-sand transition-colors border-b border-sand text-left focus:outline-none focus:ring-2 focus:ring-black\"\n >\n <div className=\"flex items-center justify-between\">\n <div className=\"flex items-center space-x-3 flex-1 min-w-0\">\n {/* Avatar */}\n <Avatar\n id={participant.id}\n name={displayName}\n image={participant.image}\n size={40}\n />\n\n {/* Info */}\n <div className=\"flex-1 min-w-0\">\n <h4 className=\"text-sm font-medium text-charcoal truncate\">\n {displayName}\n </h4>\n {displaySecondary && (\n <p className=\"text-xs text-stone truncate\">\n {displaySecondary}\n </p>\n )}\n </div>\n </div>\n\n {/* Icon */}\n <div className=\"flex-shrink-0\">\n {startingChatWithId === participant.id ? (\n <SpinnerGapIcon className=\"h-5 w-5 text-primary animate-spin\" />\n ) : (\n <ChatCircleDotsIcon className=\"h-5 w-5 text-stone\" />\n )}\n </div>\n </div>\n </button>\n </li>\n )\n })}\n\n {/* Loading indicator */}\n {loading && (\n <li className=\"p-4 flex justify-center\">\n <div className=\"flex items-center space-x-2\">\n <div className=\"w-4 h-4 animate-spin rounded-full border-2 border-primary border-t-transparent\"></div>\n <span className=\"text-sm text-stone\">Loading more...</span>\n </div>\n </li>\n )}\n </ul>\n )}\n </div>\n </div>\n )\n}\n","import React from 'react'\n\nconst ChatBubblesIllustration = ({ className }: { className?: string }) => (\n <svg\n width=\"140\"\n height=\"120\"\n viewBox=\"44 -2 144 126\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n className={className}\n >\n <g clipPath=\"url(#clip0_empty_state)\">\n <path\n d=\"M123.68 82.1932C123.383 103.675 105.839 121 84.2417 121C77.4724 121 71.0986 119.297 65.5327 116.299L52.5873 119.687L53.8036 106.673C48.1776 99.8701 44.7994 91.1453 44.7994 81.6356C44.7994 59.8965 62.4554 42.2754 84.2374 42.2754C89.1328 42.2754 93.8175 43.1633 98.1413 44.789\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M84.2458 86.0364C82.2851 86.0364 80.6957 84.4501 80.6957 82.4933C80.6957 80.5365 82.2851 78.9502 84.2458 78.9502C86.2065 78.9502 87.7959 80.5365 87.7959 82.4933C87.7959 84.4501 86.2065 86.0364 84.2458 86.0364Z\"\n fill=\"white\"\n />\n <path\n d=\"M68.3044 86.0364C66.3437 86.0364 64.7543 84.4501 64.7543 82.4933C64.7543 80.5365 66.3437 78.9502 68.3044 78.9502C70.2651 78.9502 71.8545 80.5365 71.8545 82.4933C71.8545 84.4501 70.2651 86.0364 68.3044 86.0364Z\"\n fill=\"white\"\n />\n <path\n d=\"M100.183 86.0364C98.2226 86.0364 96.6332 84.4501 96.6332 82.4933C96.6332 80.5365 98.2226 78.9502 100.183 78.9502C102.144 78.9502 103.733 80.5365 103.733 82.4933C103.733 84.4501 102.144 86.0364 100.183 86.0364Z\"\n fill=\"white\"\n />\n <g filter=\"url(#filter0_empty_state)\">\n <path\n d=\"M171.522 68.7154C177.443 61.4539 181 52.1488 181 42C181 18.8027 162.421 0 139.5 0C116.579 0 98 18.8027 98 42C98 65.1973 116.579 84 139.5 84C146.622 84 153.328 82.1857 159.184 78.9829L172.801 82.5993L171.522 68.7154Z\"\n fill=\"white\"\n />\n <path\n d=\"M171.522 68.7154C177.443 61.4539 181 52.1488 181 42C181 18.8027 162.421 0 139.5 0C116.579 0 98 18.8027 98 42C98 65.1973 116.579 84 139.5 84C146.622 84 153.328 82.1857 159.184 78.9829L172.801 82.5993L171.522 68.7154Z\"\n stroke=\"#D7D4CE\"\n strokeWidth=\"2\"\n strokeMiterlimit=\"10\"\n />\n </g>\n <path\n d=\"M139.502 45.5431C137.541 45.5431 135.952 43.9568 135.952 42C135.952 40.0432 137.541 38.4569 139.502 38.4569C141.462 38.4569 143.052 40.0432 143.052 42C143.052 43.9568 141.462 45.5431 139.502 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M123.561 45.5431C121.601 45.5431 120.011 43.9568 120.011 42C120.011 40.0432 121.601 38.4569 123.561 38.4569C125.522 38.4569 127.111 40.0432 127.111 42C127.111 43.9568 125.522 45.5431 123.561 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n <path\n d=\"M155.439 45.5431C153.478 45.5431 151.889 43.9568 151.889 42C151.889 40.0432 153.478 38.4569 155.439 38.4569C157.4 38.4569 158.989 40.0432 158.989 42C158.989 43.9568 157.4 45.5431 155.439 45.5431Z\"\n fill=\"#D7D4CE\"\n />\n </g>\n <defs>\n <filter\n id=\"filter0_empty_state\"\n x=\"97\"\n y=\"-1\"\n width=\"89\"\n height=\"90\"\n filterUnits=\"userSpaceOnUse\"\n colorInterpolationFilters=\"sRGB\"\n >\n <feFlood floodOpacity=\"0\" result=\"BackgroundImageFix\" />\n <feColorMatrix\n in=\"SourceAlpha\"\n type=\"matrix\"\n values=\"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0\"\n result=\"hardAlpha\"\n />\n <feOffset dx=\"4\" dy=\"4\" />\n <feComposite in2=\"hardAlpha\" operator=\"out\" />\n <feColorMatrix\n type=\"matrix\"\n values=\"0 0 0 0 0.8428 0 0 0 0 0.830064 0 0 0 0 0.8095 0 0 0 1 0\"\n />\n <feBlend\n mode=\"normal\"\n in2=\"BackgroundImageFix\"\n result=\"effect1_dropShadow\"\n />\n <feBlend\n mode=\"normal\"\n in=\"SourceGraphic\"\n in2=\"effect1_dropShadow\"\n result=\"shape\"\n />\n </filter>\n <clipPath id=\"clip0_empty_state\">\n <rect width=\"233\" height=\"233\" fill=\"white\" />\n </clipPath>\n </defs>\n </svg>\n)\n\n/**\n * Empty state component shown when no channel is selected\n */\nexport const EmptyState = React.memo<{ hasChannels: boolean; channelsLoaded: boolean }>(\n ({ hasChannels, channelsLoaded }) => (\n <div className=\"messaging-empty-state flex items-center justify-center h-full p-8 text-balance\">\n <div className=\"flex flex-col items-center max-w-sm text-center\">\n <ChatBubblesIllustration />\n {channelsLoaded && !hasChannels && (\n <div className=\"mt-8\">\n <h2 className=\"font-medium text-black text-[18px] mb-2\">\n Your inbox is empty\n </h2>\n <p className=\"text-[#676B5F] text-sm mb-6\">\n Share with your followers to start receiving messages\n </p>\n </div>\n )}\n </div>\n </div>\n))\nEmptyState.displayName = 'EmptyState'\n","import React from 'react'\n\ntype ErrorStateProps = {\n message: string\n onBack?: () => void\n}\n\n/**\n * Error state component shown when something goes wrong\n */\nexport const ErrorState = React.memo<ErrorStateProps>(({ message, onBack }) => (\n <div className=\"messaging-error-state flex items-center justify-center h-full p-8\">\n <div className=\"text-center max-w-sm\">\n <div className=\"w-24 h-24 bg-danger-alt/20 rounded-full flex items-center justify-center mx-auto mb-6\">\n <span className=\"text-4xl\">โš ๏ธ</span>\n </div>\n\n <h2 className=\"font-semibold text-charcoal mb-2\">Oops!</h2>\n\n <p className=\"text-stone text-sm mb-6\">{message}</p>\n\n {onBack && (\n <button\n type=\"button\"\n onClick={onBack}\n className=\"inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-white bg-[#7f22fe] hover:bg-primary-alt rounded-lg focus:outline-none focus:ring-2 focus:ring-primary transition-colors\"\n >\n Go Back\n </button>\n )}\n </div>\n </div>\n))\nErrorState.displayName = 'ErrorState'\n","import classNames from 'classnames'\nimport React, { useState, useCallback, useRef, useEffect } from 'react'\nimport type { Channel } from 'stream-chat'\n\nimport { useMessaging } from '../../hooks/useMessaging'\nimport type { MessagingShellProps, Participant } from '../../types'\nimport { ChannelList } from '../ChannelList'\nimport { ChannelView } from '../ChannelView'\nimport { ParticipantPicker } from '../ParticipantPicker'\n\nimport { EmptyState } from './EmptyState'\nimport { ErrorState } from './ErrorState'\nimport { LoadingState } from './LoadingState'\n\n/**\n * Main messaging interface component that combines channel list and channel view\n */\nexport const MessagingShell: React.FC<MessagingShellProps> = ({\n capabilities = {},\n className,\n renderMessageInputActions,\n onChannelSelect,\n onParticipantSelect,\n initialParticipantFilter,\n initialParticipantData,\n CustomChannelEmptyState,\n showChannelList = true,\n filters,\n channelListCustomEmptyStateIndicator,\n onDeleteConversationClick,\n onBlockParticipantClick,\n onReportParticipantClick,\n dmAgentEnabled,\n messageMetadata,\n onMessageSent,\n showStarButton = false,\n renderMessagePreview,\n}) => {\n const {\n service,\n client,\n isConnected,\n isLoading,\n error,\n refreshConnection,\n debug,\n } = useMessaging()\n\n const [selectedChannel, setSelectedChannel] = useState<Channel | null>(null)\n const [hasChannels, setHasChannels] = useState(false)\n const [channelsLoaded, setChannelsLoaded] = useState(false)\n const [_showParticipantPicker, setShowParticipantPicker] = useState(false)\n const [existingParticipantIds, setExistingParticipantIds] = useState<\n Set<string>\n >(new Set())\n const [pickerKey, _setPickerKey] = useState(0) // Key to force remount of ParticipantPicker\n const [directConversationMode, setDirectConversationMode] = useState(false)\n const [directConversationError, setDirectConversationError] = useState<\n string | null\n >(null)\n\n const participantPickerRef = useRef<HTMLDialogElement>(null)\n\n const {\n participantSource,\n participantLabel = 'participants',\n showDeleteConversation = true,\n } = capabilities\n\n // Create default filters and merge with provided filters\n const channelFilters = React.useMemo(() => {\n const userId = client?.userID\n\n // Base filters that should always be present\n const baseFilters = {\n type: 'messaging',\n last_message_at: { $exists: true },\n ...(userId && {\n members: { $in: [userId] },\n hidden: false,\n }),\n }\n\n // Merge provided filters with base filters\n // Provided filters can override base filters if needed\n return {\n ...baseFilters,\n ...filters,\n }\n }, [filters, client?.userID])\n\n // Track if we've already synced channels to prevent repeated API calls\n const syncedRef = useRef<string | null>(null)\n\n // Function to sync channels (extracted for reuse)\n const syncChannels = useCallback(async () => {\n if (!client || !isConnected) return\n\n const userId = client.userID\n if (!userId) return\n\n try {\n if (debug) {\n console.log('[MessagingShell] Syncing channels for user:', userId)\n }\n\n const channels = await client.queryChannels(\n {\n type: 'messaging',\n members: { $in: [userId] },\n },\n {},\n { limit: 100 }\n )\n\n const memberIds = new Set<string>()\n channels.forEach((channel: Channel) => {\n const members = channel.state.members\n Object.values(members).forEach((member) => {\n const memberId = member.user?.id\n if (memberId && memberId !== userId) {\n memberIds.add(memberId)\n }\n })\n })\n\n // Only update if the set contents have changed to prevent re-renders\n setExistingParticipantIds((prev) => {\n if (\n prev.size === memberIds.size &&\n [...prev].every((id) => memberIds.has(id))\n ) {\n return prev\n }\n return memberIds\n })\n setHasChannels(channels.length > 0)\n setChannelsLoaded(true)\n syncedRef.current = userId // Mark as synced for this user\n\n if (debug) {\n console.log('[MessagingShell] Channels synced successfully:', {\n channelCount: channels.length,\n memberCount: memberIds.size,\n })\n }\n } catch (error) {\n console.error('[MessagingShell] Failed to sync channels:', error)\n // Don't mark as synced on error, allow retry\n }\n }, [client, isConnected, debug])\n\n // Sync existing channels to track which participants we can already message\n useEffect(() => {\n if (!client || !isConnected) return\n\n const userId = client.userID\n if (!userId) return\n\n // Prevent repeated sync for the same user\n if (syncedRef.current === userId) return\n\n syncChannels()\n }, [client, isConnected, syncChannels])\n\n // Load initial channel for direct conversation mode\n useEffect(() => {\n if (!initialParticipantFilter || !client || !isConnected) return\n\n const loadInitialChannel = async () => {\n const userId = client.userID\n if (!userId) return\n\n try {\n if (debug) {\n console.log(\n '[MessagingShell] Loading initial conversation with:',\n initialParticipantFilter\n )\n }\n\n const channels = await client.queryChannels(\n {\n type: 'messaging',\n members: { $eq: [userId, initialParticipantFilter] },\n },\n {},\n { limit: 1 }\n )\n\n if (channels.length > 0) {\n setSelectedChannel(channels[0])\n setDirectConversationMode(true)\n setDirectConversationError(null)\n\n // Notify parent component of channel selection\n if (onChannelSelect) {\n onChannelSelect(channels[0])\n }\n\n if (debug) {\n console.log(\n '[MessagingShell] Initial conversation loaded:',\n channels[0].id\n )\n }\n } else {\n // No channel found - try to create one if participant data is provided\n if (initialParticipantData && service) {\n if (debug) {\n console.log(\n '[MessagingShell] No conversation found, creating one for:',\n initialParticipantData\n )\n }\n\n try {\n // Use the existing service method to create the channel\n const channel = await service.startChannelWithParticipant({\n id: initialParticipantData.id,\n name: initialParticipantData.name,\n email: initialParticipantData.email,\n phone: initialParticipantData.phone,\n })\n\n setSelectedChannel(channel)\n setDirectConversationMode(true)\n setDirectConversationError(null)\n\n // Notify parent component of channel selection\n if (onChannelSelect) {\n onChannelSelect(channel)\n }\n\n if (debug) {\n console.log(\n '[MessagingShell] Channel created and loaded:',\n channel.id\n )\n }\n } catch (createErr) {\n console.error(\n '[MessagingShell] Failed to create conversation:',\n createErr\n )\n setDirectConversationError('Failed to create conversation')\n }\n } else {\n // No participant data provided, show error\n setDirectConversationError(\n 'No conversation found with this account'\n )\n\n if (debug) {\n console.log(\n '[MessagingShell] No conversation found for:',\n initialParticipantFilter\n )\n }\n }\n }\n } catch (err) {\n console.error(\n '[MessagingShell] Failed to load initial conversation:',\n err\n )\n setDirectConversationError('Failed to load conversation')\n }\n }\n\n loadInitialChannel()\n }, [\n initialParticipantFilter,\n initialParticipantData,\n client,\n isConnected,\n service,\n debug,\n onChannelSelect,\n ])\n\n const handleChannelSelect = useCallback(\n (channel: Channel) => {\n setSelectedChannel(channel)\n onChannelSelect?.(channel)\n },\n [onChannelSelect]\n )\n\n const handleBackToChannelList = useCallback(() => {\n // In direct conversation mode, don't allow going back to channel list\n // The parent component should handle navigation\n if (directConversationMode) return\n\n setSelectedChannel(null)\n }, [directConversationMode])\n\n const handleSelectParticipant = useCallback(\n async (participant: Participant) => {\n if (!service) return\n\n try {\n if (debug) {\n console.log(\n '[MessagingShell] Starting conversation with:',\n participant.id\n )\n }\n\n const channel = await service.startChannelWithParticipant({\n id: participant.id,\n name: participant.name,\n email: participant.email,\n phone: participant.phone,\n })\n\n // Show the channel\n try {\n await channel.show()\n } catch (error) {\n console.warn('[MessagingShell] Failed to unhide channel:', error)\n }\n\n setSelectedChannel(channel)\n setShowParticipantPicker(false)\n participantPickerRef.current?.close()\n\n onParticipantSelect?.(participant)\n } catch (error) {\n console.error('[MessagingShell] Failed to start conversation:', error)\n }\n },\n [service, onParticipantSelect, debug]\n )\n\n const handleCloseParticipantPicker = useCallback(() => {\n setShowParticipantPicker(false)\n participantPickerRef.current?.close()\n }, [])\n\n const handleDialogBackdropClick = useCallback(\n (e: React.MouseEvent<HTMLDialogElement>) => {\n if (e.target === participantPickerRef.current) {\n handleCloseParticipantPicker()\n }\n },\n [handleCloseParticipantPicker]\n )\n\n const handleLeaveConversation = useCallback(\n async (channel: Channel) => {\n if (debug) {\n console.log('[MessagingShell] Leaving conversation:', channel.id)\n }\n setSelectedChannel(null)\n setDirectConversationMode(false) // Exit direct conversation mode\n\n // Force re-sync to update the existing participants list\n syncedRef.current = null\n await syncChannels()\n },\n [syncChannels, debug]\n )\n\n const handleBlockParticipant = useCallback(\n async (participantId?: string) => {\n if (debug) {\n console.log('[MessagingShell] Blocking participant:', participantId)\n }\n setSelectedChannel(null)\n setDirectConversationMode(false) // Exit direct conversation mode\n\n // Force re-sync to update the existing participants list\n syncedRef.current = null\n await syncChannels()\n },\n [syncChannels, debug]\n )\n\n const isChannelSelected = Boolean(selectedChannel)\n\n // Show loading state\n if (isLoading) {\n return (\n <div className={classNames('h-full', className)}>\n <LoadingState />\n </div>\n )\n }\n\n // Show error state\n if (error) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState message={error} onBack={refreshConnection} />\n </div>\n )\n }\n\n // Show not connected state\n if (!isConnected || !client) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState\n message=\"Not connected to messaging service\"\n onBack={refreshConnection}\n />\n </div>\n )\n }\n\n // Show direct conversation error state\n if (directConversationError) {\n return (\n <div className={classNames('h-full', className)}>\n <ErrorState message={directConversationError} />\n </div>\n )\n }\n\n return (\n <div\n className={classNames(\n 'messaging-shell h-full bg-background-primary overflow-hidden',\n className\n )}\n >\n <div className=\"flex h-full min-h-0\">\n {/* Channel List Sidebar */}\n <div\n className={classNames(\n 'messaging-channel-list-sidebar min-h-0 min-w-0 lg:flex lg:flex-col',\n {\n '!hidden': showChannelList === false || directConversationMode,\n // Hide on mobile when channel selected, show on desktop with consistent wide width\n 'hidden lg:flex lg:flex-1 lg:max-w-2xl':\n showChannelList !== false &&\n !directConversationMode &&\n isChannelSelected,\n // Show on mobile when no channel selected, use same wide width on desktop\n 'flex flex-col w-full lg:flex-1 lg:max-w-2xl':\n showChannelList !== false &&\n !directConversationMode &&\n !isChannelSelected,\n }\n )}\n >\n <ChannelList\n onChannelSelect={handleChannelSelect}\n selectedChannel={selectedChannel || undefined}\n filters={channelFilters}\n customEmptyStateIndicator={channelListCustomEmptyStateIndicator}\n renderMessagePreview={renderMessagePreview}\n />\n </div>\n\n {/* Channel View */}\n <div\n className={classNames(\n 'messaging-conversation-view flex-1 flex-col min-w-0 min-h-0',\n {\n // In direct conversation mode (or waiting for it), always show (full width)\n flex:\n directConversationMode ||\n isChannelSelected ||\n initialParticipantFilter,\n // Normal mode: hide on mobile when no channel selected\n 'hidden lg:flex':\n !directConversationMode &&\n !isChannelSelected &&\n !initialParticipantFilter,\n }\n )}\n >\n {selectedChannel ? (\n <div className=\"flex-1 min-h-0 flex flex-col\">\n <ChannelView\n channel={selectedChannel}\n key={selectedChannel.id}\n onBack={handleBackToChannelList}\n showBackButton={!directConversationMode}\n renderMessageInputActions={renderMessageInputActions}\n onLeaveConversation={handleLeaveConversation}\n onBlockParticipant={handleBlockParticipant}\n CustomChannelEmptyState={CustomChannelEmptyState}\n showDeleteConversation={showDeleteConversation}\n onDeleteConversationClick={onDeleteConversationClick}\n onBlockParticipantClick={onBlockParticipantClick}\n onReportParticipantClick={onReportParticipantClick}\n dmAgentEnabled={dmAgentEnabled}\n messageMetadata={messageMetadata}\n onMessageSent={onMessageSent}\n showStarButton={showStarButton}\n />\n </div>\n ) : initialParticipantFilter ? (\n // Show loading while creating/loading direct conversation channel\n <LoadingState />\n ) : (\n <EmptyState\n hasChannels={hasChannels}\n channelsLoaded={channelsLoaded}\n />\n )}\n </div>\n </div>\n\n {/* Participant Picker Dialog */}\n {participantSource && (\n // eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-noninteractive-element-interactions\n <dialog\n ref={participantPickerRef}\n className=\"mes-dialog\"\n onClick={handleDialogBackdropClick}\n onClose={handleCloseParticipantPicker}\n >\n <div className=\"h-full w-full bg-white shadow-max-elevation-light\">\n <ParticipantPicker\n key={pickerKey}\n participantSource={participantSource}\n onSelectParticipant={handleSelectParticipant}\n onClose={handleCloseParticipantPicker}\n existingParticipantIds={existingParticipantIds}\n participantLabel={participantLabel}\n searchPlaceholder={`Search ${participantLabel}...`}\n />\n </div>\n </dialog>\n )}\n </div>\n )\n}\n","import classNames from 'classnames'\nimport React from 'react'\nexport interface FaqListItemProps {\n question: string\n onClick: () => void\n loading?: boolean\n className?: string\n}\n\nexport const FaqListItem: React.FC<FaqListItemProps> = ({\n question,\n onClick,\n loading = false,\n className,\n}) => {\n return (\n <button\n type=\"button\"\n onClick={onClick}\n disabled={loading}\n style={{ backgroundColor: '#E6E5E3' }}\n className={classNames(\n 'w-full text-center p-4 rounded-xl text-charcoal font-medium transition-colors',\n {\n 'hover:brightness-95 active:brightness-90': !loading,\n 'opacity-50 cursor-not-allowed': loading,\n },\n className\n )}\n >\n {question}\n </button>\n )\n}\n","import React from 'react'\n\nimport { Avatar } from '../Avatar'\n\nimport { FaqListItem } from './FaqListItem'\n\nexport interface Faq {\n id: string\n question: string\n answer: string\n enabled: boolean\n order?: number | null\n}\n\nexport interface FaqListProps {\n faqs: Faq[]\n onFaqClick: (faqId: string) => void\n loadingFaqId?: string | null\n headerText?: string\n className?: string\n avatarImage?: string\n avatarName?: string\n}\n\nexport const FaqList: React.FC<FaqListProps> = ({\n faqs,\n onFaqClick,\n loadingFaqId,\n headerText,\n className,\n avatarImage,\n avatarName,\n}) => {\n const enabledFaqs = faqs\n .filter((faq) => faq.enabled)\n .sort((a, b) => (a.order ?? 0) - (b.order ?? 0))\n\n if (enabledFaqs.length === 0) {\n return null\n }\n\n return (\n <div className={className}>\n <div className=\"flex gap-3 items-end\">\n {/* Avatar at bottom-left, outside grey background */}\n {(avatarImage || avatarName) && (\n <div className=\"flex-none\">\n <Avatar\n id={avatarName || 'account'}\n name={avatarName || 'Account'}\n image={avatarImage}\n size={24}\n shape=\"circle\"\n />\n </div>\n )}\n\n {/* FAQs with grey background */}\n <div\n className=\"flex-1 flex flex-col gap-3 rounded-lg p-4\"\n style={{ backgroundColor: '#F1F0EE' }}\n >\n {headerText && (\n <p className=\"text-md text-charcoal mb-4\">{headerText}</p>\n )}\n {enabledFaqs.map((faq) => (\n <FaqListItem\n key={faq.id}\n question={faq.question}\n onClick={() => onFaqClick(faq.id)}\n loading={loadingFaqId === faq.id}\n />\n ))}\n </div>\n </div>\n </div>\n )\n}\n","import { useState, useEffect, useCallback } from 'react';\n\nimport type { ParticipantSource, Participant } from '../types';\n\n/**\n * Hook for managing participant loading with search and pagination\n */\nexport const useParticipants = (\n participantSource: ParticipantSource,\n options: {\n initialSearch?: string;\n pageSize?: number;\n } = {}\n) => {\n const { initialSearch = '', pageSize = 20 } = options;\n \n const [participants, setParticipants] = useState<Participant[]>([]);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n const [searchQuery, setSearchQuery] = useState(initialSearch);\n const [hasMore, setHasMore] = useState(true);\n const [cursor, setCursor] = useState<string | undefined>();\n\n // Load participants with current search query\n const loadParticipants = useCallback(async (\n reset = false,\n customSearch?: string\n ) => {\n if (loading) return;\n \n const search = customSearch !== undefined ? customSearch : searchQuery;\n \n setLoading(true);\n setError(null);\n\n try {\n const result = await participantSource.loadParticipants({\n search: search || undefined,\n limit: pageSize,\n cursor: reset ? undefined : cursor,\n });\n\n setParticipants(prev => \n reset ? result.participants : [...prev, ...result.participants]\n );\n setHasMore(result.hasMore);\n setCursor(result.nextCursor);\n } catch (err) {\n const errorMessage = err instanceof Error ? err.message : 'Failed to load participants';\n setError(errorMessage);\n console.error('[useParticipants] Load error:', err);\n } finally {\n setLoading(false);\n }\n }, [participantSource, searchQuery, cursor, pageSize, loading]);\n\n // Load more participants (pagination)\n const loadMore = useCallback(() => {\n if (hasMore && !loading) {\n loadParticipants(false);\n }\n }, [hasMore, loading, loadParticipants]);\n\n // Search participants\n const search = useCallback((query: string) => {\n setSearchQuery(query);\n setCursor(undefined);\n loadParticipants(true, query);\n }, [loadParticipants]);\n\n // Refresh participants\n const refresh = useCallback(() => {\n setCursor(undefined);\n loadParticipants(true);\n }, [loadParticipants]);\n\n // Initial load - only run once when participantSource changes\n useEffect(() => {\n loadParticipants(true);\n }, [participantSource.loadParticipants]); // Only depend on the function to avoid loops\n\n return {\n participants,\n loading,\n error,\n searchQuery,\n hasMore,\n totalCount: participantSource.totalCount,\n loadMore,\n search,\n refresh,\n };\n};\n"],"names":["MessagingContext","createContext","useMessagingContext","useContext","MessagingProvider","children","user","serviceConfig","apiKey","capabilities","debug","debugLog","useCallback","message","args","service","setService","useState","client","setClient","isConnected","setIsConnected","isLoading","setIsLoading","error","setError","connectingRef","useRef","prevPropsRef","renderCountRef","useEffect","currentRender","newService","StreamChatService","connectedUserRef","_a","_b","streamClient","err","errorMessage","refreshConnection","contextValue","React","jsx","Chat","useMessaging","getDaysDifference","date1","date2","d1","diffTime","formatRelativeTime","date","now","daysDiff","EMOJIS","hashString","str","hash","i","char","getAvatarEmoji","id","index","Avatar","image","size","className","shape","emoji","fontSizeClass","borderStyle","classNames","SparkleIcon","isTipMessage","isPaidMessage","isChatbotMessage","isTipOrPaidMessage","isTipOnlyMessage","MessageTag","standalone","isTipOrPaid","isChatbot","amountText","label","jsxs","GiftIcon","CustomChannelPreview","channel","selectedChannel","onChannelSelect","unread","renderMessagePreview","isSelected","handleClick","participant","member","participantName","participantImage","_c","lastMessage","_e","_d","lastMessageText","attachment","lastMessageTime","isLastMessageFromChatbot","messagePreview","unreadCount","_f","ChannelList","filters","customEmptyStateIndicator","PreviewComponent","props","StreamChannelList","ActionButton","variant","rest","IconButton","CloseButton","onClick","XIcon","CustomDateSeparator","DateSeparator","CustomMessageWithContext","additionalMessageInputProps","editing","endOfGroup","firstOfGroup","groupedByUser","handleAction","handleOpenThread","handleRetry","highlighted","isMessageAIGenerated","isMyMessage","renderText","threadList","useChatContext","isBounceDialogOpen","setIsBounceDialogOpen","reminder","useMessageReminder","Attachment","DefaultAttachment","EditMessageModal","DefaultEditMessageModal","MessageBlocked","DefaultMessageBlocked","MessageBouncePrompt","DefaultMessageBouncePrompt","MessageDeleted","DefaultMessageDeleted","MessageIsThreadReplyInChannelButtonIndicator","DefaultMessageIsThreadReplyInChannelButtonIndicator","MessageRepliesCountButton","DefaultMessageRepliesCountButton","ReminderNotification","DefaultReminderNotification","StreamedMessageText","DefaultStreamedMessageText","PinIndicator","useComponentContext","hasAttachment","messageHasAttachments","hasReactions","messageHasReactions","isAIGenerated","useMemo","finalAttachments","isDateSeparatorMessage","isMessageBlocked","showReplyCountButton","showIsReplyInChannel","allowRetry","isBounced","isMessageBounced","rootClassName","poll","isTipOnly","Fragment","MessageBounceModal","Poll","MessageText","MessageErrorIcon","MemoizedCustomMessage","areMessageUIPropsEqual","CustomMessage","messageContext","useMessageContext","linkPreviewsManagerStateSelector","state","preview","LinkPreviewsManager","CustomLinkPreviewCard","link","onDismiss","og_scrape_url","title","image_url","e","CustomLinkPreviewList","linkPreviewsManager","useMessageComposer","stateLinkPreviews","useStateStore","handleDismiss","url","linkPreview","CustomMessageInputInner","handleSubmit","useMessageInputContext","hasSendableData","useMessageComposerHasSendableData","SimpleAttachmentSelector","QuotedMessagePreview","AttachmentPreviewList","TextareaComposer","ArrowUpIcon","CustomMessageInput","renderActions","MessageInput","CustomSystemMessage","isDateHidden","MessageTimestamp","ChannelEmptyState","Loading","LoadingState","CustomChannelHeader","onBack","showBackButton","onShowInfo","canShowInfo","showStarButton","useChannelStateContext","isStarred","setIsStarred","handleMemberUpdate","event","handleStarClick","ArrowLeftIcon","StarIcon","DotsThreeIcon","ChannelInfoDialog","dialogRef","onClose","followerStatusLabel","onLeaveConversation","onBlockParticipant","showDeleteConversation","onDeleteConversationClick","onBlockParticipantClick","onReportParticipantClick","isParticipantBlocked","setIsParticipantBlocked","isLeaving","setIsLeaving","isUpdatingBlockStatus","setIsUpdatingBlockStatus","checkIsParticipantBlocked","isBlocked","handleLeaveConversation","actingUserId","handleBlockUser","handleUnblockUser","handleReportUser","participantEmail","participantUsername","participantSecondary","participantId","_g","SpinnerGapIcon","SignOutIcon","ProhibitInsetIcon","FlagIcon","ChannelViewInner","renderMessageInputActions","infoDialogRef","channelExtraData","handleShowInfo","handleCloseInfo","WithComponents","Window","MessageList","ChannelView","CustomChannelEmptyState","dmAgentEnabled","messageMetadata","onMessageSent","doSendMessageRequest","_channel","options","agentPaused","shouldSuppressNotifications","finalMessage","finalOptions","response","Channel","SearchInput","searchQuery","setSearchQuery","placeholder","searchInputRef","MagnifyingGlassIcon","ParticipantPicker","participantSource","onSelectParticipant","existingParticipantIds","participantLabel","searchPlaceholder","participants","setParticipants","loading","setLoading","startingChatWithId","setStartingChatWithId","loadedRef","result","availableParticipants","searchLower","handleSelectParticipant","handleKeyDown","ChatCircleDotsIcon","displayName","displaySecondary","ChatBubblesIllustration","EmptyState","hasChannels","channelsLoaded","ErrorState","MessagingShell","onParticipantSelect","initialParticipantFilter","initialParticipantData","showChannelList","channelListCustomEmptyStateIndicator","setSelectedChannel","setHasChannels","setChannelsLoaded","_showParticipantPicker","setShowParticipantPicker","setExistingParticipantIds","pickerKey","_setPickerKey","directConversationMode","setDirectConversationMode","directConversationError","setDirectConversationError","participantPickerRef","channelFilters","userId","syncedRef","syncChannels","channels","memberIds","members","memberId","prev","createErr","handleChannelSelect","handleBackToChannelList","handleCloseParticipantPicker","handleDialogBackdropClick","handleBlockParticipant","isChannelSelected","FaqListItem","question","FaqList","faqs","onFaqClick","loadingFaqId","headerText","avatarImage","avatarName","enabledFaqs","faq","a","b","useParticipants","initialSearch","pageSize","hasMore","setHasMore","cursor","setCursor","loadParticipants","reset","customSearch","search","loadMore","query","refresh"],"mappings":"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"}