agent-ecosystem/src/renderer/components/team/messages/MessagesPanel.tsx

1166 lines
43 KiB
TypeScript

import { memo, useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
import { Sheet, type SheetRef } from 'react-modal-sheet';
import { Badge } from '@renderer/components/ui/badge';
import { Button } from '@renderer/components/ui/button';
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { useStableTeamMentionMeta } from '@renderer/hooks/useStableTeamMentionMeta';
import { useTeamMessagesExpanded } from '@renderer/hooks/useTeamMessagesExpanded';
import { useTeamMessagesRead } from '@renderer/hooks/useTeamMessagesRead';
import { useStore } from '@renderer/store';
import { selectTeamMessages } from '@renderer/store/slices/teamSlice';
import { filterTeamMessages } from '@renderer/utils/teamMessageFiltering';
import { toMessageKey } from '@renderer/utils/teamMessageKey';
import { shouldExcludeInboxTextFromReplyCandidates } from '@shared/utils/idleNotificationSemantics';
import {
CheckCheck,
ChevronsDownUp,
ChevronsUpDown,
MessageSquare,
PanelBottom,
PanelBottomClose,
PanelBottomOpen,
PanelLeft,
PanelLeftClose,
Search,
X,
} from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { ActivityTimeline } from '../activity/ActivityTimeline';
import { getThoughtGroupKey, groupTimelineItems } from '../activity/LeadThoughtsGroup';
import { MessageExpandDialog } from '../activity/MessageExpandDialog';
import { CollapsibleTeamSection } from '../CollapsibleTeamSection';
import {
getTeamMessagesSidebarUiState,
setTeamMessagesSidebarUiState,
} from '../sidebar/teamSidebarUiState';
import { MessageComposer } from './MessageComposer';
import { MessagesFilterPopover } from './MessagesFilterPopover';
import { StatusBlock } from './StatusBlock';
import type { TimelineItem } from '../activity/LeadThoughtsGroup';
import type { ActionMode } from './ActionModeSelector';
import type { MessagesFilterState } from './MessagesFilterPopover';
import type { TeamMessagesPanelMode } from '@renderer/types/teamMessagesPanelMode';
import type { InboxMessage, ResolvedTeamMember, TaskRef, TeamTaskWithKanban } from '@shared/types';
interface TimeWindow {
start: number;
end: number;
}
const BOTTOM_SHEET_HEADER_HEIGHT = 40;
const BOTTOM_SHEET_COLLAPSED_SNAP_INDEX = 1;
const BOTTOM_SHEET_COMPOSER_SNAP_INDEX = 2;
const BOTTOM_SHEET_FULL_SNAP_INDEX = 4;
interface MessagesPanelProps {
teamName: string;
position: TeamMessagesPanelMode;
onPositionChange: (position: TeamMessagesPanelMode) => void;
mountPoint?: Element | null;
/** Active (non-removed) members. */
members: ResolvedTeamMember[];
/** All team tasks. */
tasks: TeamTaskWithKanban[];
/** Whether the team is alive. */
isTeamAlive?: boolean;
/** Live lead activity status for the current team. */
leadActivity?: string;
/** Latest lead context timestamp for the current team. */
leadContextUpdatedAt?: string;
/** Time window for filtering. */
timeWindow: TimeWindow | null;
/** Current lead session ID. */
currentLeadSessionId?: string;
/** Pending replies tracker (shared with parent for MemberList). */
pendingRepliesByMember: Record<string, number>;
/** Update pending replies tracker. */
onPendingReplyChange: (updater: (prev: Record<string, number>) => Record<string, number>) => void;
/** Callback when a member is clicked in the timeline. */
onMemberClick?: (member: ResolvedTeamMember) => void;
/** Callback when a task is clicked from timeline or status block. */
onTaskClick?: (task: TeamTaskWithKanban) => void;
/** Callback to open create task dialog from a message. */
onCreateTaskFromMessage?: (subject: string, description: string) => void;
/** Callback to open reply dialog for a message. */
onReplyToMessage?: (message: InboxMessage) => void;
/** Callback when "Restart team" is clicked. */
onRestartTeam?: () => void;
/** Callback when a task ID link is clicked. */
onTaskIdClick?: (taskId: string) => void;
}
export const MessagesPanel = memo(function MessagesPanel({
teamName,
position,
onPositionChange,
mountPoint,
members,
tasks,
isTeamAlive,
leadActivity,
leadContextUpdatedAt,
timeWindow,
currentLeadSessionId,
pendingRepliesByMember,
onPendingReplyChange,
onMemberClick,
onTaskClick,
onCreateTaskFromMessage,
onReplyToMessage,
onRestartTeam,
onTaskIdClick,
}: MessagesPanelProps): React.JSX.Element {
const {
sendTeamMessage,
sendCrossTeamMessage,
sendingMessage,
sendMessageError,
lastSendMessageResult,
teams,
openTeamTab,
messages,
messagesState,
loadOlderTeamMessages,
} = useStore(
useShallow((s) => ({
sendTeamMessage: s.sendTeamMessage,
sendCrossTeamMessage: s.sendCrossTeamMessage,
sendingMessage: s.sendingMessage,
sendMessageError: s.sendMessageError,
lastSendMessageResult: s.lastSendMessageResult,
teams: s.teams,
openTeamTab: s.openTeamTab,
messages: selectTeamMessages(s, teamName),
messagesState: teamName ? s.teamMessagesByName[teamName] : undefined,
loadOlderTeamMessages: s.loadOlderTeamMessages,
}))
);
const loadOlderMessages = useCallback(async () => {
if (!messagesState?.hasMore || messagesState.loadingHead || messagesState.loadingOlder) {
return;
}
await loadOlderTeamMessages(teamName);
}, [loadOlderTeamMessages, messagesState, teamName]);
const messagesLoading =
(messagesState?.loadingHead ?? false) || (messagesState?.loadingOlder ?? false);
const loadingOlderMessages = messagesState?.loadingOlder ?? false;
const hasMore = messagesState?.hasMore ?? false;
const effectiveMessages = messages;
const composerTextareaRef = useRef<HTMLTextAreaElement | null>(null);
const sidebarScrollRef = useRef<HTMLDivElement | null>(null);
const bottomSheetRef = useRef<SheetRef>(null);
const bottomSheetStickyTopRef = useRef<HTMLDivElement | null>(null);
const handleExpandContent = useCallback(() => {
// no-op: user is reading expanded content, not composing
}, []);
const initialSidebarStateRef = useRef(getTeamMessagesSidebarUiState(teamName));
const [messagesSearchQuery, setMessagesSearchQuery] = useState(
initialSidebarStateRef.current.messagesSearchQuery
);
const [messagesFilter, setMessagesFilter] = useState<MessagesFilterState>(
initialSidebarStateRef.current.messagesFilter
);
const [messagesFilterOpen, setMessagesFilterOpen] = useState(
initialSidebarStateRef.current.messagesFilterOpen
);
const [messagesCollapsed, setMessagesCollapsed] = useState(
initialSidebarStateRef.current.messagesCollapsed
);
const [messagesSearchBarVisible, setMessagesSearchBarVisible] = useState(
initialSidebarStateRef.current.messagesSearchBarVisible
);
const [expandedItemKey, setExpandedItemKey] = useState<string | null>(
initialSidebarStateRef.current.expandedItemKey
);
const [messagesScrollTop, setMessagesScrollTop] = useState(
initialSidebarStateRef.current.messagesScrollTop
);
const [bottomSheetSnapIndex, setBottomSheetSnapIndex] = useState(
initialSidebarStateRef.current.bottomSheetSnapIndex
);
const [bottomSheetStickyTopHeight, setBottomSheetStickyTopHeight] = useState(196);
const [bottomSheetMountHeight, setBottomSheetMountHeight] = useState(0);
useEffect(() => {
initialSidebarStateRef.current = getTeamMessagesSidebarUiState(teamName);
setMessagesSearchQuery(initialSidebarStateRef.current.messagesSearchQuery);
setMessagesFilter(initialSidebarStateRef.current.messagesFilter);
setMessagesFilterOpen(initialSidebarStateRef.current.messagesFilterOpen);
setMessagesCollapsed(initialSidebarStateRef.current.messagesCollapsed);
setMessagesSearchBarVisible(initialSidebarStateRef.current.messagesSearchBarVisible);
setExpandedItemKey(initialSidebarStateRef.current.expandedItemKey);
setMessagesScrollTop(initialSidebarStateRef.current.messagesScrollTop);
setBottomSheetSnapIndex(initialSidebarStateRef.current.bottomSheetSnapIndex);
}, [teamName]);
useEffect(() => {
setTeamMessagesSidebarUiState(teamName, {
messagesSearchQuery,
messagesFilter,
messagesFilterOpen,
messagesCollapsed,
messagesSearchBarVisible,
expandedItemKey,
messagesScrollTop,
bottomSheetSnapIndex,
});
}, [
teamName,
messagesSearchQuery,
messagesFilter,
messagesFilterOpen,
messagesCollapsed,
messagesSearchBarVisible,
expandedItemKey,
messagesScrollTop,
bottomSheetSnapIndex,
]);
useLayoutEffect(() => {
if (position !== 'sidebar') return;
const el = sidebarScrollRef.current;
if (!el) return;
el.scrollTop = messagesScrollTop;
}, [position, messagesScrollTop]);
useLayoutEffect(() => {
if (position !== 'bottom-sheet' || typeof ResizeObserver === 'undefined') return;
const mountPointElement = mountPoint instanceof HTMLElement ? mountPoint : null;
const observedEntries: [Element | null, (height: number) => void][] = [
[bottomSheetStickyTopRef.current, setBottomSheetStickyTopHeight],
[mountPointElement, setBottomSheetMountHeight],
];
const observers: ResizeObserver[] = [];
for (const [element, setHeight] of observedEntries) {
if (!element) continue;
const updateHeight = (): void => {
const nextHeight = Math.ceil(element.getBoundingClientRect().height);
if (nextHeight > 0) {
setHeight(nextHeight);
}
};
updateHeight();
const observer = new ResizeObserver(() => {
updateHeight();
});
observer.observe(element);
observers.push(observer);
}
return () => {
observers.forEach((observer) => observer.disconnect());
};
}, [position, mountPoint]);
const filteredMessages = useMemo(() => {
return filterTeamMessages(effectiveMessages, {
timeWindow,
filter: messagesFilter,
searchQuery: messagesSearchQuery,
});
}, [effectiveMessages, messagesFilter, messagesSearchQuery, timeWindow]);
const activityTimelineMessages = useMemo(() => {
return filterTeamMessages(effectiveMessages, {
includePassiveIdlePeerSummariesWhenNoiseHidden: true,
timeWindow,
filter: messagesFilter,
searchQuery: messagesSearchQuery,
});
}, [effectiveMessages, messagesFilter, messagesSearchQuery, timeWindow]);
const replyCandidateMessages = useMemo(
() =>
effectiveMessages.filter(
(m) =>
m.messageKind !== 'task_comment_notification' &&
!shouldExcludeInboxTextFromReplyCandidates(typeof m.text === 'string' ? m.text : '')
),
[effectiveMessages]
);
// Resolve the expanded item from filtered messages
const expandedItem = useMemo<TimelineItem | null>(() => {
if (!expandedItemKey) {
return null;
}
if (!expandedItemKey.startsWith('thoughts-')) {
const msg = activityTimelineMessages.find((m) => toMessageKey(m) === expandedItemKey);
return msg ? { type: 'message', message: msg } : null;
}
const allItems = groupTimelineItems(activityTimelineMessages);
return (
allItems.find(
(item) =>
item.type === 'lead-thoughts' && getThoughtGroupKey(item.group) === expandedItemKey
) ?? null
);
}, [expandedItemKey, activityTimelineMessages]);
// Auto-clear stale expanded key
useEffect(() => {
if (expandedItemKey && expandedItem === null) {
setExpandedItemKey(null);
}
}, [expandedItemKey, expandedItem]);
const handleExpandItem = useCallback((key: string) => {
setExpandedItemKey(key);
}, []);
const handleExpandDialogChange = useCallback((open: boolean) => {
if (!open) setExpandedItemKey(null);
}, []);
const { readSet, markRead, markAllRead } = useTeamMessagesRead(teamName);
const { expandedSet, toggle: toggleExpandOverride } = useTeamMessagesExpanded(teamName);
const messagesUnreadCount = useMemo(
() => filteredMessages.filter((m) => !m.read && !readSet.has(toMessageKey(m))).length,
[filteredMessages, readSet]
);
const handleMessageVisible = useCallback(
(message: InboxMessage) => markRead(toMessageKey(message)),
[markRead]
);
const readState = useMemo(() => ({ readSet, getMessageKey: toMessageKey }), [readSet]);
const { teamNames, teamColorByName } = useStableTeamMentionMeta(teams);
const handleMarkAllRead = useCallback(() => {
const keys = filteredMessages
.filter((m) => !m.read && !readSet.has(toMessageKey(m)))
.map((m) => toMessageKey(m));
markAllRead(keys);
}, [filteredMessages, readSet, markAllRead]);
// Auto-clear pending replies when a member actually responds
useEffect(() => {
if (Object.keys(pendingRepliesByMember).length === 0) return;
const next = { ...pendingRepliesByMember };
let changed = false;
for (const [memberName, sentAtMs] of Object.entries(pendingRepliesByMember)) {
const hasReply = replyCandidateMessages.some((m) => {
if (m.from !== memberName) return false;
const ts = Date.parse(m.timestamp);
return Number.isFinite(ts) && ts > sentAtMs;
});
if (hasReply) {
delete next[memberName];
changed = true;
}
}
if (changed) onPendingReplyChange(() => next);
}, [onPendingReplyChange, pendingRepliesByMember, replyCandidateMessages]);
const handleSend = useCallback(
(
member: string,
text: string,
summary?: string,
attachments?: Parameters<typeof sendTeamMessage>[1] extends { attachments?: infer A }
? A
: never,
actionMode?: ActionMode,
taskRefs?: TaskRef[]
) => {
const sentAtMs = Date.now();
onPendingReplyChange((prev) => ({ ...prev, [member]: sentAtMs }));
void sendTeamMessage(teamName, {
member,
text,
summary,
attachments,
actionMode,
taskRefs,
}).catch(() => {
onPendingReplyChange((prev) => {
if (prev[member] !== sentAtMs) return prev;
const next = { ...prev };
delete next[member];
return next;
});
});
},
[teamName, sendTeamMessage, onPendingReplyChange]
);
const handleCrossTeamSend = useCallback(
(
toTeam: string,
text: string,
summary?: string,
actionMode?: ActionMode,
taskRefs?: TaskRef[]
) => {
void sendCrossTeamMessage({
fromTeam: teamName,
fromMember: 'user',
toTeam,
text,
taskRefs,
actionMode,
summary,
});
},
[teamName, sendCrossTeamMessage]
);
const moveToInline = useCallback(() => {
onPositionChange('inline');
}, [onPositionChange]);
const moveToSidebar = useCallback(() => {
onPositionChange('sidebar');
}, [onPositionChange]);
const moveToBottomSheet = useCallback(() => {
setBottomSheetSnapIndex(BOTTOM_SHEET_COMPOSER_SNAP_INDEX);
onPositionChange('bottom-sheet');
}, [onPositionChange]);
const snapBottomSheetTo = useCallback((snapIndex: number) => {
setBottomSheetSnapIndex(snapIndex);
bottomSheetRef.current?.snapTo(snapIndex);
}, []);
const toggleBottomSheetExpansion = useCallback(() => {
if (bottomSheetSnapIndex === BOTTOM_SHEET_COLLAPSED_SNAP_INDEX) {
snapBottomSheetTo(BOTTOM_SHEET_COMPOSER_SNAP_INDEX);
return;
}
snapBottomSheetTo(BOTTOM_SHEET_COLLAPSED_SNAP_INDEX);
}, [bottomSheetSnapIndex, snapBottomSheetTo]);
const bottomSheetSnapPoints = useMemo(() => {
const maxOpenHeight =
bottomSheetMountHeight > 0
? Math.max(bottomSheetMountHeight - 1, 96)
: Number.POSITIVE_INFINITY;
const collapsedHeight = Math.min(BOTTOM_SHEET_HEADER_HEIGHT, maxOpenHeight);
const composerHeight = Math.min(
Math.max(collapsedHeight + bottomSheetStickyTopHeight, collapsedHeight + 120),
maxOpenHeight
);
const centeredHeight = Math.min(
Math.max(
bottomSheetMountHeight > 0 ? Math.round(bottomSheetMountHeight * 0.58) : 520,
composerHeight + 140
),
maxOpenHeight
);
return [0, collapsedHeight, composerHeight, centeredHeight, 1];
}, [bottomSheetMountHeight, bottomSheetStickyTopHeight]);
const normalizedBottomSheetSnapIndex = useMemo(() => {
return Math.min(
Math.max(bottomSheetSnapIndex, BOTTOM_SHEET_COLLAPSED_SNAP_INDEX),
BOTTOM_SHEET_FULL_SNAP_INDEX
);
}, [bottomSheetSnapIndex]);
// ---- Shared content (used in both modes) ----
const searchAndFilterControls = (
<div className="flex items-center gap-2">
<div className="flex min-w-0 flex-1 items-center gap-1.5 rounded-md border border-[var(--color-border)] bg-transparent px-2 py-1">
<Search size={12} className="shrink-0 text-[var(--color-text-muted)]" />
<input
type="text"
placeholder="Search..."
value={messagesSearchQuery}
onChange={(e) => setMessagesSearchQuery(e.target.value)}
onPointerDown={(e) => e.stopPropagation()}
onClick={(e) => e.stopPropagation()}
className="min-w-0 flex-1 bg-transparent text-xs text-[var(--color-text)] placeholder:text-[var(--color-text-muted)] focus:outline-none"
/>
{messagesSearchQuery && (
<button
type="button"
className="shrink-0 rounded p-0.5 text-[var(--color-text-muted)] transition-colors hover:bg-[var(--color-surface-raised)] hover:text-[var(--color-text)]"
onClick={() => setMessagesSearchQuery('')}
>
<X size={14} />
</button>
)}
</div>
<MessagesFilterPopover
teamName={teamName}
members={members}
filter={messagesFilter}
messages={effectiveMessages}
open={messagesFilterOpen}
onOpenChange={setMessagesFilterOpen}
onApply={setMessagesFilter}
/>
</div>
);
const searchAndFilterBar = (
<div className="flex items-center gap-2">
{searchAndFilterControls}
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="pointer-events-auto size-7 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={(e) => {
e.stopPropagation();
setMessagesCollapsed((v) => !v);
}}
>
{messagesCollapsed ? <ChevronsUpDown size={14} /> : <ChevronsDownUp size={14} />}
</Button>
</TooltipTrigger>
<TooltipContent side="bottom">
{messagesCollapsed ? 'Expand all messages' : 'Collapse all messages'}
</TooltipContent>
</Tooltip>
</div>
);
const messagesContent = (
<div className="pb-14">
<MessageComposer
teamName={teamName}
members={members}
isTeamAlive={isTeamAlive}
sending={sendingMessage}
sendError={sendMessageError}
lastResult={lastSendMessageResult}
textareaRef={composerTextareaRef}
onSend={handleSend}
onCrossTeamSend={handleCrossTeamSend}
/>
<StatusBlock
members={members}
tasks={tasks}
messages={effectiveMessages}
pendingRepliesByMember={pendingRepliesByMember}
layout="flow"
position="inline"
onMemberClick={onMemberClick}
onTaskClick={onTaskClick}
/>
<ActivityTimeline
messages={activityTimelineMessages}
teamName={teamName}
members={members}
readState={readState}
allCollapsed={messagesCollapsed}
expandOverrides={expandedSet}
onToggleExpandOverride={toggleExpandOverride}
currentLeadSessionId={currentLeadSessionId}
isTeamAlive={isTeamAlive}
leadActivity={leadActivity}
leadContextUpdatedAt={leadContextUpdatedAt}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
onMemberClick={onMemberClick}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMessageVisible={handleMessageVisible}
onRestartTeam={onRestartTeam}
onTaskIdClick={onTaskIdClick}
onExpandItem={handleExpandItem}
onExpandContent={handleExpandContent}
/>
{hasMore && (
<div className="flex justify-center py-2">
<Button
variant="ghost"
size="sm"
className="text-xs text-text-muted"
aria-busy={loadingOlderMessages}
disabled={loadingOlderMessages}
onClick={() => void loadOlderMessages()}
>
Load older messages
</Button>
</div>
)}
<MessageExpandDialog
expandedItem={expandedItem}
open={expandedItemKey !== null}
onOpenChange={handleExpandDialogChange}
teamName={teamName}
members={members}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMemberClick={onMemberClick}
onTaskIdClick={onTaskIdClick}
onRestartTeam={onRestartTeam}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
/>
</div>
);
// ---- Sidebar mode ----
if (position === 'sidebar') {
return (
<div className="flex size-full flex-col overflow-hidden bg-[var(--color-surface-sidebar)]">
{/* Header */}
<div className="flex shrink-0 items-center gap-2 border-b border-[var(--color-border)] bg-[var(--color-surface-sidebar)] px-3 py-2">
<MessageSquare size={14} className="shrink-0 text-[var(--color-text-muted)]" />
<span className="text-sm font-medium text-[var(--color-text)]">Messages</span>
{filteredMessages.length > 0 && (
<Badge
variant="secondary"
className="px-1.5 py-0.5 text-[10px] font-normal leading-none"
>
{filteredMessages.length}
</Badge>
)}
{messagesUnreadCount > 0 && (
<Tooltip>
<TooltipTrigger asChild>
<Badge
variant="secondary"
className="bg-blue-500/20 px-1.5 py-0.5 text-[10px] font-normal leading-none text-blue-600 dark:text-blue-400"
>
{messagesUnreadCount} new
</Badge>
</TooltipTrigger>
<TooltipContent side="bottom">{messagesUnreadCount} unread</TooltipContent>
</Tooltip>
)}
{messagesUnreadCount > 0 && (
<Tooltip>
<TooltipTrigger asChild>
<button
type="button"
className="flex items-center gap-1 rounded-md px-1.5 py-1 text-[11px] text-blue-400 transition-colors hover:bg-blue-500/10"
onClick={handleMarkAllRead}
>
<CheckCheck size={12} />
</button>
</TooltipTrigger>
<TooltipContent side="bottom">Mark all as read</TooltipContent>
</Tooltip>
)}
<div className="ml-auto flex items-center gap-1">
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-7 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={() => setMessagesCollapsed((v) => !v)}
aria-label={messagesCollapsed ? 'Expand all messages' : 'Collapse all messages'}
>
{messagesCollapsed ? <ChevronsUpDown size={14} /> : <ChevronsDownUp size={14} />}
</Button>
</TooltipTrigger>
<TooltipContent side="bottom">
{messagesCollapsed ? 'Expand all messages' : 'Collapse all messages'}
</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-7 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={() => setMessagesSearchBarVisible((v) => !v)}
aria-label={
messagesSearchBarVisible ? 'Hide message search' : 'Show message search'
}
>
{messagesSearchBarVisible ? <X size={14} /> : <Search size={14} />}
</Button>
</TooltipTrigger>
<TooltipContent side="bottom">
{messagesSearchBarVisible ? 'Hide search' : 'Search messages'}
</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-7 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={moveToInline}
aria-label="Move messages to inline panel"
>
<PanelLeftClose size={14} />
</Button>
</TooltipTrigger>
<TooltipContent side="bottom">Move to inline</TooltipContent>
</Tooltip>
</div>
</div>
{/* Search & filter bar (toggleable) */}
{messagesSearchBarVisible && (
<div className="shrink-0 border-b border-[var(--color-border)] px-3 py-1.5">
{searchAndFilterControls}
</div>
)}
{/* Scrollable content */}
<div
ref={sidebarScrollRef}
className="min-h-0 min-w-0 flex-1 overflow-y-auto overflow-x-hidden pb-14 pr-3 pt-2"
onScroll={(e) => setMessagesScrollTop(e.currentTarget.scrollTop)}
>
<div className="pl-3">
<MessageComposer
teamName={teamName}
members={members}
isTeamAlive={isTeamAlive}
sending={sendingMessage}
sendError={sendMessageError}
lastResult={lastSendMessageResult}
textareaRef={composerTextareaRef}
onSend={handleSend}
onCrossTeamSend={handleCrossTeamSend}
/>
<StatusBlock
members={members}
tasks={tasks}
messages={effectiveMessages}
pendingRepliesByMember={pendingRepliesByMember}
layout="flow"
position="sidebar"
onMemberClick={onMemberClick}
onTaskClick={onTaskClick}
/>{' '}
</div>
<ActivityTimeline
messages={activityTimelineMessages}
teamName={teamName}
members={members}
readState={readState}
allCollapsed={messagesCollapsed}
expandOverrides={expandedSet}
onToggleExpandOverride={toggleExpandOverride}
currentLeadSessionId={currentLeadSessionId}
isTeamAlive={isTeamAlive}
leadActivity={leadActivity}
leadContextUpdatedAt={leadContextUpdatedAt}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
onMemberClick={onMemberClick}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMessageVisible={handleMessageVisible}
onRestartTeam={onRestartTeam}
onTaskIdClick={onTaskIdClick}
onExpandItem={handleExpandItem}
onExpandContent={handleExpandContent}
/>
{hasMore && (
<div className="flex justify-center py-2">
<Button
variant="ghost"
size="sm"
className="text-xs text-text-muted"
aria-busy={loadingOlderMessages}
disabled={loadingOlderMessages}
onClick={() => void loadOlderMessages()}
>
Load older messages
</Button>
</div>
)}
<MessageExpandDialog
expandedItem={expandedItem}
open={expandedItemKey !== null}
onOpenChange={handleExpandDialogChange}
teamName={teamName}
members={members}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMemberClick={onMemberClick}
onTaskIdClick={onTaskIdClick}
onRestartTeam={onRestartTeam}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
/>
</div>
</div>
);
}
if (position === 'bottom-sheet') {
if (!mountPoint) {
return <div className="hidden" aria-hidden="true" />;
}
const isBottomSheetCollapsed =
normalizedBottomSheetSnapIndex === BOTTOM_SHEET_COLLAPSED_SNAP_INDEX;
return (
<Sheet
ref={bottomSheetRef}
isOpen
onClose={moveToInline}
mountPoint={mountPoint}
avoidKeyboard={false}
detent="full"
snapPoints={bottomSheetSnapPoints}
initialSnap={normalizedBottomSheetSnapIndex}
onSnap={setBottomSheetSnapIndex}
disableDismiss
disableScrollLocking
style={{ zIndex: 30 }}
className="!pointer-events-none !absolute !inset-0"
unstyled
>
<Sheet.Container
unstyled
className="flex max-h-full w-full flex-col overflow-hidden rounded-t-[20px] border border-[var(--color-border)] bg-[var(--color-surface-sidebar)] shadow-[0_-18px_48px_rgba(0,0,0,0.35)]"
>
<Sheet.Header
unstyled
className="shrink-0 cursor-grab select-none border-b border-[var(--color-border)] bg-[var(--color-surface-sidebar)] active:cursor-grabbing"
>
<div className="relative h-10 px-3">
<div className="pointer-events-none absolute inset-x-0 top-1 flex justify-center">
<Sheet.DragIndicator
className="!h-1 !w-9 cursor-grab !rounded-full active:cursor-grabbing"
style={{
backgroundColor: 'color-mix(in srgb, var(--color-text-muted) 45%, transparent)',
}}
/>
</div>
<div className="flex h-full items-center gap-1.5">
<MessageSquare size={13} className="shrink-0 text-[var(--color-text-muted)]" />
<span className="text-[13px] font-medium text-[var(--color-text)]">Messages</span>
{filteredMessages.length > 0 && (
<Badge
variant="secondary"
className="px-1 py-0 text-[9px] font-normal leading-none"
>
{filteredMessages.length}
</Badge>
)}
{messagesUnreadCount > 0 && (
<Tooltip>
<TooltipTrigger asChild>
<Badge
variant="secondary"
className="bg-blue-500/20 px-1 py-0 text-[9px] font-normal leading-none text-blue-600 dark:text-blue-400"
>
{messagesUnreadCount} new
</Badge>
</TooltipTrigger>
<TooltipContent side="top">{messagesUnreadCount} unread</TooltipContent>
</Tooltip>
)}
<div
className="ml-auto flex items-center gap-1"
onPointerDown={(e) => e.stopPropagation()}
>
{messagesUnreadCount > 0 && (
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-blue-400 hover:bg-blue-500/10 hover:text-blue-300"
onClick={handleMarkAllRead}
aria-label="Mark all messages as read"
>
<CheckCheck size={13} />
</Button>
</TooltipTrigger>
<TooltipContent side="top">Mark all as read</TooltipContent>
</Tooltip>
)}
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={() => setMessagesCollapsed((value) => !value)}
aria-label={
messagesCollapsed ? 'Expand all messages' : 'Collapse all messages'
}
>
{messagesCollapsed ? (
<ChevronsUpDown size={14} />
) : (
<ChevronsDownUp size={14} />
)}
</Button>
</TooltipTrigger>
<TooltipContent side="top">
{messagesCollapsed ? 'Expand all messages' : 'Collapse all messages'}
</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={() => setMessagesSearchBarVisible((value) => !value)}
aria-label={
messagesSearchBarVisible ? 'Hide message search' : 'Show message search'
}
>
{messagesSearchBarVisible ? <X size={14} /> : <Search size={14} />}
</Button>
</TooltipTrigger>
<TooltipContent side="top">
{messagesSearchBarVisible ? 'Hide search' : 'Search messages'}
</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={toggleBottomSheetExpansion}
aria-label={
isBottomSheetCollapsed
? 'Expand messages bottom sheet'
: 'Collapse messages bottom sheet'
}
>
{isBottomSheetCollapsed ? (
<PanelBottomOpen size={14} />
) : (
<PanelBottomClose size={14} />
)}
</Button>
</TooltipTrigger>
<TooltipContent side="top">
{isBottomSheetCollapsed ? 'Expand sheet' : 'Collapse sheet'}
</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={moveToInline}
aria-label="Move messages to inline panel"
>
<PanelBottom size={14} />
</Button>
</TooltipTrigger>
<TooltipContent side="top">Move to inline</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="size-[22px] p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={moveToSidebar}
aria-label="Move messages to sidebar"
>
<PanelLeft size={14} />
</Button>
</TooltipTrigger>
<TooltipContent side="top">Move to sidebar</TooltipContent>
</Tooltip>
</div>
</div>
</div>
</Sheet.Header>
{!isBottomSheetCollapsed && (
<Sheet.Content
className="min-h-0 bg-[var(--color-surface-sidebar)]"
scrollClassName="flex min-h-full flex-col"
disableDrag={(state) => state.scrollPosition !== 'top'}
>
<div
ref={bottomSheetStickyTopRef}
className="sticky top-0 z-[1] shrink-0 border-b border-[var(--color-border)] backdrop-blur"
style={{
backgroundColor: 'var(--color-surface-sidebar)',
}}
>
{messagesSearchBarVisible && (
<div className="border-b border-[var(--color-border)] px-3 py-2">
{searchAndFilterControls}
</div>
)}
<div className="p-3">
<MessageComposer
teamName={teamName}
layout="compact"
members={members}
isTeamAlive={isTeamAlive}
sending={sendingMessage}
sendError={sendMessageError}
lastResult={lastSendMessageResult}
textareaRef={composerTextareaRef}
onSend={handleSend}
onCrossTeamSend={handleCrossTeamSend}
/>
</div>
</div>
<div className="shrink-0 px-3 pt-2">
<StatusBlock
members={members}
tasks={tasks}
messages={effectiveMessages}
pendingRepliesByMember={pendingRepliesByMember}
layout="flow"
position="inline"
onMemberClick={onMemberClick}
onTaskClick={onTaskClick}
/>
</div>
<div className="flex-1 px-3 pb-4 pt-2">
<ActivityTimeline
messages={activityTimelineMessages}
teamName={teamName}
members={members}
readState={readState}
allCollapsed={messagesCollapsed}
expandOverrides={expandedSet}
onToggleExpandOverride={toggleExpandOverride}
currentLeadSessionId={currentLeadSessionId}
isTeamAlive={isTeamAlive}
leadActivity={leadActivity}
leadContextUpdatedAt={leadContextUpdatedAt}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
onMemberClick={onMemberClick}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMessageVisible={handleMessageVisible}
onRestartTeam={onRestartTeam}
onTaskIdClick={onTaskIdClick}
onExpandItem={handleExpandItem}
onExpandContent={handleExpandContent}
/>
{hasMore && (
<div className="flex justify-center py-2">
<Button
variant="ghost"
size="sm"
className="text-xs text-text-muted"
aria-busy={loadingOlderMessages}
disabled={loadingOlderMessages}
onClick={() => void loadOlderMessages()}
>
Load older messages
</Button>
</div>
)}
</div>
<MessageExpandDialog
expandedItem={expandedItem}
open={expandedItemKey !== null}
onOpenChange={handleExpandDialogChange}
teamName={teamName}
members={members}
onCreateTaskFromMessage={onCreateTaskFromMessage}
onReplyToMessage={onReplyToMessage}
onMemberClick={onMemberClick}
onTaskIdClick={onTaskIdClick}
onRestartTeam={onRestartTeam}
teamNames={teamNames}
teamColorByName={teamColorByName}
onTeamClick={openTeamTab}
/>
</Sheet.Content>
)}
</Sheet.Container>
</Sheet>
);
}
// ---- Inline mode (wrapped in CollapsibleTeamSection) ----
return (
<CollapsibleTeamSection
sectionId="messages"
title="Messages"
icon={<MessageSquare size={14} />}
badge={filteredMessages.length}
secondaryBadge={
filteredMessages.length > 0 && messagesUnreadCount > 0 ? messagesUnreadCount : undefined
}
afterBadge={
messagesUnreadCount > 0 ? (
<Tooltip>
<TooltipTrigger asChild>
<button
type="button"
className="pointer-events-auto flex items-center gap-1 rounded-md px-1.5 py-1 text-[11px] text-blue-400 transition-colors hover:bg-blue-500/10"
onClick={(e) => {
e.stopPropagation();
handleMarkAllRead();
}}
>
<CheckCheck size={12} />
</button>
</TooltipTrigger>
<TooltipContent side="bottom">Mark all as read</TooltipContent>
</Tooltip>
) : undefined
}
headerExtra={
<div className="flex items-center gap-1">
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="pointer-events-auto size-6 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={(e) => {
e.stopPropagation();
moveToBottomSheet();
}}
aria-label="Move messages to bottom sheet"
>
<PanelBottom size={14} />
</Button>
</TooltipTrigger>
<TooltipContent side="top">Move to bottom sheet</TooltipContent>
</Tooltip>
<Tooltip>
<TooltipTrigger asChild>
<Button
variant="ghost"
size="sm"
className="pointer-events-auto size-6 p-0 text-[var(--color-text-muted)] hover:text-[var(--color-text-secondary)]"
onClick={(e) => {
e.stopPropagation();
moveToSidebar();
}}
aria-label="Move messages to sidebar"
>
<PanelLeft size={14} />
</Button>
</TooltipTrigger>
<TooltipContent side="top">Move to sidebar</TooltipContent>
</Tooltip>
</div>
}
defaultOpen
action={<div className="flex items-center gap-2 px-2">{searchAndFilterBar}</div>}
>
{messagesContent}
</CollapsibleTeamSection>
);
});