agent-ecosystem/src/renderer/components/layout/Sidebar.tsx
iliya ec547e0662 refactor: enhance task management protocols and UI components
- Updated task management guidelines to emphasize the importance of posting task comments before marking tasks as complete, ensuring visibility of results on the task board.
- Introduced a new notification system for team leads after task completion, summarizing key findings and linking to detailed comments.
- Improved UI components, including a new CLI checking spinner with a delayed hint and enhanced task filters to support unread/read task states.
- Refactored sidebar task item styles to visually indicate unread tasks and adjusted task sorting options to include 'unread' as a criterion.
- Enhanced settings tabs with tooltips for better user guidance and improved layout for advanced settings options.
2026-03-20 13:45:03 +02:00

215 lines
7.3 KiB
TypeScript

/**
* Sidebar - Navigation with task list and session list.
*
* Structure:
* - Tab bar: Collapse button + Tasks | Sessions
* - Scrollable Body: Task list or date-grouped session list
* - Resizable: Drag right edge to resize
* - Collapsible: Cmd+B to toggle (Notion-style)
*/
import { useCallback, useEffect, useRef, useState } from 'react';
import { useStore } from '@renderer/store';
import { formatShortcut } from '@renderer/utils/stringUtils';
import { PanelLeft } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { DateGroupedSessions } from '../sidebar/DateGroupedSessions';
import { GlobalTaskList } from '../sidebar/GlobalTaskList';
import { defaultTaskFiltersState } from '../sidebar/taskFiltersState';
import type { TaskFiltersState } from '../sidebar/taskFiltersState';
type SidebarTab = 'tasks' | 'sessions';
const MIN_WIDTH = 200;
const MAX_WIDTH = 500;
const DEFAULT_WIDTH = 280;
export const Sidebar = (): React.JSX.Element => {
const { sidebarCollapsed, toggleSidebar } = useStore(
useShallow((s) => ({
sidebarCollapsed: s.sidebarCollapsed,
toggleSidebar: s.toggleSidebar,
}))
);
const [width, setWidth] = useState(DEFAULT_WIDTH);
const [isResizing, setIsResizing] = useState(false);
const [sidebarTab, setSidebarTab] = useState<SidebarTab>('tasks');
const [taskFilters, setTaskFilters] = useState<TaskFiltersState>(defaultTaskFiltersState);
const [taskFiltersPopoverOpen, setTaskFiltersPopoverOpen] = useState(false);
const [isCollapseHovered, setIsCollapseHovered] = useState(false);
const sidebarRef = useRef<HTMLDivElement>(null);
// Handle mouse move during resize (right sidebar: width = viewport - clientX)
const handleMouseMove = useCallback(
(e: MouseEvent) => {
if (!isResizing) return;
const newWidth = window.innerWidth - e.clientX;
if (newWidth >= MIN_WIDTH && newWidth <= MAX_WIDTH) {
setWidth(newWidth);
}
},
[isResizing]
);
// Handle mouse up to stop resizing
const handleMouseUp = useCallback(() => {
setIsResizing(false);
}, []);
// Add/remove event listeners for resize
useEffect(() => {
if (isResizing) {
document.addEventListener('mousemove', handleMouseMove);
document.addEventListener('mouseup', handleMouseUp);
document.body.style.cursor = 'col-resize';
document.body.style.userSelect = 'none';
}
return () => {
document.removeEventListener('mousemove', handleMouseMove);
document.removeEventListener('mouseup', handleMouseUp);
document.body.style.cursor = '';
document.body.style.userSelect = '';
};
}, [isResizing, handleMouseMove, handleMouseUp]);
const handleResizeStart = (e: React.MouseEvent): void => {
e.preventDefault();
setIsResizing(true);
};
return (
<div
ref={sidebarRef}
className="relative flex shrink-0 flex-col overflow-hidden border-l"
style={{
backgroundColor: 'var(--color-surface-sidebar)',
borderColor: 'var(--color-border)',
width: sidebarCollapsed ? 0 : width,
minWidth: sidebarCollapsed ? 0 : undefined,
borderLeftWidth: sidebarCollapsed ? 0 : undefined,
transition: 'width 0.22s ease-out, border-width 0.22s ease-out',
}}
>
<div
className="flex min-w-0 flex-1 flex-col overflow-hidden"
style={{
width: '100%',
minWidth: sidebarCollapsed ? 0 : width,
}}
>
{/* Tab bar: Collapse button + Tasks | Sessions */}
<div
className="flex shrink-0 items-end gap-2 border-b px-3 pt-1"
style={{ borderColor: 'var(--color-border)' }}
>
{/* Collapse sidebar button */}
<button
onClick={toggleSidebar}
onMouseEnter={() => setIsCollapseHovered(true)}
onMouseLeave={() => setIsCollapseHovered(false)}
className="mb-1 shrink-0 rounded-md p-1 transition-colors"
style={{
color: isCollapseHovered ? 'var(--color-text-secondary)' : 'var(--color-text-muted)',
backgroundColor: isCollapseHovered ? 'var(--color-surface-raised)' : 'transparent',
}}
title={`Collapse sidebar (${formatShortcut('B')})`}
>
<PanelLeft className="size-3.5" />
</button>
<div className="flex-1" />
<div className="flex" role="tablist" aria-label="Sidebar view">
<button
type="button"
role="tab"
aria-selected={sidebarTab === 'tasks'}
aria-controls="sidebar-tasks-panel"
id="sidebar-tab-tasks"
className={`relative px-3 py-1.5 text-[11px] font-medium transition-colors ${
sidebarTab === 'tasks' ? 'text-text' : 'text-text-muted hover:text-text-secondary'
}`}
style={
sidebarTab === 'tasks'
? {
borderBottom: '2px solid var(--color-text)',
marginBottom: '-1px',
}
: undefined
}
onClick={() => setSidebarTab('tasks')}
>
Tasks
</button>
<button
type="button"
role="tab"
aria-selected={sidebarTab === 'sessions'}
aria-controls="sidebar-sessions-panel"
id="sidebar-tab-sessions"
className={`relative px-3 py-1.5 text-[11px] font-medium transition-colors ${
sidebarTab === 'sessions'
? 'text-text'
: 'text-text-muted hover:text-text-secondary'
}`}
style={
sidebarTab === 'sessions'
? {
borderBottom: '2px solid var(--color-text)',
marginBottom: '-1px',
}
: undefined
}
onClick={() => setSidebarTab('sessions')}
>
Sessions
</button>
</div>
<div className="flex-1" />
</div>
{/* Content: Tasks list or Sessions list */}
<div
id="sidebar-tasks-panel"
role="tabpanel"
aria-labelledby="sidebar-tab-tasks"
hidden={sidebarTab !== 'tasks'}
className="min-w-0 flex-1 overflow-hidden"
>
<GlobalTaskList
hideHeader
filters={taskFilters}
onFiltersChange={setTaskFilters}
filtersPopoverOpen={taskFiltersPopoverOpen}
onFiltersPopoverOpenChange={setTaskFiltersPopoverOpen}
/>
</div>
<div
id="sidebar-sessions-panel"
role="tabpanel"
aria-labelledby="sidebar-tab-sessions"
hidden={sidebarTab !== 'sessions'}
className="min-w-0 flex-1 overflow-hidden"
>
<DateGroupedSessions />
</div>
</div>
{/* Resize handle - only interactive when expanded */}
{!sidebarCollapsed && (
<button
type="button"
aria-label="Resize sidebar"
className={`absolute left-0 top-0 h-full w-1 cursor-col-resize border-0 bg-transparent p-0 transition-colors hover:bg-blue-500/50 ${
isResizing ? 'bg-blue-500/50' : ''
}`}
onMouseDown={handleResizeStart}
/>
)}
</div>
);
};