fix(team): polish task log rendering
This commit is contained in:
parent
f7e14e2b9a
commit
8f1ee5603c
2 changed files with 163 additions and 21 deletions
|
|
@ -4,7 +4,10 @@ import { DisplayItemList } from '@renderer/components/chat/DisplayItemList';
|
|||
import { LastOutputDisplay } from '@renderer/components/chat/LastOutputDisplay';
|
||||
import { SystemChatGroup } from '@renderer/components/chat/SystemChatGroup';
|
||||
import { MarkdownViewer } from '@renderer/components/chat/viewers/MarkdownViewer';
|
||||
import { MemberBadge } from '@renderer/components/team/MemberBadge';
|
||||
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
|
||||
import { getTeamColorSet, getThemedBorder } from '@renderer/constants/teamColors';
|
||||
import { useTheme } from '@renderer/hooks/useTheme';
|
||||
import { enhanceAIGroup } from '@renderer/utils/aiGroupEnhancer';
|
||||
import { transformChunksToConversation } from '@renderer/utils/groupTransformer';
|
||||
import { extractAgentBlockContents, stripAgentBlocks } from '@shared/constants/agentBlocks';
|
||||
|
|
@ -17,6 +20,8 @@ import type { AIGroup, UserGroup } from '@renderer/types/groups';
|
|||
interface MemberExecutionLogProps {
|
||||
chunks: EnhancedChunk[];
|
||||
memberName?: string;
|
||||
memberColor?: string;
|
||||
teamName?: string;
|
||||
}
|
||||
|
||||
type ExpandedItemIdsByGroup = Map<string, Set<string>>;
|
||||
|
|
@ -24,6 +29,8 @@ type ExpandedItemIdsByGroup = Map<string, Set<string>>;
|
|||
export const MemberExecutionLog = ({
|
||||
chunks,
|
||||
memberName,
|
||||
memberColor,
|
||||
teamName,
|
||||
}: MemberExecutionLogProps): React.JSX.Element => {
|
||||
const conversation = useMemo(() => transformChunksToConversation(chunks, [], false), [chunks]);
|
||||
|
||||
|
|
@ -60,6 +67,8 @@ export const MemberExecutionLog = ({
|
|||
key={item.group.id}
|
||||
group={item.group}
|
||||
memberName={memberName}
|
||||
memberColor={memberColor}
|
||||
teamName={teamName}
|
||||
expanded={!collapsedGroupIds.has(item.group.id)}
|
||||
expandedItemIds={expandedItemIdsByGroup.get(item.group.id) ?? new Set()}
|
||||
onToggleExpanded={() => {
|
||||
|
|
@ -151,6 +160,8 @@ const UserLogItem = ({ group }: { group: UserGroup }): React.JSX.Element => {
|
|||
interface AIExecutionGroupProps {
|
||||
group: AIGroup;
|
||||
memberName?: string;
|
||||
memberColor?: string;
|
||||
teamName?: string;
|
||||
expanded: boolean;
|
||||
expandedItemIds: Set<string>;
|
||||
onToggleExpanded: () => void;
|
||||
|
|
@ -160,11 +171,14 @@ interface AIExecutionGroupProps {
|
|||
const AIExecutionGroup = ({
|
||||
group,
|
||||
memberName,
|
||||
memberColor,
|
||||
teamName,
|
||||
expanded,
|
||||
expandedItemIds,
|
||||
onToggleExpanded,
|
||||
onToggleItem,
|
||||
}: AIExecutionGroupProps): React.JSX.Element => {
|
||||
const { isLight } = useTheme();
|
||||
const enhanced = useMemo(() => {
|
||||
if (!memberName) {
|
||||
return enhanceAIGroup(group);
|
||||
|
|
@ -175,13 +189,18 @@ const AIExecutionGroup = ({
|
|||
);
|
||||
return enhanceAIGroup({ ...group, processes: filteredProcesses });
|
||||
}, [group, memberName]);
|
||||
const groupLabel = memberName?.trim() ? `${memberName.trim()} turn` : 'Agent turn';
|
||||
const normalizedMemberName = memberName?.trim();
|
||||
const groupLabel = normalizedMemberName ? `${normalizedMemberName} turn` : 'Agent turn';
|
||||
const hasToggleContent = enhanced.displayItems.length > 0;
|
||||
const visibleLastOutput =
|
||||
enhanced.lastOutput?.type === 'tool_result' && hasToggleContent ? null : enhanced.lastOutput;
|
||||
const groupColors = getTeamColorSet(memberColor ?? '');
|
||||
const borderColor = normalizedMemberName
|
||||
? getThemedBorder(groupColors, isLight)
|
||||
: 'var(--chat-ai-border)';
|
||||
|
||||
return (
|
||||
<div className="space-y-3 border-l-2 pl-3" style={{ borderColor: 'var(--chat-ai-border)' }}>
|
||||
<div className="space-y-3 border-l-2 pl-3" style={{ borderColor }}>
|
||||
{hasToggleContent ? (
|
||||
<Tooltip>
|
||||
<TooltipTrigger asChild>
|
||||
|
|
@ -191,10 +210,27 @@ const AIExecutionGroup = ({
|
|||
onClick={onToggleExpanded}
|
||||
aria-expanded={expanded}
|
||||
>
|
||||
<Bot className="size-4 shrink-0 text-[var(--color-text-secondary)]" />
|
||||
<span className="shrink-0 text-xs font-semibold text-[var(--color-text-secondary)]">
|
||||
{groupLabel}
|
||||
</span>
|
||||
{normalizedMemberName ? (
|
||||
<>
|
||||
<MemberBadge
|
||||
name={normalizedMemberName}
|
||||
color={memberColor}
|
||||
teamName={teamName}
|
||||
size="sm"
|
||||
disableHoverCard
|
||||
/>
|
||||
<span className="shrink-0 text-xs font-semibold text-[var(--color-text-secondary)]">
|
||||
turn
|
||||
</span>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<Bot className="size-4 shrink-0 text-[var(--color-text-secondary)]" />
|
||||
<span className="shrink-0 text-xs font-semibold text-[var(--color-text-secondary)]">
|
||||
{groupLabel}
|
||||
</span>
|
||||
</>
|
||||
)}
|
||||
<span className="min-w-0 flex-1 truncate text-xs text-[var(--color-text-muted)]">
|
||||
{enhanced.itemsSummary}
|
||||
</span>
|
||||
|
|
|
|||
|
|
@ -1,12 +1,25 @@
|
|||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||
|
||||
import { api } from '@renderer/api';
|
||||
import { MemberBadge } from '@renderer/components/team/MemberBadge';
|
||||
import { MemberExecutionLog } from '@renderer/components/team/members/MemberExecutionLog';
|
||||
import {
|
||||
getTeamColorSet,
|
||||
getThemedBadge,
|
||||
getThemedBorder,
|
||||
getThemedText,
|
||||
} from '@renderer/constants/teamColors';
|
||||
import { useTheme } from '@renderer/hooks/useTheme';
|
||||
import { useStore } from '@renderer/store';
|
||||
import { selectResolvedMembersForTeamName } from '@renderer/store/slices/teamSlice';
|
||||
import { asEnhancedChunkArray } from '@renderer/types/data';
|
||||
import { buildMemberColorMap } from '@renderer/utils/memberHelpers';
|
||||
import { isLeadMember } from '@shared/utils/leadDetection';
|
||||
import { AlertCircle, Clock, FileText, Loader2 } from 'lucide-react';
|
||||
|
||||
import type {
|
||||
BoardTaskLogActor,
|
||||
ResolvedTeamMember,
|
||||
BoardTaskLogSegment,
|
||||
BoardTaskLogStreamResponse,
|
||||
} from '@shared/types';
|
||||
|
|
@ -87,12 +100,51 @@ function describeStreamSource(stream: BoardTaskLogStreamResponse | null): string
|
|||
return 'Task-scoped transcript logs rendered with the same execution-log components used in Logs.';
|
||||
}
|
||||
|
||||
interface ParticipantVisual {
|
||||
name: string;
|
||||
color?: string;
|
||||
}
|
||||
|
||||
function buildParticipantVisualMap(
|
||||
stream: BoardTaskLogStreamResponse | null,
|
||||
members: readonly ResolvedTeamMember[],
|
||||
memberColorMap: ReadonlyMap<string, string>
|
||||
): Map<string, ParticipantVisual> {
|
||||
const visuals = new Map<string, ParticipantVisual>();
|
||||
const leadMember = members.find((member) => isLeadMember(member));
|
||||
|
||||
for (const participant of stream?.participants ?? []) {
|
||||
const matchingSegment = stream?.segments.find(
|
||||
(segment) => segment.participantKey === participant.key
|
||||
);
|
||||
const name =
|
||||
matchingSegment?.actor.memberName ??
|
||||
(participant.isLead ? leadMember?.name : undefined) ??
|
||||
participant.label;
|
||||
|
||||
visuals.set(participant.key, {
|
||||
name,
|
||||
color: memberColorMap.get(name) ?? memberColorMap.get(participant.label),
|
||||
});
|
||||
}
|
||||
|
||||
for (const segment of stream?.segments ?? []) {
|
||||
if (visuals.has(segment.participantKey)) {
|
||||
continue;
|
||||
}
|
||||
const name = segment.actor.memberName ?? actorLabel(segment.actor);
|
||||
visuals.set(segment.participantKey, {
|
||||
name,
|
||||
color: memberColorMap.get(name),
|
||||
});
|
||||
}
|
||||
|
||||
return visuals;
|
||||
}
|
||||
|
||||
const SegmentMarker = ({ segment }: { segment: BoardTaskLogSegment }): React.JSX.Element => {
|
||||
return (
|
||||
<div className="mb-2 flex items-center gap-2 text-[10px] text-[var(--color-text-muted)]">
|
||||
<span className="rounded-full border border-[var(--color-border)] px-2 py-0.5 font-medium text-[var(--color-text-secondary)]">
|
||||
{actorLabel(segment.actor)}
|
||||
</span>
|
||||
<div className="mb-2 flex items-center gap-1.5 text-[10px] text-[var(--color-text-muted)]">
|
||||
<span className="flex items-center gap-1">
|
||||
<Clock size={10} />
|
||||
{formatRelativeTime(segment.endTimestamp)}
|
||||
|
|
@ -104,18 +156,68 @@ const SegmentMarker = ({ segment }: { segment: BoardTaskLogSegment }): React.JSX
|
|||
const SegmentBlock = ({
|
||||
segment,
|
||||
showHeader,
|
||||
teamName,
|
||||
visual,
|
||||
}: {
|
||||
segment: BoardTaskLogSegment;
|
||||
showHeader: boolean;
|
||||
teamName: string;
|
||||
visual?: ParticipantVisual;
|
||||
}): React.JSX.Element => {
|
||||
return (
|
||||
<div className="min-w-0 overflow-hidden">
|
||||
{showHeader ? <SegmentMarker segment={segment} /> : null}
|
||||
<MemberExecutionLog chunks={segment.chunks} memberName={segment.actor.memberName} />
|
||||
<MemberExecutionLog
|
||||
chunks={segment.chunks}
|
||||
memberName={segment.actor.memberName}
|
||||
memberColor={visual?.color}
|
||||
teamName={teamName}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
const ParticipantFilterChip = ({
|
||||
label,
|
||||
selected,
|
||||
visual,
|
||||
teamName,
|
||||
onClick,
|
||||
}: {
|
||||
label: string;
|
||||
selected: boolean;
|
||||
visual?: ParticipantVisual;
|
||||
teamName: string;
|
||||
onClick: () => void;
|
||||
}): React.JSX.Element => {
|
||||
const { isLight } = useTheme();
|
||||
const colors = getTeamColorSet(visual?.color ?? '');
|
||||
const borderColor = selected ? getThemedBorder(colors, isLight) : 'var(--color-border)';
|
||||
const backgroundColor = selected ? getThemedBadge(colors, isLight) : 'transparent';
|
||||
const textColor = selected ? getThemedText(colors, isLight) : 'var(--color-text-muted)';
|
||||
|
||||
return (
|
||||
<button
|
||||
type="button"
|
||||
className="rounded-full border px-2 py-1 text-[11px] transition-colors hover:text-[var(--color-text)]"
|
||||
style={{ borderColor, backgroundColor, color: textColor }}
|
||||
onClick={onClick}
|
||||
>
|
||||
{visual ? (
|
||||
<MemberBadge
|
||||
name={visual.name}
|
||||
color={visual.color}
|
||||
teamName={teamName}
|
||||
size="xs"
|
||||
disableHoverCard
|
||||
/>
|
||||
) : (
|
||||
label
|
||||
)}
|
||||
</button>
|
||||
);
|
||||
};
|
||||
|
||||
export const TaskLogStreamSection = ({
|
||||
teamName,
|
||||
taskId,
|
||||
|
|
@ -126,6 +228,7 @@ export const TaskLogStreamSection = ({
|
|||
const [loading, setLoading] = useState(true);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [selectedParticipantKey, setSelectedParticipantKey] = useState<'all' | string>('all');
|
||||
const teamMembers = useStore((s) => selectResolvedMembersForTeamName(s, teamName));
|
||||
const requestSeqRef = useRef(0);
|
||||
const streamRef = useRef<BoardTaskLogStreamResponse | null>(null);
|
||||
const reloadTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||
|
|
@ -277,6 +380,11 @@ export const TaskLogStreamSection = ({
|
|||
}, [liveEnabled, loadStream, taskId, teamName]);
|
||||
|
||||
const participants = stream?.participants ?? [];
|
||||
const memberColorMap = useMemo(() => buildMemberColorMap(teamMembers), [teamMembers]);
|
||||
const participantVisuals = useMemo(
|
||||
() => buildParticipantVisualMap(stream, teamMembers, memberColorMap),
|
||||
[memberColorMap, stream, teamMembers]
|
||||
);
|
||||
const showChips = participants.length > 1;
|
||||
const streamDescription = useMemo(() => describeStreamSource(stream), [stream]);
|
||||
const visibleSegments = useMemo(() => {
|
||||
|
|
@ -340,18 +448,14 @@ export const TaskLogStreamSection = ({
|
|||
All
|
||||
</button>
|
||||
{participants.map((participant) => (
|
||||
<button
|
||||
<ParticipantFilterChip
|
||||
key={participant.key}
|
||||
type="button"
|
||||
className={`rounded-full border px-2.5 py-1 text-[11px] transition-colors ${
|
||||
selectedParticipantKey === participant.key
|
||||
? 'bg-[var(--color-accent)]/10 border-[var(--color-accent)] text-[var(--color-text)]'
|
||||
: 'border-[var(--color-border)] text-[var(--color-text-muted)] hover:text-[var(--color-text)]'
|
||||
}`}
|
||||
label={participant.label}
|
||||
selected={selectedParticipantKey === participant.key}
|
||||
visual={participantVisuals.get(participant.key)}
|
||||
teamName={teamName}
|
||||
onClick={() => setSelectedParticipantKey(participant.key)}
|
||||
>
|
||||
{participant.label}
|
||||
</button>
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
) : null}
|
||||
|
|
@ -372,6 +476,8 @@ export const TaskLogStreamSection = ({
|
|||
key={buildStableSegmentRenderKey(segment)}
|
||||
segment={segment}
|
||||
showHeader={showSegmentHeaders}
|
||||
teamName={teamName}
|
||||
visual={participantVisuals.get(segment.participantKey)}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
|
|
|
|||
Loading…
Reference in a new issue