perf: offload heavy I/O to worker thread, reduce renderer re-renders

Main process — worker thread for team data:
- New team-data-worker thread handles getTeamData and findLogsForTask,
  isolating heavy file I/O (scanning 300+ subagent JSONL files) from
  Electron's main event loop. getTeamData dropped from ~2000ms on the
  main thread to ~110ms via the worker.
- Worker-side dedup and 10s result cache for findLogsForTask prevents
  redundant scans when the same task is queried multiple times.
- Discovery cache TTL raised from 5s to 30s — avoids re-scanning the
  entire project directory on every call.
- Message cap at 200 in TeamDataService to keep IPC payloads under 1MB
  (was sending 2200+ messages / ~3MB, stalling Chromium IPC serialization).
- IPC handlers fall back to main-thread execution if the worker is
  unavailable (graceful degradation).

Renderer — useShallow and memoization (55 files):
- Added useShallow to store selectors across 55 renderer files. Batched
  individual useStore() calls (e.g. 17 calls in ExtensionStoreView,
  10 in ConnectionSection) into single useShallow selectors, cutting
  unnecessary re-render checks on every store update.
- MemberLogsTab: three 5-second polling intervals now pause when the
  parent tab is hidden (display:none). Previously 5 hidden tabs × 3
  intervals = 15 polling timers firing continuously.
- KanbanColumn wrapped in React.memo to skip re-renders when props
  haven't changed.
- MemberList: memoized activeMembers/removedMembers/colorMap; replaced
  O(n×m) per-member task scan with a pre-computed reviewer map.
- Bounded timer Maps in store initialization to prevent unbounded growth
  of debounce/throttle tracking maps during long sessions.
This commit is contained in:
Artem Rootman 2026-04-05 16:21:05 +00:00
parent 7ff9317b6f
commit da58917032
No known key found for this signature in database
GPG key ID: B7C30676209A822C
61 changed files with 843 additions and 232 deletions

View file

@ -77,7 +77,8 @@ export default defineConfig({
input: {
index: resolve(__dirname, 'src/main/index.ts'),
'team-fs-worker': resolve(__dirname, 'src/main/workers/team-fs-worker.ts'),
'task-change-worker': resolve(__dirname, 'src/main/workers/task-change-worker.ts')
'task-change-worker': resolve(__dirname, 'src/main/workers/task-change-worker.ts'),
'team-data-worker': resolve(__dirname, 'src/main/workers/team-data-worker.ts')
},
output: {
// CJS format so bundled deps can use __dirname/require.

View file

@ -1,5 +1,6 @@
import { addMainBreadcrumb } from '@main/sentry';
import { setCurrentMainOp } from '@main/services/infrastructure/EventLoopLagMonitor';
import { getTeamDataWorkerClient } from '@main/services/team/TeamDataWorkerClient';
import { getAppIconPath } from '@main/utils/appIcon';
import { getAppDataPath, getTeamsBasePath } from '@main/utils/pathDecoder';
import { stripMarkdown } from '@main/utils/textFormatting';
@ -535,7 +536,20 @@ async function handleGetData(
const startedAt = Date.now();
let data: TeamData;
try {
data = await getTeamDataService().getTeamData(tn);
// Prefer worker thread to keep main event loop responsive
const worker = getTeamDataWorkerClient();
if (worker.isAvailable()) {
try {
data = await worker.getTeamData(tn);
} catch (workerErr) {
logger.warn(
`[teams:getData] worker failed, falling back: ${workerErr instanceof Error ? workerErr.message : workerErr}`
);
data = await getTeamDataService().getTeamData(tn);
}
} else {
data = await getTeamDataService().getTeamData(tn);
}
} catch (error) {
const message = error instanceof Error ? error.message : String(error);
if (
@ -555,6 +569,7 @@ async function handleGetData(
return { success: false, error: message };
}
const getDataMs = Date.now() - startedAt;
if (getDataMs >= 1500) {
logger.warn(`[teams:getData] slow team=${tn} ms=${getDataMs}`);
}
@ -2138,6 +2153,19 @@ async function handleGetLogsForTask(
: undefined,
}
: undefined;
// Prefer worker thread to keep main event loop responsive
const worker = getTeamDataWorkerClient();
if (worker.isAvailable()) {
try {
return await wrapTeamHandler('getLogsForTask', () =>
worker.findLogsForTask(vTeam.value!, vTask.value!, opts)
);
} catch (workerErr) {
logger.warn(
`[teams:getLogsForTask] worker failed, falling back: ${workerErr instanceof Error ? workerErr.message : workerErr}`
);
}
}
return wrapTeamHandler('getLogsForTask', () =>
getTeamMemberLogsFinder().findLogsForTask(vTeam.value!, vTask.value!, opts)
);

View file

@ -728,12 +728,19 @@ export class TeamDataService {
this.processHealthTeams.delete(teamName);
}
// Cap messages to keep IPC/postMessage payloads under ~300KB.
// Without this, teams with 2000+ messages produce 3MB+ payloads that
// stall Chromium's IPC serialization for ~1 second per transfer.
const MAX_RETURN_MESSAGES = 200;
const cappedMessages =
messages.length > MAX_RETURN_MESSAGES ? messages.slice(0, MAX_RETURN_MESSAGES) : messages;
return {
teamName,
config,
tasks: tasksWithKanban,
members,
messages,
messages: cappedMessages,
kanbanState,
processes,
warnings: warnings.length > 0 ? warnings : undefined,

View file

@ -0,0 +1,169 @@
/**
* Main-thread client for team-data-worker.
*
* Proxies getTeamData and findLogsForTask calls to a worker thread
* so they don't block the Electron main event loop.
* Falls back to main-thread execution if the worker is unavailable.
*/
import * as fs from 'node:fs';
import * as path from 'node:path';
import { fileURLToPath } from 'node:url';
import { Worker } from 'node:worker_threads';
import { createLogger } from '@shared/utils/logger';
import type { MemberLogSummary, TeamData } from '@shared/types';
import type { TeamDataWorkerRequest, TeamDataWorkerResponse } from './teamDataWorkerTypes';
const logger = createLogger('Service:TeamDataWorkerClient');
const WORKER_CALL_TIMEOUT_MS = 30_000;
function makeId(): string {
return `${Date.now()}-${crypto.randomUUID().slice(0, 12)}`;
}
function resolveWorkerPath(): string | null {
const baseDir =
typeof __dirname === 'string' && __dirname.length > 0
? __dirname
: path.dirname(fileURLToPath(import.meta.url));
const candidates = [
path.join(baseDir, 'team-data-worker.cjs'),
path.join(process.cwd(), 'dist-electron', 'main', 'team-data-worker.cjs'),
];
for (const candidate of candidates) {
try {
if (fs.existsSync(candidate)) return candidate;
} catch {
/* ignore */
}
}
logger.warn('team-data-worker not found in expected locations');
return null;
}
type PendingEntry = {
resolve: (v: unknown) => void;
reject: (e: Error) => void;
};
export class TeamDataWorkerClient {
private worker: Worker | null = null;
private readonly workerPath: string | null = resolveWorkerPath();
private warnedUnavailable = false;
private pending = new Map<string, PendingEntry>();
isAvailable(): boolean {
if (!this.workerPath && !this.warnedUnavailable) {
this.warnedUnavailable = true;
logger.warn('team-data-worker not found; falling back to main-thread execution');
}
return this.workerPath !== null;
}
private ensureWorker(): Worker {
if (!this.workerPath) throw new Error('Worker not available');
if (this.worker) return this.worker;
this.worker = new Worker(this.workerPath);
this.worker.on('message', (msg: TeamDataWorkerResponse) => {
const entry = this.pending.get(msg.id);
if (!entry) return;
this.pending.delete(msg.id);
if (msg.ok) {
entry.resolve(msg.result);
} else {
entry.reject(new Error(msg.error));
}
});
this.worker.on('error', (err) => {
logger.error('Worker error', err);
for (const [, entry] of this.pending) {
entry.reject(err instanceof Error ? err : new Error(String(err)));
}
this.pending.clear();
this.worker = null;
});
this.worker.on('exit', (code) => {
if (code !== 0) logger.warn(`Worker exited with code ${code}`);
for (const [, entry] of this.pending) {
entry.reject(new Error(`Worker exited with code ${code}`));
}
this.pending.clear();
this.worker = null;
});
return this.worker;
}
private call(
op: TeamDataWorkerRequest['op'],
payload: TeamDataWorkerRequest['payload']
): Promise<unknown> {
const worker = this.ensureWorker();
const id = makeId();
return new Promise((resolve, reject) => {
const timeout = setTimeout(() => {
this.pending.delete(id);
this.worker?.terminate().catch(() => undefined);
this.worker = null;
reject(new Error(`Worker call timeout after ${WORKER_CALL_TIMEOUT_MS}ms`));
}, WORKER_CALL_TIMEOUT_MS);
this.pending.set(id, {
resolve: (value) => {
clearTimeout(timeout);
resolve(value);
},
reject: (error) => {
clearTimeout(timeout);
reject(error);
},
});
worker.postMessage({ id, op, payload } as TeamDataWorkerRequest);
});
}
async getTeamData(teamName: string): Promise<TeamData> {
return this.call('getTeamData', { teamName }) as Promise<TeamData>;
}
async findLogsForTask(
teamName: string,
taskId: string,
options?: {
owner?: string;
status?: string;
intervals?: { startedAt: string; completedAt?: string }[];
since?: string;
}
): Promise<MemberLogSummary[]> {
return this.call('findLogsForTask', { teamName, taskId, options }) as Promise<
MemberLogSummary[]
>;
}
dispose(): void {
this.worker?.terminate().catch(() => undefined);
this.worker = null;
for (const [, entry] of this.pending) {
entry.reject(new Error('Client disposed'));
}
this.pending.clear();
}
}
// Singleton
let singleton: TeamDataWorkerClient | null = null;
export function getTeamDataWorkerClient(): TeamDataWorkerClient {
if (!singleton) singleton = new TeamDataWorkerClient();
return singleton;
}

View file

@ -31,7 +31,7 @@ const ATTRIBUTION_CACHE_MAX = 5_000;
const SCAN_CONCURRENCY = 15;
/** TTL for discoverProjectSessions cache — avoids re-reading config/dirs within rapid successive calls. */
const DISCOVERY_CACHE_TTL = 5_000;
const DISCOVERY_CACHE_TTL = 30_000;
/** Signal sources for subagent member attribution, ordered by reliability. */
type AttributionSignalSource = 'process_team' | 'routing_sender' | 'teammate_id' | 'text_mention';

View file

@ -0,0 +1,32 @@
/**
* Shared request/response types for the team-data-worker thread.
*/
import type { MemberLogSummary, TeamData } from '@shared/types';
// ── Payloads ──
export interface GetTeamDataPayload {
teamName: string;
}
export interface FindLogsForTaskPayload {
teamName: string;
taskId: string;
options?: {
owner?: string;
status?: string;
intervals?: { startedAt: string; completedAt?: string }[];
since?: string;
};
}
// ── Request / Response ──
export type TeamDataWorkerRequest =
| { id: string; op: 'getTeamData'; payload: GetTeamDataPayload }
| { id: string; op: 'findLogsForTask'; payload: FindLogsForTaskPayload };
export type TeamDataWorkerResponse =
| { id: string; ok: true; result: TeamData | MemberLogSummary[] }
| { id: string; ok: false; error: string };

View file

@ -0,0 +1,87 @@
/**
* Worker thread for heavy team I/O operations (getTeamData, findLogsForTask).
*
* Runs in its own event loop, completely isolated from the Electron main thread.
* This prevents file-heavy operations (scanning 300+ subagent JSONL files,
* parsing large session files) from stalling the main process UI/IPC.
*/
import { parentPort } from 'node:worker_threads';
import { TeamDataService } from '@main/services/team/TeamDataService';
import { TeamMemberLogsFinder } from '@main/services/team/TeamMemberLogsFinder';
import { createLogger } from '@shared/utils/logger';
import type { MemberLogSummary } from '@shared/types';
import type {
TeamDataWorkerRequest,
TeamDataWorkerResponse,
} from '@main/services/team/teamDataWorkerTypes';
const logger = createLogger('Worker:TeamData');
// Instantiate services with default dependencies — worker has its own event loop
const teamDataService = new TeamDataService();
const logsFinder = new TeamMemberLogsFinder();
// In-flight dedup: concurrent calls for the same task piggyback on one request
const logsInFlight = new Map<string, Promise<unknown>>();
// Result cache with TTL to avoid re-scanning files
const logsResultCache = new Map<string, { result: MemberLogSummary[]; cachedAt: number }>();
const LOGS_CACHE_TTL_MS = 10_000;
function respond(msg: TeamDataWorkerResponse): void {
parentPort?.postMessage(msg);
}
parentPort?.on('message', async (msg: TeamDataWorkerRequest) => {
try {
switch (msg.op) {
case 'getTeamData': {
const result = await teamDataService.getTeamData(msg.payload.teamName);
respond({ id: msg.id, ok: true, result });
break;
}
case 'findLogsForTask': {
const { teamName, taskId, options } = msg.payload;
const cacheKey = `${teamName}:${taskId}:${options?.owner ?? ''}`;
// Check result cache
const cached = logsResultCache.get(cacheKey);
if (cached && Date.now() - cached.cachedAt < LOGS_CACHE_TTL_MS) {
respond({ id: msg.id, ok: true, result: cached.result });
break;
}
// Dedup concurrent calls
let promise = logsInFlight.get(cacheKey) as Promise<MemberLogSummary[]> | undefined;
if (!promise) {
promise = logsFinder.findLogsForTask(teamName, taskId, options).then((result) => {
logsInFlight.delete(cacheKey);
logsResultCache.set(cacheKey, { result, cachedAt: Date.now() });
// Cap cache
if (logsResultCache.size > 100) {
const firstKey = logsResultCache.keys().next().value;
if (firstKey !== undefined) logsResultCache.delete(firstKey);
}
return result;
});
logsInFlight.set(cacheKey, promise);
}
const result = await promise;
respond({ id: msg.id, ok: true, result });
break;
}
default: {
const _exhaustive: never = msg;
respond({ id: (_exhaustive as { id: string }).id, ok: false, error: `Unknown op` });
}
}
} catch (err) {
const message = err instanceof Error ? err.message : String(err);
logger.error(`[${msg.op}] ${message}`);
respond({ id: msg.id, ok: false, error: message });
}
});
logger.info('team-data-worker started');

View file

@ -171,7 +171,7 @@ const AIChatGroupInner = ({
);
// Notification color map for tool item dots
const notifications = useStore((s) => s.notifications);
const notifications = useStore(useShallow((s) => s.notifications));
const notificationColorMap = useMemo(() => {
const map = new Map<string, TriggerColor>();
for (const n of notifications) {

View file

@ -128,7 +128,7 @@ export const ChatHistory = ({ tabId }: ChatHistoryProps): JSX.Element => {
const thisTab = effectiveTabId ? openTabs.find((t) => t.id === effectiveTabId) : null;
const pendingNavigation = thisTab?.pendingNavigation;
const teamBySessionId = useStore((s) => s.teamBySessionId);
const teamBySessionId = useStore(useShallow((s) => s.teamBySessionId));
// Look up whether this session belongs to a team
const sessionTeam = useMemo(() => {

View file

@ -388,15 +388,17 @@ const UserChatGroupInner = ({ userGroup }: Readonly<UserChatGroupProps>): React.
return (td?.sessionDetail ?? s.sessionDetail)?.session?.projectPath;
});
// Get team members for @mention highlighting
const members = useStore((s) => s.selectedTeamData?.members);
// Get team members for @mention highlighting and team names for @team linkification
const { members, teams } = useStore(
useShallow((s) => ({
members: s.selectedTeamData?.members,
teams: s.teams,
}))
);
const memberColorMap = useMemo(
() => (members ? buildMemberColorMap(members) : new Map<string, string>()),
[members]
);
// Get team names for @team linkification
const teams = useStore((s) => s.teams);
const teamNames = useMemo(
() => teams.filter((t) => !t.deletedAt).map((t) => t.teamName),
[teams]

View file

@ -20,6 +20,7 @@ import {
import { useTabUI } from '@renderer/hooks/useTabUI';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { buildDisplayItemsFromMessages, buildSummary } from '@renderer/utils/aiGroupEnhancer';
import { computeSubagentPhaseBreakdown } from '@renderer/utils/aiGroupHelpers';
import { formatDuration, formatTokensCompact } from '@renderer/utils/formatters';
@ -82,7 +83,7 @@ export const SubagentItem: React.FC<SubagentItemProps> = ({
const truncatedDesc = description.length > 60 ? description.slice(0, 60) + '...' : description;
// Agent configs from .claude/agents/ for color lookup
const agentConfigs = useStore((s) => s.agentConfigs);
const agentConfigs = useStore(useShallow((s) => s.agentConfigs));
// Team member colors (when this subagent is a team member)
const teamColors = subagent.team ? getTeamColorSet(subagent.team.memberColor) : null;
@ -171,7 +172,7 @@ export const SubagentItem: React.FC<SubagentItemProps> = ({
}, [subagent.messages]);
// Search expansion
const searchExpandedSubagentIds = useStore((s) => s.searchExpandedSubagentIds);
const searchExpandedSubagentIds = useStore(useShallow((s) => s.searchExpandedSubagentIds));
const searchCurrentSubagentItemId = useStore((s) => s.searchCurrentSubagentItemId);
const shouldExpandForSearch = searchExpandedSubagentIds.has(subagent.id);

View file

@ -10,6 +10,7 @@ import {
import { getTeamColorSet, getThemedBadge } from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { detectOperationalNoise } from '@renderer/utils/agentMessageFormatting';
import { formatTokensCompact } from '@renderer/utils/formatters';
import { buildMemberColorMap } from '@renderer/utils/memberHelpers';
@ -85,14 +86,14 @@ export const TeammateMessageItem: React.FC<TeammateMessageItemProps> = ({
const { isLight } = useTheme();
// Get team members for @mention highlighting
const members = useStore((s) => s.selectedTeamData?.members);
const members = useStore(useShallow((s) => s.selectedTeamData?.members));
const memberColorMap = useMemo(
() => (members ? buildMemberColorMap(members) : new Map<string, string>()),
[members]
);
// Get team names for @team linkification
const teams = useStore((s) => s.teams);
const teams = useStore(useShallow((s) => s.teams));
const teamNames = useMemo(
() => teams.filter((t) => !t.deletedAt).map((t) => t.teamName),
[teams]

View file

@ -690,7 +690,7 @@ export const MarkdownViewer: React.FC<MarkdownViewerProps> = ({
const [showRaw, setShowRaw] = React.useState(false);
const [rawLimit, setRawLimit] = React.useState(LARGE_PREVIEW_CHARS);
const { isLight } = useTheme();
const teams = useStore((s) => (providedTeamColorByName ? EMPTY_TEAMS : s.teams));
const teams = useStore(useShallow((s) => (providedTeamColorByName ? EMPTY_TEAMS : s.teams)));
const openTeamTab = useStore((s) => (providedOnTeamClick ? NOOP_TEAM_CLICK : s.openTeamTab));
const fallbackTeamColorByName = React.useMemo(() => {

View file

@ -8,10 +8,11 @@
import { isElectronMode } from '@renderer/api';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { AlertTriangle } from 'lucide-react';
export const CliInstallWarningBanner = (): React.JSX.Element | null => {
const cliStatus = useStore((s) => s.cliStatus);
const cliStatus = useStore(useShallow((s) => s.cliStatus));
const openDashboard = useStore((s) => s.openDashboard);
// Returns a primitive boolean — minimizes re-renders

View file

@ -11,6 +11,7 @@
import { useStore } from '@renderer/store';
import { Loader2, Monitor, Wifi, WifiOff } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
interface ConnectionStatusBadgeProps {
contextId: string;
@ -21,10 +22,12 @@ export const ConnectionStatusBadge = ({
contextId,
className,
}: Readonly<ConnectionStatusBadgeProps>): React.JSX.Element => {
const { connectionState, connectedHost } = useStore((s) => ({
connectionState: s.connectionState,
connectedHost: s.connectedHost,
}));
const { connectionState, connectedHost } = useStore(
useShallow((s) => ({
connectionState: s.connectionState,
connectedHost: s.connectedHost,
}))
);
// Local context always shows Monitor icon
if (contextId === 'local') {

View file

@ -6,14 +6,26 @@
import { useStore } from '@renderer/store';
import { CheckCircle, Loader2, X } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
export const UpdateBanner = (): React.JSX.Element | null => {
const showUpdateBanner = useStore((s) => s.showUpdateBanner);
const updateStatus = useStore((s) => s.updateStatus);
const downloadProgress = useStore((s) => s.downloadProgress);
const availableVersion = useStore((s) => s.availableVersion);
const installUpdate = useStore((s) => s.installUpdate);
const dismissUpdateBanner = useStore((s) => s.dismissUpdateBanner);
const {
showUpdateBanner,
updateStatus,
downloadProgress,
availableVersion,
installUpdate,
dismissUpdateBanner,
} = useStore(
useShallow((s) => ({
showUpdateBanner: s.showUpdateBanner,
updateStatus: s.updateStatus,
downloadProgress: s.downloadProgress,
availableVersion: s.availableVersion,
installUpdate: s.installUpdate,
dismissUpdateBanner: s.dismissUpdateBanner,
}))
);
if (!showUpdateBanner || (updateStatus !== 'downloading' && updateStatus !== 'downloaded')) {
return null;

View file

@ -15,15 +15,28 @@ import { useStore } from '@renderer/store';
import { REHYPE_PLUGINS } from '@renderer/utils/markdownPlugins';
import { ExternalLink, X } from 'lucide-react';
import remarkGfm from 'remark-gfm';
import { useShallow } from 'zustand/react/shallow';
export const UpdateDialog = (): React.JSX.Element | null => {
const showUpdateDialog = useStore((s) => s.showUpdateDialog);
const updateStatus = useStore((s) => s.updateStatus);
const availableVersion = useStore((s) => s.availableVersion);
const releaseNotes = useStore((s) => s.releaseNotes);
const downloadUpdate = useStore((s) => s.downloadUpdate);
const installUpdate = useStore((s) => s.installUpdate);
const dismissUpdateDialog = useStore((s) => s.dismissUpdateDialog);
const {
showUpdateDialog,
updateStatus,
availableVersion,
releaseNotes,
downloadUpdate,
installUpdate,
dismissUpdateDialog,
} = useStore(
useShallow((s) => ({
showUpdateDialog: s.showUpdateDialog,
updateStatus: s.updateStatus,
availableVersion: s.availableVersion,
releaseNotes: s.releaseNotes,
downloadUpdate: s.downloadUpdate,
installUpdate: s.installUpdate,
dismissUpdateDialog: s.dismissUpdateDialog,
}))
);
const dialogRef = useRef<HTMLDivElement>(null);

View file

@ -18,6 +18,7 @@ import {
import { useTabIdOptional } from '@renderer/contexts/useTabUIContext';
import { useExtensionsTabState } from '@renderer/hooks/useExtensionsTabState';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { AlertTriangle, BookOpen, Info, Key, Plus, Puzzle, RefreshCw, Server } from 'lucide-react';
import { ApiKeysPanel } from './apikeys/ApiKeysPanel';
@ -29,18 +30,35 @@ import { ExtensionsSubTabTrigger } from './ExtensionsSubTabTrigger';
export const ExtensionStoreView = (): React.JSX.Element => {
const tabId = useTabIdOptional();
const fetchPluginCatalog = useStore((s) => s.fetchPluginCatalog);
const fetchApiKeys = useStore((s) => s.fetchApiKeys);
const fetchSkillsCatalog = useStore((s) => s.fetchSkillsCatalog);
const mcpBrowse = useStore((s) => s.mcpBrowse);
const mcpFetchInstalled = useStore((s) => s.mcpFetchInstalled);
const pluginCatalogLoading = useStore((s) => s.pluginCatalogLoading);
const mcpBrowseLoading = useStore((s) => s.mcpBrowseLoading);
const skillsLoading = useStore((s) => s.skillsLoading);
const cliStatus = useStore((s) => s.cliStatus);
const cliInstalled = cliStatus?.installed ?? true; // assume installed until checked
const hasOngoingSessions = useStore((s) => s.sessions.some((sess) => sess.isOngoing));
const projects = useStore((s) => s.projects);
const {
fetchPluginCatalog,
fetchApiKeys,
fetchSkillsCatalog,
mcpBrowse,
mcpFetchInstalled,
pluginCatalogLoading,
mcpBrowseLoading,
skillsLoading,
cliStatus,
sessions,
projects,
} = useStore(
useShallow((s) => ({
fetchPluginCatalog: s.fetchPluginCatalog,
fetchApiKeys: s.fetchApiKeys,
fetchSkillsCatalog: s.fetchSkillsCatalog,
mcpBrowse: s.mcpBrowse,
mcpFetchInstalled: s.mcpFetchInstalled,
pluginCatalogLoading: s.pluginCatalogLoading,
mcpBrowseLoading: s.mcpBrowseLoading,
skillsLoading: s.skillsLoading,
cliStatus: s.cliStatus,
sessions: s.sessions,
projects: s.projects,
}))
);
const cliInstalled = cliStatus?.installed ?? true;
const hasOngoingSessions = sessions.some((sess) => sess.isOngoing);
const extensionsTabProjectId = useStore((s) =>
tabId
? (s.paneLayout.panes.flatMap((pane) => pane.tabs).find((tab) => tab.id === tabId)

View file

@ -7,6 +7,7 @@ import { useEffect, useState } from 'react';
import { Button } from '@renderer/components/ui/button';
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { AlertTriangle, Info, Key, Plus } from 'lucide-react';
import { ApiKeyCard } from './ApiKeyCard';
@ -15,11 +16,15 @@ import { ApiKeyFormDialog } from './ApiKeyFormDialog';
import type { ApiKeyEntry } from '@shared/types/extensions';
export const ApiKeysPanel = (): React.JSX.Element => {
const apiKeys = useStore((s) => s.apiKeys);
const apiKeysLoading = useStore((s) => s.apiKeysLoading);
const apiKeysError = useStore((s) => s.apiKeysError);
const storageStatus = useStore((s) => s.apiKeyStorageStatus);
const fetchStorageStatus = useStore((s) => s.fetchApiKeyStorageStatus);
const { apiKeys, apiKeysLoading, apiKeysError, storageStatus, fetchStorageStatus } = useStore(
useShallow((s) => ({
apiKeys: s.apiKeys,
apiKeysLoading: s.apiKeysLoading,
apiKeysError: s.apiKeysError,
storageStatus: s.apiKeyStorageStatus,
fetchStorageStatus: s.fetchApiKeyStorageStatus,
}))
);
const [dialogOpen, setDialogOpen] = useState(false);
const [editingKey, setEditingKey] = useState<ApiKeyEntry | null>(null);

View file

@ -13,6 +13,7 @@ import {
TooltipTrigger,
} from '@renderer/components/ui/tooltip';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { Check, Loader2, Trash2 } from 'lucide-react';
import type { ExtensionOperationState } from '@shared/types/extensions';
@ -36,7 +37,7 @@ export const InstallButton = ({
size = 'sm',
errorMessage,
}: InstallButtonProps) => {
const cliStatus = useStore((s) => s.cliStatus);
const cliStatus = useStore(useShallow((s) => s.cliStatus));
const cliMissing = cliStatus !== null && !cliStatus.installed;
const isDisabled = disabled || cliMissing;
const [lastAction, setLastAction] = useState<'install' | 'uninstall' | null>(null);

View file

@ -14,6 +14,7 @@ import {
SelectValue,
} from '@renderer/components/ui/select';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatRelativeTime } from '@renderer/utils/formatters';
import { CLI_NOT_FOUND_MARKER } from '@shared/constants/cli';
import { sanitizeMcpServerName } from '@shared/utils/extensionNormalizers';
@ -72,18 +73,35 @@ export const McpServersPanel = ({
selectedMcpServerId,
setSelectedMcpServerId,
}: McpServersPanelProps): React.JSX.Element => {
const browseCatalog = useStore((s) => s.mcpBrowseCatalog);
const browseNextCursor = useStore((s) => s.mcpBrowseNextCursor);
const browseLoading = useStore((s) => s.mcpBrowseLoading);
const browseError = useStore((s) => s.mcpBrowseError);
const mcpBrowse = useStore((s) => s.mcpBrowse);
const installedServers = useStore((s) => s.mcpInstalledServers);
const fetchMcpGitHubStars = useStore((s) => s.fetchMcpGitHubStars);
const mcpDiagnostics = useStore((s) => s.mcpDiagnostics);
const mcpDiagnosticsLoading = useStore((s) => s.mcpDiagnosticsLoading);
const mcpDiagnosticsError = useStore((s) => s.mcpDiagnosticsError);
const mcpDiagnosticsLastCheckedAt = useStore((s) => s.mcpDiagnosticsLastCheckedAt);
const runMcpDiagnostics = useStore((s) => s.runMcpDiagnostics);
const {
browseCatalog,
browseNextCursor,
browseLoading,
browseError,
mcpBrowse,
installedServers,
fetchMcpGitHubStars,
mcpDiagnostics,
mcpDiagnosticsLoading,
mcpDiagnosticsError,
mcpDiagnosticsLastCheckedAt,
runMcpDiagnostics,
} = useStore(
useShallow((s) => ({
browseCatalog: s.mcpBrowseCatalog,
browseNextCursor: s.mcpBrowseNextCursor,
browseLoading: s.mcpBrowseLoading,
browseError: s.mcpBrowseError,
mcpBrowse: s.mcpBrowse,
installedServers: s.mcpInstalledServers,
fetchMcpGitHubStars: s.fetchMcpGitHubStars,
mcpDiagnostics: s.mcpDiagnostics,
mcpDiagnosticsLoading: s.mcpDiagnosticsLoading,
mcpDiagnosticsError: s.mcpDiagnosticsError,
mcpDiagnosticsLastCheckedAt: s.mcpDiagnosticsLastCheckedAt,
runMcpDiagnostics: s.runMcpDiagnostics,
}))
);
const [mcpSort, setMcpSort] = useState<McpSortValue>('name-asc');

View file

@ -24,6 +24,7 @@ import {
SelectValue,
} from '@renderer/components/ui/select';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import {
getCapabilityLabel,
inferCapabilities,
@ -53,14 +54,18 @@ export const PluginDetailDialog = ({
open,
onClose,
}: PluginDetailDialogProps): React.JSX.Element => {
const fetchPluginReadme = useStore((s) => s.fetchPluginReadme);
const readmes = useStore((s) => s.pluginReadmes);
const readmeLoading = useStore((s) => s.pluginReadmeLoading);
const { fetchPluginReadme, readmes, readmeLoading, installPlugin, uninstallPlugin } = useStore(
useShallow((s) => ({
fetchPluginReadme: s.fetchPluginReadme,
readmes: s.pluginReadmes,
readmeLoading: s.pluginReadmeLoading,
installPlugin: s.installPlugin,
uninstallPlugin: s.uninstallPlugin,
}))
);
const installProgress = useStore(
(s) => (plugin ? s.pluginInstallProgress[plugin.pluginId] : undefined) ?? 'idle'
);
const installPlugin = useStore((s) => s.installPlugin);
const uninstallPlugin = useStore((s) => s.uninstallPlugin);
const installError = useStore((s) => (plugin ? s.installErrors[plugin.pluginId] : undefined));
const [scope, setScope] = useState<InstallScope>('user');

View file

@ -16,6 +16,7 @@ import {
SelectValue,
} from '@renderer/components/ui/select';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { inferCapabilities, normalizeCategory } from '@shared/utils/extensionNormalizers';
import { ArrowUpDown, Filter, Puzzle, Search } from 'lucide-react';
@ -122,9 +123,13 @@ export const PluginsPanel = ({
hasActiveFilters,
setPluginSort,
}: PluginsPanelProps): React.JSX.Element => {
const catalog = useStore((s) => s.pluginCatalog);
const loading = useStore((s) => s.pluginCatalogLoading);
const error = useStore((s) => s.pluginCatalogError);
const { catalog, loading, error } = useStore(
useShallow((s) => ({
catalog: s.pluginCatalog,
loading: s.pluginCatalogLoading,
error: s.pluginCatalogError,
}))
);
const filtered = useMemo(
() => selectFilteredPlugins(catalog, pluginFilters, pluginSort),

View file

@ -23,6 +23,7 @@ import {
DialogTitle,
} from '@renderer/components/ui/dialog';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { AlertTriangle, ExternalLink, FolderOpen, Pencil, Trash2 } from 'lucide-react';
interface SkillDetailDialogProps {
@ -44,7 +45,7 @@ export const SkillDetailDialog = ({
}: SkillDetailDialogProps): React.JSX.Element => {
const fetchSkillDetail = useStore((s) => s.fetchSkillDetail);
const deleteSkill = useStore((s) => s.deleteSkill);
const detail = useStore((s) => (skillId ? s.skillsDetailsById[skillId] : undefined));
const detail = useStore(useShallow((s) => (skillId ? s.skillsDetailsById[skillId] : undefined)));
const loading = useStore((s) =>
skillId ? (s.skillsDetailLoadingById[skillId] ?? false) : false
);

View file

@ -6,6 +6,7 @@ import { Button } from '@renderer/components/ui/button';
import { Popover, PopoverContent, PopoverTrigger } from '@renderer/components/ui/popover';
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import {
AlertTriangle,
ArrowUpAZ,
@ -94,10 +95,10 @@ export const SkillsPanel = ({
const fetchSkillDetail = useStore((s) => s.fetchSkillDetail);
const skillsLoading = useStore((s) => s.skillsCatalogLoadingByProjectPath[catalogKey] ?? false);
const skillsError = useStore((s) => s.skillsCatalogErrorByProjectPath[catalogKey] ?? null);
const detailById = useStore((s) => s.skillsDetailsById);
const userSkills = useStore((s) => s.skillsUserCatalog);
const projectSkills = useStore((s) =>
projectPath ? (s.skillsProjectCatalogByProjectPath[projectPath] ?? []) : []
const detailById = useStore(useShallow((s) => s.skillsDetailsById));
const userSkills = useStore(useShallow((s) => s.skillsUserCatalog));
const projectSkills = useStore(
useShallow((s) => (projectPath ? (s.skillsProjectCatalogByProjectPath[projectPath] ?? []) : []))
);
const [createOpen, setCreateOpen] = useState(false);
const [editOpen, setEditOpen] = useState(false);

View file

@ -6,12 +6,13 @@
import { Fragment } from 'react';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { PaneResizeHandle } from './PaneResizeHandle';
import { PaneView } from './PaneView';
export const PaneContainer = (): React.JSX.Element => {
const panes = useStore((s) => s.paneLayout.panes);
const panes = useStore(useShallow((s) => s.paneLayout.panes));
return (
<div id="pane-container" className="flex flex-1 overflow-hidden">

View file

@ -6,6 +6,7 @@
import { useCallback, useEffect, useState } from 'react';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
interface PaneResizeHandleProps {
leftPaneId: string;
@ -15,7 +16,7 @@ interface PaneResizeHandleProps {
export const PaneResizeHandle = ({ leftPaneId }: PaneResizeHandleProps): React.JSX.Element => {
const [isResizing, setIsResizing] = useState(false);
const resizePanes = useStore((s) => s.resizePanes);
const paneLayout = useStore((s) => s.paneLayout);
const paneLayout = useStore(useShallow((s) => s.paneLayout));
const handleMouseMove = useCallback(
(e: MouseEvent) => {

View file

@ -76,17 +76,19 @@ export const SortableTab = ({
)
);
const teamColorSet = useStore((s) => {
if (tab.type !== 'team' || !tab.teamName) return null;
const team = s.teamByName[tab.teamName];
const explicitColor =
team?.color ??
(s.selectedTeamName === tab.teamName ? s.selectedTeamData?.config.color : undefined);
if (explicitColor) return getTeamColorSet(explicitColor);
// Fallback: deterministic color derived from display name
const displayName = team?.displayName ?? tab.label;
return nameColorSet(displayName);
});
const teamColorSet = useStore(
useShallow((s) => {
if (tab.type !== 'team' || !tab.teamName) return null;
const team = s.teamByName[tab.teamName];
const explicitColor =
team?.color ??
(s.selectedTeamName === tab.teamName ? s.selectedTeamData?.config.color : undefined);
if (explicitColor) return getTeamColorSet(explicitColor);
// Fallback: deterministic color derived from display name
const displayName = team?.displayName ?? tab.label;
return nameColorSet(displayName);
})
);
const activeBorderColor = teamColorSet?.border ?? 'var(--color-accent, #6366f1)';
const { attributes, listeners, setNodeRef, transform, transition, isDragging } = useSortable({

View file

@ -26,6 +26,7 @@ import { useFullScreen } from '@renderer/hooks/useFullScreen';
import { useKeyboardShortcuts } from '@renderer/hooks/useKeyboardShortcuts';
import { useZoomFactor } from '@renderer/hooks/useZoomFactor';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { CliInstallWarningBanner } from '../common/CliInstallWarningBanner';
import { UpdateBanner } from '../common/UpdateBanner';
@ -54,7 +55,7 @@ export const TabbedLayout = (): React.JSX.Element => {
: getTrafficLightPaddingForZoom(zoomFactor);
// --- DnD state (lifted from PaneContainer) ---
const panes = useStore((s) => s.paneLayout.panes);
const panes = useStore(useShallow((s) => s.paneLayout.panes));
const [activeTab, setActiveTab] = useState<Tab | null>(null);
const sensors = useSensors(

View file

@ -1,6 +1,7 @@
import { useMemo } from 'react';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { computeTakeaways } from '@renderer/utils/reportAssessments';
import { analyzeSession } from '@renderer/utils/sessionAnalyzer';
@ -25,11 +26,13 @@ interface SessionReportTabProps {
export const SessionReportTab = ({ tab }: SessionReportTabProps) => {
// Find session data from any session tab with matching sessionId
const sessionDetail = useStore((s) => {
const allTabs = s.paneLayout.panes.flatMap((p) => p.tabs);
const sourceTab = allTabs.find((t) => t.type === 'session' && t.sessionId === tab.sessionId);
return sourceTab ? s.tabSessionData[sourceTab.id]?.sessionDetail : null;
});
const sessionDetail = useStore(
useShallow((s) => {
const allTabs = s.paneLayout.panes.flatMap((p) => p.tabs);
const sourceTab = allTabs.find((t) => t.type === 'session' && t.sessionId === tab.sessionId);
return sourceTab ? s.tabSessionData[sourceTab.id]?.sessionDetail : null;
})
);
const report = useMemo(
() => (sessionDetail ? analyzeSession(sessionDetail) : null),

View file

@ -6,6 +6,7 @@ import { Popover, PopoverContent, PopoverTrigger } from '@renderer/components/ui
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { getTeamColorSet } from '@renderer/constants/teamColors';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { nameColorSet } from '@renderer/utils/projectColor';
import { formatNextRun, getCronDescription } from '@renderer/utils/scheduleFormatters';
import {
@ -68,7 +69,7 @@ const ScheduleListItem = ({
}: ScheduleListItemProps): React.JSX.Element => {
const [expanded, setExpanded] = useState(false);
const [selectedRun, setSelectedRun] = useState<ScheduleRun | null>(null);
const runs = useStore((s) => s.scheduleRuns[schedule.id] ?? []);
const runs = useStore(useShallow((s) => s.scheduleRuns[schedule.id] ?? []));
const runsLoading = useStore((s) => s.scheduleRunsLoading[schedule.id] ?? false);
const fetchRunHistory = useStore((s) => s.fetchRunHistory);
@ -240,15 +241,29 @@ const ScheduleListItem = ({
// =============================================================================
export const SchedulesView = (): React.JSX.Element => {
const schedules = useStore((s) => s.schedules);
const schedulesLoading = useStore((s) => s.schedulesLoading);
const fetchSchedules = useStore((s) => s.fetchSchedules);
const pauseSchedule = useStore((s) => s.pauseSchedule);
const resumeSchedule = useStore((s) => s.resumeSchedule);
const deleteSchedule = useStore((s) => s.deleteSchedule);
const triggerNow = useStore((s) => s.triggerNow);
const openTeamTab = useStore((s) => s.openTeamTab);
const teamByName = useStore((s) => s.teamByName);
const {
schedules,
schedulesLoading,
fetchSchedules,
pauseSchedule,
resumeSchedule,
deleteSchedule,
triggerNow,
openTeamTab,
teamByName,
} = useStore(
useShallow((s) => ({
schedules: s.schedules,
schedulesLoading: s.schedulesLoading,
fetchSchedules: s.fetchSchedules,
pauseSchedule: s.pauseSchedule,
resumeSchedule: s.resumeSchedule,
deleteSchedule: s.deleteSchedule,
triggerNow: s.triggerNow,
openTeamTab: s.openTeamTab,
teamByName: s.teamByName,
}))
);
/** Resolve team color dot style for a given team name */
const getTeamColor = useCallback(

View file

@ -7,6 +7,7 @@ import { useEffect, useState } from 'react';
import { useStore } from '@renderer/store';
import { Loader2 } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { useSettingsConfig, useSettingsHandlers } from './hooks';
import {
@ -19,8 +20,12 @@ import { type SettingsSection, SettingsTabs } from './SettingsTabs';
export const SettingsView = (): React.JSX.Element | null => {
const [activeSection, setActiveSection] = useState<SettingsSection>('general');
const pendingSettingsSection = useStore((s) => s.pendingSettingsSection);
const clearPendingSettingsSection = useStore((s) => s.clearPendingSettingsSection);
const { pendingSettingsSection, clearPendingSettingsSection } = useStore(
useShallow((s) => ({
pendingSettingsSection: s.pendingSettingsSection,
clearPendingSettingsSection: s.clearPendingSettingsSection,
}))
);
// Consume pending section (avoid setState during render)
useEffect(() => {

View file

@ -8,6 +8,7 @@ import { api, isElectronMode } from '@renderer/api';
import appIcon from '@renderer/favicon.png';
import { useStore } from '@renderer/store';
import { CheckCircle, Code2, Download, FileEdit, Loader2, RefreshCw, Upload } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { SettingsSectionHeader } from '../components';
@ -32,9 +33,13 @@ export const AdvancedSection = ({
const isElectron = useMemo(() => isElectronMode(), []);
const [version, setVersion] = useState<string>('');
const [configEditorOpen, setConfigEditorOpen] = useState(false);
const updateStatus = useStore((s) => s.updateStatus);
const availableVersion = useStore((s) => s.availableVersion);
const checkForUpdates = useStore((s) => s.checkForUpdates);
const { updateStatus, availableVersion, checkForUpdates } = useStore(
useShallow((s) => ({
updateStatus: s.updateStatus,
availableVersion: s.availableVersion,
checkForUpdates: s.checkForUpdates,
}))
);
// Auto-revert "not-available" / "error" status back to idle after a brief display
const revertTimerRef = useRef<ReturnType<typeof setTimeout>>(undefined);

View file

@ -14,6 +14,7 @@ import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { api } from '@renderer/api';
import { useStore } from '@renderer/store';
import { Loader2, Monitor, Server, Wifi, WifiOff } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { SettingRow } from '../components/SettingRow';
import { SettingsSectionHeader } from '../components/SettingsSectionHeader';
@ -37,16 +38,31 @@ const authMethodOptions: readonly { value: SshAuthMethod; label: string }[] = [
];
export const ConnectionSection = (): React.JSX.Element => {
const connectionState = useStore((s) => s.connectionState);
const connectedHost = useStore((s) => s.connectedHost);
const connectionError = useStore((s) => s.connectionError);
const connectSsh = useStore((s) => s.connectSsh);
const disconnectSsh = useStore((s) => s.disconnectSsh);
const testConnection = useStore((s) => s.testConnection);
const sshConfigHosts = useStore((s) => s.sshConfigHosts);
const fetchSshConfigHosts = useStore((s) => s.fetchSshConfigHosts);
const lastSshConfig = useStore((s) => s.lastSshConfig);
const loadLastConnection = useStore((s) => s.loadLastConnection);
const {
connectionState,
connectedHost,
connectionError,
connectSsh,
disconnectSsh,
testConnection,
sshConfigHosts,
fetchSshConfigHosts,
lastSshConfig,
loadLastConnection,
} = useStore(
useShallow((s) => ({
connectionState: s.connectionState,
connectedHost: s.connectedHost,
connectionError: s.connectionError,
connectSsh: s.connectSsh,
disconnectSsh: s.disconnectSsh,
testConnection: s.testConnection,
sshConfigHosts: s.sshConfigHosts,
fetchSshConfigHosts: s.fetchSshConfigHosts,
lastSshConfig: s.lastSshConfig,
loadLastConnection: s.loadLastConnection,
}))
);
// Form state
const [host, setHost] = useState('');

View file

@ -12,6 +12,7 @@ import { useStore } from '@renderer/store';
import { getFullResetState } from '@renderer/store/utils/stateResetHelpers';
import { AGENT_LANGUAGE_OPTIONS, resolveLanguageName } from '@shared/utils/agentLanguage';
import { Check, Copy, FolderOpen, Laptop, Loader2, RotateCcw } from 'lucide-react';
import { useShallow } from 'zustand/react/shallow';
import { SettingRow, SettingsSectionHeader, SettingsToggle } from '../components';
@ -50,9 +51,13 @@ export const GeneralSection = ({
const [copied, setCopied] = useState(false);
// Claude Root state
const connectionMode = useStore((s) => s.connectionMode);
const fetchProjects = useStore((s) => s.fetchProjects);
const fetchRepositoryGroups = useStore((s) => s.fetchRepositoryGroups);
const { connectionMode, fetchProjects, fetchRepositoryGroups } = useStore(
useShallow((s) => ({
connectionMode: s.connectionMode,
fetchProjects: s.fetchProjects,
fetchRepositoryGroups: s.fetchRepositoryGroups,
}))
);
const [claudeRootInfo, setClaudeRootInfo] = useState<ClaudeRootInfo | null>(null);
const [updatingClaudeRoot, setUpdatingClaudeRoot] = useState(false);

View file

@ -5,6 +5,7 @@ import { getTeamColorSet } from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useUnreadCommentCount } from '@renderer/hooks/useUnreadCommentCount';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { buildMemberColorMap, REVIEW_STATE_DISPLAY } from '@renderer/utils/memberHelpers';
import { nameColorSet } from '@renderer/utils/projectColor';
import { projectColor } from '@renderer/utils/projectColor';
@ -78,7 +79,7 @@ export const SidebarTaskItem = ({
getDisplaySubject,
}: SidebarTaskItemProps): React.JSX.Element => {
const openGlobalTaskDetail = useStore((s) => s.openGlobalTaskDetail);
const teamMembers = useStore((s) => s.teamByName[task.teamName]?.members);
const teamMembers = useStore(useShallow((s) => s.teamByName[task.teamName]?.members));
const unreadCount = useUnreadCommentCount(task.teamName, task.id, task.comments);
const { isLight } = useTheme();

View file

@ -1,4 +1,5 @@
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatDistanceToNowStrict } from 'date-fns';
import { ExternalLink, Square, Terminal } from 'lucide-react';
@ -25,7 +26,7 @@ function formatShortTime(date: Date): string {
export const ProcessesSection = (): React.JSX.Element | null => {
const teamName = useStore((s) => s.selectedTeamName);
const data = useStore((s) => s.selectedTeamData);
const data = useStore(useShallow((s) => s.selectedTeamData));
if (!teamName || !data?.processes?.length) return null;

View file

@ -4,6 +4,7 @@ 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 { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { buildMemberColorMap, REVIEW_STATE_DISPLAY } from '@renderer/utils/memberHelpers';
import { linkifyTaskIdsInMarkdown } from '@renderer/utils/taskReferenceUtils';
import { getTaskKanbanColumn } from '@shared/utils/reviewState';
@ -69,10 +70,14 @@ export const TaskTooltip = ({
children,
side = 'top',
}: TaskTooltipProps): React.JSX.Element => {
const selectedTeamName = useStore((s) => s.selectedTeamName);
const selectedTeamData = useStore((s) => s.selectedTeamData);
const globalTasks = useStore((s) => s.globalTasks);
const teamByName = useStore((s) => s.teamByName);
const { selectedTeamName, selectedTeamData, globalTasks, teamByName } = useStore(
useShallow((s) => ({
selectedTeamName: s.selectedTeamName,
selectedTeamData: s.selectedTeamData,
globalTasks: s.globalTasks,
teamByName: s.teamByName,
}))
);
const task = useMemo(() => {
if (teamName && selectedTeamName === teamName) {

View file

@ -457,6 +457,9 @@ export const TeamDetailView = ({
messagesPanelWidth,
setMessagesPanelMode,
setMessagesPanelWidth,
selectReviewFile,
pendingReviewRequest,
setPendingReviewRequest,
} = useStore(
useShallow((s) => ({
data: s.selectedTeamData,
@ -506,6 +509,9 @@ export const TeamDetailView = ({
messagesPanelWidth: s.messagesPanelWidth,
setMessagesPanelMode: s.setMessagesPanelMode,
setMessagesPanelWidth: s.setMessagesPanelWidth,
selectReviewFile: s.selectReviewFile,
pendingReviewRequest: s.pendingReviewRequest,
setPendingReviewRequest: s.setPendingReviewRequest,
}))
);
@ -961,10 +967,6 @@ export const TeamDetailView = ({
}
}, [teamName, refreshTeamData]);
const selectReviewFile = useStore((s) => s.selectReviewFile);
const pendingReviewRequest = useStore((s) => s.pendingReviewRequest);
const setPendingReviewRequest = useStore((s) => s.setPendingReviewRequest);
// Pick up pending review request from GlobalTaskDetailDialog
useEffect(() => {
if (!pendingReviewRequest) return;

View file

@ -224,6 +224,8 @@ export const TeamListView = (): React.JSX.Element => {
fetchTeams,
openTeamTab,
deleteTeam,
restoreTeam,
permanentlyDeleteTeam,
projects,
globalTasks,
fetchAllTasks,
@ -232,6 +234,7 @@ export const TeamListView = (): React.JSX.Element => {
selectedRepositoryId,
selectedWorktreeId,
activeProjectId,
branchByPath,
} = useStore(
useShallow((s) => ({
teams: s.teams,
@ -250,6 +253,7 @@ export const TeamListView = (): React.JSX.Element => {
selectedRepositoryId: s.selectedRepositoryId,
selectedWorktreeId: s.selectedWorktreeId,
activeProjectId: s.activeProjectId,
branchByPath: s.branchByPath,
}))
);
const {
@ -432,10 +436,6 @@ export const TeamListView = (): React.JSX.Element => {
[filteredTeams]
);
useBranchSync(teamPaths, { live: false });
const branchByPath = useStore((s) => s.branchByPath);
const restoreTeam = useStore((s) => s.restoreTeam);
const permanentlyDeleteTeam = useStore((s) => s.permanentlyDeleteTeam);
const handleDeleteTeam = useCallback(
(teamName: string, isDraft: boolean, e: React.MouseEvent) => {

View file

@ -3,6 +3,7 @@ import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'
import { getTeamColorSet, getThemedBadge } from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { shortenDisplayPath } from '@renderer/utils/pathDisplay';
import { highlightLines } from '@renderer/utils/syntaxHighlighter';
import { AlertTriangle, FileText, MessageCircleQuestion, Search, Terminal } from 'lucide-react';
@ -141,12 +142,23 @@ function useElapsed(receivedAt: string): number {
const RESPOND_TIMEOUT_MS = 10_000;
export const ToolApprovalSheet: React.FC = () => {
const pendingApprovals = useStore((s) => s.pendingApprovals);
const respondToToolApproval = useStore((s) => s.respondToToolApproval);
const updateToolApprovalSettings = useStore((s) => s.updateToolApprovalSettings);
const teams = useStore((s) => s.teams);
const selectedTeamName = useStore((s) => s.selectedTeamName);
const selectedTeamData = useStore((s) => s.selectedTeamData);
const {
pendingApprovals,
respondToToolApproval,
updateToolApprovalSettings,
teams,
selectedTeamName,
selectedTeamData,
} = useStore(
useShallow((s) => ({
pendingApprovals: s.pendingApprovals,
respondToToolApproval: s.respondToToolApproval,
updateToolApprovalSettings: s.updateToolApprovalSettings,
teams: s.teams,
selectedTeamName: s.selectedTeamName,
selectedTeamData: s.selectedTeamData,
}))
);
const { isLight } = useTheme();
const current: ToolApprovalRequest | undefined = pendingApprovals[0];
@ -606,7 +618,7 @@ const ToolInputPreview = ({
// ---------------------------------------------------------------------------
const TimeoutProgress = ({ receivedAt }: { receivedAt: string }): React.JSX.Element | null => {
const settings = useStore((s) => s.toolApprovalSettings);
const settings = useStore(useShallow((s) => s.toolApprovalSettings));
const elapsed = useElapsed(receivedAt);
if (settings.timeoutAction === 'wait') return null;

View file

@ -17,6 +17,7 @@ import {
import { getTeamColorSet, getThemedBorder } from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import {
getMessageTypeLabel,
getStructuredMessageSummary,
@ -601,8 +602,8 @@ export const ActivityItem = memo(
const summaryText = useMemo(() => extractMarkdownPlainText(rawSummary), [rawSummary]);
// Permission request status icon (check/x/clock)
const pendingApprovals = useStore((s) => s.pendingApprovals);
const resolvedApprovals = useStore((s) => s.resolvedApprovals);
const pendingApprovals = useStore(useShallow((s) => s.pendingApprovals));
const resolvedApprovals = useStore(useShallow((s) => s.resolvedApprovals));
const permissionIcon = useMemo(() => {
if (!structured) return null;
const type = typeof structured.type === 'string' ? structured.type : null;

View file

@ -2,6 +2,7 @@ import { CARD_BG, CARD_BORDER_STYLE, CARD_ICON_MUTED } from '@renderer/constants
import { getTeamColorSet, getThemedBadge } from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatAgentRole } from '@renderer/utils/formatAgentRole';
import {
agentAvatarUrl,
@ -33,7 +34,7 @@ export const PendingRepliesBlock = ({
onMemberClick,
}: PendingRepliesBlockProps): React.JSX.Element | null => {
const { isLight } = useTheme();
const pendingApprovals = useStore((s) => s.pendingApprovals);
const pendingApprovals = useStore(useShallow((s) => s.pendingApprovals));
const colorMap = buildMemberColorMap(members);
const memberPending = Object.entries(pendingRepliesByMember)
.map(([name, sentAtMs]) => ({

View file

@ -23,6 +23,7 @@ import { useDraftPersistence } from '@renderer/hooks/useDraftPersistence';
import { useFileListCacheWarmer } from '@renderer/hooks/useFileListCacheWarmer';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatAgentRole } from '@renderer/utils/formatAgentRole';
import { buildMemberColorMap } from '@renderer/utils/memberHelpers';
import { normalizePath } from '@renderer/utils/pathNormalize';
@ -119,8 +120,12 @@ export const LaunchTeamDialog = (props: LaunchTeamDialogProps): React.JSX.Elemen
// Team name: always present for launch mode, may be absent in schedule mode (standalone page)
const propsTeamName = props.teamName ?? '';
const [selectedTeamName, setSelectedTeamName] = useState('');
const teamByName = useStore((s) => s.teamByName);
const openDashboard = useStore((s) => s.openDashboard);
const { teamByName, openDashboard } = useStore(
useShallow((s) => ({
teamByName: s.teamByName,
openDashboard: s.openDashboard,
}))
);
const teamOptions = useMemo(
() =>
Object.values(teamByName)
@ -401,7 +406,7 @@ export const LaunchTeamDialog = (props: LaunchTeamDialogProps): React.JSX.Elemen
// Shared effects: projects
// ---------------------------------------------------------------------------
const repositoryGroups = useStore((s) => s.repositoryGroups);
const repositoryGroups = useStore(useShallow((s) => s.repositoryGroups));
useEffect(() => {
if (!open) return;
@ -490,7 +495,7 @@ export const LaunchTeamDialog = (props: LaunchTeamDialogProps): React.JSX.Elemen
// Mention suggestions (shared — from props in launch, from store in schedule)
// ---------------------------------------------------------------------------
const storeMembers = useStore((s) => s.selectedTeamData?.members ?? []);
const storeMembers = useStore(useShallow((s) => s.selectedTeamData?.members ?? []));
const members = isLaunch ? props.members : storeMembers;
const colorMap = useMemo(() => buildMemberColorMap(members), [members]);

View file

@ -9,6 +9,7 @@ import {
SelectValue,
} from '@renderer/components/ui/select';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { ChevronDown, ChevronRight, Settings } from 'lucide-react';
import type { ToolApprovalSettings, ToolApprovalTimeoutAction } from '@shared/types';
@ -42,7 +43,7 @@ export const ToolApprovalSettingsContent: React.FC<{
teamName?: string;
}> = ({ expanded, teamName }) => {
const [localSeconds, setLocalSeconds] = useState<string>('');
const settings = useStore((s) => s.toolApprovalSettings);
const settings = useStore(useShallow((s) => s.toolApprovalSettings));
const rawUpdateSettings = useStore((s) => s.updateToolApprovalSettings);
const updateSettings = useCallback(
(patch: Partial<ToolApprovalSettings>) => rawUpdateSettings(patch, teamName),

View file

@ -1,3 +1,5 @@
import { memo } from 'react';
import { Badge } from '@renderer/components/ui/badge';
import { cn } from '@renderer/lib/utils';
@ -15,7 +17,7 @@ interface KanbanColumnProps {
children: React.ReactNode;
}
export const KanbanColumn = ({
export const KanbanColumn = memo(function KanbanColumn({
title,
count,
icon,
@ -27,7 +29,7 @@ export const KanbanColumn = ({
headerDragClassName,
headerAccessory,
children,
}: KanbanColumnProps): React.JSX.Element => {
}: KanbanColumnProps): React.JSX.Element {
return (
<section
className={cn(
@ -64,4 +66,4 @@ export const KanbanColumn = ({
</div>
</section>
);
};
});

View file

@ -8,6 +8,7 @@ import {
} from '@renderer/constants/teamColors';
import { useTheme } from '@renderer/hooks/useTheme';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatAgentRole } from '@renderer/utils/formatAgentRole';
import {
agentAvatarUrl,
@ -44,14 +45,19 @@ export const MemberHoverCard = ({
children,
}: MemberHoverCardProps): React.JSX.Element => {
const { isLight } = useTheme();
const member = useStore((s) => s.selectedTeamData?.members.find((m) => m.name === name) ?? null);
const isTeamAlive = useStore((s) => s.selectedTeamData?.isAlive);
const teamName = useStore((s) => s.selectedTeamName);
const leadActivity: LeadActivityState | undefined = useStore((s) =>
teamName ? s.leadActivityByTeam[teamName] : undefined
const { member, isTeamAlive, teamName, leadActivity, openMemberProfile, tasks } = useStore(
useShallow((s) => {
const tn = s.selectedTeamName;
return {
member: s.selectedTeamData?.members.find((m) => m.name === name) ?? null,
isTeamAlive: s.selectedTeamData?.isAlive,
teamName: tn,
leadActivity: tn ? s.leadActivityByTeam[tn] : undefined,
openMemberProfile: s.openMemberProfile,
tasks: s.selectedTeamData?.tasks,
};
})
);
const openMemberProfile = useStore((s) => s.openMemberProfile);
const tasks = useStore((s) => s.selectedTeamData?.tasks);
if (!member) {
return <>{children}</>;

View file

@ -1,4 +1,4 @@
import { useCallback, useEffect, useRef, useState } from 'react';
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { buildMemberColorMap } from '@renderer/utils/memberHelpers';
import { isLeadAgentType, isLeadMember } from '@shared/utils/leadDetection';
@ -67,15 +67,19 @@ export const MemberList = ({
}, [handleResize]);
const gridClass = isWide ? 'grid grid-cols-2 gap-1' : 'grid grid-cols-1 gap-1';
const activeMembers = members
.filter((m) => !m.removedAt)
.sort((a, b) => {
if (isLeadMember(a)) return -1;
if (isLeadMember(b)) return 1;
return 0;
});
const removedMembers = members.filter((m) => m.removedAt);
const colorMap = buildMemberColorMap(members);
const activeMembers = useMemo(
() =>
members
.filter((m) => !m.removedAt)
.sort((a, b) => {
if (isLeadMember(a)) return -1;
if (isLeadMember(b)) return 1;
return 0;
}),
[members]
);
const removedMembers = useMemo(() => members.filter((m) => m.removedAt), [members]);
const colorMap = useMemo(() => buildMemberColorMap(members), [members]);
if (members.length === 0) {
return (
@ -85,17 +89,24 @@ export const MemberList = ({
);
}
// Pre-compute reviewer→task map to avoid O(n×m) scan per member
const reviewTaskByMember = useMemo(() => {
const result = new Map<string, TeamTaskWithKanban>();
if (!taskMap) return result;
for (const task of taskMap.values()) {
if (task.reviewer && (task.reviewState === 'review' || task.kanbanColumn === 'review')) {
result.set(task.reviewer, task);
}
}
return result;
}, [taskMap]);
const renderCard = (member: ResolvedTeamMember, isRemoved: boolean): React.JSX.Element => {
const currentTask =
member.currentTaskId && taskMap ? (taskMap.get(member.currentTaskId) ?? null) : null;
const reviewTask = taskMap
? (Array.from(taskMap.values()).find(
(task) =>
task.reviewer === member.name &&
task.id !== member.currentTaskId &&
(task.reviewState === 'review' || task.kanbanColumn === 'review')
) ?? null)
: null;
const reviewCandidate = reviewTaskByMember.get(member.name) ?? null;
const reviewTask =
reviewCandidate && reviewCandidate.id !== member.currentTaskId ? reviewCandidate : null;
const awaitingReply = Boolean(pendingRepliesByMember?.[member.name]);
const spawnEntry = memberSpawnStatuses?.get(member.name);
return (

View file

@ -1,5 +1,8 @@
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { useStore } from '@renderer/store';
import { useTabIdOptional } from '@renderer/contexts/useTabUIContext';
import { api } from '@renderer/api';
import { MemberExecutionLog } from '@renderer/components/team/members/MemberExecutionLog';
import {
@ -107,6 +110,11 @@ export const MemberLogsTab = ({
showLeadPreview = false,
onPreviewOnlineChange,
}: MemberLogsTabProps): React.JSX.Element => {
// Visibility check: skip polling when tab is hidden (display:none) to avoid OOM
const tabId = useTabIdOptional();
const activeTabId = useStore((s) => s.activeTabId);
const isTabActive = tabId ? activeTabId === tabId : true; // default true when no tab context (e.g. standalone dialog)
const MIN_REFRESH_VISIBLE_MS = 250;
const intervalsKey = useMemo(
() => (taskWorkIntervals ? JSON.stringify(taskWorkIntervals) : ''),
@ -475,14 +483,14 @@ export const MemberLogsTab = ({
void load();
const interval = shouldAutoRefresh ? setInterval(() => void load(), 5000) : null;
const interval = shouldAutoRefresh && isTabActive ? setInterval(() => void load(), 5000) : null;
return () => {
cancelled = true;
if (interval) clearInterval(interval);
};
// eslint-disable-next-line react-hooks/exhaustive-deps -- intervalsKey + taskSince drive refresh; deps intentionally minimal to avoid refetch loops
}, [teamName, memberName, taskId, taskOwner, taskStatus, intervalsKey, taskSince]);
}, [teamName, memberName, taskId, taskOwner, taskStatus, intervalsKey, taskSince, isTabActive]);
const fetchDetailForLog = useCallback(
async (
@ -537,7 +545,7 @@ export const MemberLogsTab = ({
if (!previewLog) return;
const shouldAutoRefreshPreview = taskStatus === 'in_progress' || previewLog.isOngoing;
if (!shouldAutoRefreshPreview) return;
if (!shouldAutoRefreshPreview || !isTabActive) return;
let cancelled = false;
const interval = setInterval(async () => {
@ -566,12 +574,14 @@ export const MemberLogsTab = ({
shouldShowPreview,
taskStatus,
intervalsKey,
isTabActive,
]);
useEffect(() => {
const shouldAutoRefreshSummary = taskId != null && taskStatus === 'in_progress';
if (!expandedLogSummary) return;
if (!shouldAutoRefreshSummary && !expandedLogSummary.isOngoing) return;
if (!isTabActive) return;
let cancelled = false;
@ -604,6 +614,7 @@ export const MemberLogsTab = ({
taskId,
taskStatus,
intervalsKey,
isTabActive,
]);
const handleExpand = useCallback(

View file

@ -14,6 +14,7 @@ import { useTaskSuggestions } from '@renderer/hooks/useTaskSuggestions';
import { useTeamSuggestions } from '@renderer/hooks/useTeamSuggestions';
import { cn } from '@renderer/lib/utils';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { isTeamProvisioningActive } from '@renderer/store/slices/teamSlice';
import { serializeChipsWithText } from '@renderer/types/inlineChip';
import { formatAgentRole } from '@renderer/utils/formatAgentRole';
@ -104,7 +105,7 @@ export const MessageComposer = ({
const [selectedTeam, setSelectedTeam] = useState<string | null>(null);
const [teamSelectorOpen, setTeamSelectorOpen] = useState(false);
const [aliveTeams, setAliveTeams] = useState<Set<string>>(new Set());
const allCrossTeamTargets = useStore((s) => s.crossTeamTargets);
const allCrossTeamTargets = useStore(useShallow((s) => s.crossTeamTargets));
const fetchCrossTeamTargets = useStore((s) => s.fetchCrossTeamTargets);
useEffect(() => {

View file

@ -6,6 +6,7 @@ import { Checkbox } from '@renderer/components/ui/checkbox';
import { Popover, PopoverContent, PopoverTrigger } from '@renderer/components/ui/popover';
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { buildMemberColorMap } from '@renderer/utils/memberHelpers';
import { Filter } from 'lucide-react';
@ -67,7 +68,7 @@ export const MessagesFilterPopover = ({
}
}, [open, filter.from, filter.to, filter.showNoise]);
const members = useStore((s) => s.selectedTeamData?.members ?? []);
const members = useStore(useShallow((s) => s.selectedTeamData?.members ?? []));
const colorMap = useMemo(() => buildMemberColorMap(members), [members]);
const fromOptions = useMemo(() => collectFromOptions(messages), [messages]);

View file

@ -7,6 +7,7 @@ import { useStableTeamMentionMeta } from '@renderer/hooks/useStableTeamMentionMe
import { useTeamMessagesExpanded } from '@renderer/hooks/useTeamMessagesExpanded';
import { useTeamMessagesRead } from '@renderer/hooks/useTeamMessagesRead';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { filterTeamMessages } from '@renderer/utils/teamMessageFiltering';
import { toMessageKey } from '@renderer/utils/teamMessageKey';
import {
@ -105,13 +106,25 @@ export const MessagesPanel = memo(function MessagesPanel({
onRestartTeam,
onTaskIdClick,
}: MessagesPanelProps): React.JSX.Element {
const sendTeamMessage = useStore((s) => s.sendTeamMessage);
const sendCrossTeamMessage = useStore((s) => s.sendCrossTeamMessage);
const sendingMessage = useStore((s) => s.sendingMessage);
const sendMessageError = useStore((s) => s.sendMessageError);
const lastSendMessageResult = useStore((s) => s.lastSendMessageResult);
const teams = useStore((s) => s.teams);
const openTeamTab = useStore((s) => s.openTeamTab);
const {
sendTeamMessage,
sendCrossTeamMessage,
sendingMessage,
sendMessageError,
lastSendMessageResult,
teams,
openTeamTab,
} = useStore(
useShallow((s) => ({
sendTeamMessage: s.sendTeamMessage,
sendCrossTeamMessage: s.sendCrossTeamMessage,
sendingMessage: s.sendingMessage,
sendMessageError: s.sendMessageError,
lastSendMessageResult: s.lastSendMessageResult,
teams: s.teams,
openTeamTab: s.openTeamTab,
}))
);
const composerTextareaRef = useRef<HTMLTextAreaElement | null>(null);
const sidebarScrollRef = useRef<HTMLDivElement | null>(null);

View file

@ -10,6 +10,7 @@ import {
DialogTitle,
} from '@renderer/components/ui/dialog';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { AlertTriangle, Clock, Loader2, Terminal } from 'lucide-react';
import { CliLogsRichView } from '../CliLogsRichView';
@ -72,11 +73,13 @@ export const ScheduleRunLogDialog = ({
onClose,
}: ScheduleRunLogDialogProps): React.JSX.Element => {
// Read live run data from store — falls back to initial prop if not found
const liveRun = useStore((s) => {
if (!initialRun) return null;
const runs = s.scheduleRuns[scheduleId] ?? [];
return runs.find((r) => r.id === initialRun.id) ?? initialRun;
});
const liveRun = useStore(
useShallow((s) => {
if (!initialRun) return null;
const runs = s.scheduleRuns[scheduleId] ?? [];
return runs.find((r) => r.id === initialRun.id) ?? initialRun;
})
);
const run = liveRun ?? initialRun;
const [logs, setLogs] = useState<{ stdout: string; stderr: string } | null>(null);

View file

@ -4,6 +4,7 @@ import { Button } from '@renderer/components/ui/button';
import { Popover, PopoverContent, PopoverTrigger } from '@renderer/components/ui/popover';
import { Tooltip, TooltipContent, TooltipTrigger } from '@renderer/components/ui/tooltip';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { formatNextRun, getCronDescription } from '@renderer/utils/scheduleFormatters';
import {
ChevronDown,
@ -57,7 +58,7 @@ const ScheduleRow = ({
}: ScheduleRowProps): React.JSX.Element => {
const [expanded, setExpanded] = useState(false);
const [selectedRun, setSelectedRun] = useState<ScheduleRun | null>(null);
const runs = useStore((s) => s.scheduleRuns[schedule.id] ?? []);
const runs = useStore(useShallow((s) => s.scheduleRuns[schedule.id] ?? []));
const runsLoading = useStore((s) => s.scheduleRunsLoading[schedule.id] ?? false);
const fetchRunHistory = useStore((s) => s.fetchRunHistory);
@ -207,17 +208,22 @@ const ScheduleRow = ({
// =============================================================================
export const ScheduleSection = ({ teamName }: ScheduleSectionProps): React.JSX.Element => {
const schedules = useStore((s) => s.schedules.filter((sch) => sch.teamName === teamName));
const pauseSchedule = useStore((s) => s.pauseSchedule);
const resumeSchedule = useStore((s) => s.resumeSchedule);
const deleteSchedule = useStore((s) => s.deleteSchedule);
const triggerNow = useStore((s) => s.triggerNow);
const { schedules, pauseSchedule, resumeSchedule, deleteSchedule, triggerNow, fetchSchedules } =
useStore(
useShallow((s) => ({
schedules: s.schedules.filter((sch) => sch.teamName === teamName),
pauseSchedule: s.pauseSchedule,
resumeSchedule: s.resumeSchedule,
deleteSchedule: s.deleteSchedule,
triggerNow: s.triggerNow,
fetchSchedules: s.fetchSchedules,
}))
);
const [dialogOpen, setDialogOpen] = useState(false);
const [editingSchedule, setEditingSchedule] = useState<Schedule | null>(null);
// Fetch schedules on mount
const fetchSchedules = useStore((s) => s.fetchSchedules);
useEffect(() => {
void fetchSchedules();
}, [fetchSchedules]);

View file

@ -1,6 +1,7 @@
import { createContext, useContext, useId, useLayoutEffect, useState } from 'react';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import {
removeTeamSidebarHost,
@ -33,10 +34,12 @@ export const TeamSidebarHost = ({
}: TeamSidebarHostProps): React.JSX.Element => {
const hostId = useId();
const [element, setElement] = useState<HTMLDivElement | null>(null);
const { messagesPanelMode, messagesPanelWidth } = useStore((s) => ({
messagesPanelMode: s.messagesPanelMode,
messagesPanelWidth: s.messagesPanelWidth,
}));
const { messagesPanelMode, messagesPanelWidth } = useStore(
useShallow((s) => ({
messagesPanelMode: s.messagesPanelMode,
messagesPanelWidth: s.messagesPanelWidth,
}))
);
const snapshot = useTeamSidebarPortalSnapshot();
const isVisible = messagesPanelMode === 'sidebar';
const isOwner = isVisible && snapshot.activeHostIdByTeam[teamName] === hostId;

View file

@ -7,6 +7,7 @@ import { useMemo } from 'react';
import { KanbanTaskCard } from '@renderer/components/team/kanban/KanbanTaskCard';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import type { GraphNode } from '@claude-teams/agent-graph';
import type { KanbanColumnId, TeamTask, TeamTaskWithKanban } from '@shared/types';
@ -84,9 +85,13 @@ export const GraphTaskCard = ({
}: GraphTaskCardProps): React.JSX.Element => {
const taskId = node.domainRef.kind === 'task' ? node.domainRef.taskId : '';
const task = useStore((s) => s.selectedTeamData?.tasks.find((t) => t.id === taskId));
const tasks = useStore((s) => s.selectedTeamData?.tasks ?? []);
const members = useStore((s) => s.selectedTeamData?.members ?? []);
const { task, tasks, members } = useStore(
useShallow((s) => ({
task: s.selectedTeamData?.tasks.find((t) => t.id === taskId),
tasks: s.selectedTeamData?.tasks ?? [],
members: s.selectedTeamData?.members ?? [],
}))
);
const taskMap = useMemo(() => {
const map = new Map<string, TeamTask>();

View file

@ -6,6 +6,7 @@
*/
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import type { CliInstallationStatus } from '@shared/types';
@ -33,20 +34,39 @@ export function useCliInstaller(): {
installCli: () => void;
isBusy: boolean;
} {
const cliStatus = useStore((s) => s.cliStatus);
const cliStatusLoading = useStore((s) => s.cliStatusLoading);
const cliStatusError = useStore((s) => s.cliStatusError);
const installerState = useStore((s) => s.cliInstallerState);
const downloadProgress = useStore((s) => s.cliDownloadProgress);
const downloadTransferred = useStore((s) => s.cliDownloadTransferred);
const downloadTotal = useStore((s) => s.cliDownloadTotal);
const installerError = useStore((s) => s.cliInstallerError);
const installerDetail = useStore((s) => s.cliInstallerDetail);
const installerRawChunks = useStore((s) => s.cliInstallerRawChunks);
const completedVersion = useStore((s) => s.cliCompletedVersion);
const fetchCliStatus = useStore((s) => s.fetchCliStatus);
const invalidateCliStatus = useStore((s) => s.invalidateCliStatus);
const installCli = useStore((s) => s.installCli);
const {
cliStatus,
cliStatusLoading,
cliStatusError,
installerState,
downloadProgress,
downloadTransferred,
downloadTotal,
installerError,
installerDetail,
installerRawChunks,
completedVersion,
fetchCliStatus,
invalidateCliStatus,
installCli,
} = useStore(
useShallow((s) => ({
cliStatus: s.cliStatus,
cliStatusLoading: s.cliStatusLoading,
cliStatusError: s.cliStatusError,
installerState: s.cliInstallerState,
downloadProgress: s.cliDownloadProgress,
downloadTransferred: s.cliDownloadTransferred,
downloadTotal: s.cliDownloadTotal,
installerError: s.cliInstallerError,
installerDetail: s.cliInstallerDetail,
installerRawChunks: s.cliInstallerRawChunks,
completedVersion: s.cliCompletedVersion,
fetchCliStatus: s.fetchCliStatus,
invalidateCliStatus: s.invalidateCliStatus,
installCli: s.installCli,
}))
);
const isBusy = installerState !== 'idle' && installerState !== 'error';

View file

@ -65,7 +65,7 @@ export function useTabUI(): UseTabUIReturn {
// Subscribe to tabUIStates MAP directly for reactivity
// This ensures re-renders when any tab state changes
const tabUIStates = useStore((s) => s.tabUIStates);
const tabUIStates = useStore(useShallow((s) => s.tabUIStates));
// Get the current tab's state (derived from subscribed state)
const tabState = useMemo(() => {

View file

@ -1,6 +1,7 @@
import { useMemo } from 'react';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import { createEncodedTaskReference } from '@renderer/utils/taskReferenceUtils';
import { getTaskDisplayId } from '@shared/utils/taskIdentity';
@ -56,10 +57,14 @@ function isVisibleTask(task: TeamTaskWithKanban | GlobalTask): boolean {
}
export function useTaskSuggestions(currentTeamName: string | null): UseTaskSuggestionsResult {
const globalTasks = useStore((s) => s.globalTasks);
const selectedTeamName = useStore((s) => s.selectedTeamName);
const selectedTeamData = useStore((s) => s.selectedTeamData);
const teamByName = useStore((s) => s.teamByName);
const { globalTasks, selectedTeamName, selectedTeamData, teamByName } = useStore(
useShallow((s) => ({
globalTasks: s.globalTasks,
selectedTeamName: s.selectedTeamName,
selectedTeamData: s.selectedTeamData,
teamByName: s.teamByName,
}))
);
const suggestions = useMemo<MentionSuggestion[]>(() => {
const tasks: TaskWithTeamContext[] = [];

View file

@ -12,6 +12,7 @@ import { useCallback, useEffect, useMemo, useState } from 'react';
import { api } from '@renderer/api';
import { useStore } from '@renderer/store';
import { useShallow } from 'zustand/react/shallow';
import type { MentionSuggestion } from '@renderer/types/mention';
@ -26,7 +27,7 @@ export interface UseTeamSuggestionsResult {
* @param currentTeamName - The current team name to exclude from suggestions
*/
export function useTeamSuggestions(currentTeamName: string | null): UseTeamSuggestionsResult {
const teams = useStore((s) => s.teams);
const teams = useStore(useShallow((s) => s.teams));
const [aliveTeams, setAliveTeams] = useState<Set<string>>(new Set());
const [loading, setLoading] = useState(false);

View file

@ -175,6 +175,19 @@ export function initializeNotificationListeners(): () => void {
const TEAM_PRESENCE_REFRESH_THROTTLE_MS = 400;
const TEAM_LIST_REFRESH_THROTTLE_MS = 2000;
const GLOBAL_TASKS_REFRESH_THROTTLE_MS = 500;
/** Cap a Map at maxSize by clearing oldest entries (FIFO via insertion order). */
const capTimerMap = (map: Map<string, ReturnType<typeof setTimeout>>, maxSize: number): void => {
if (map.size <= maxSize) return;
const excess = map.size - maxSize;
let cleared = 0;
for (const [key, value] of map) {
if (cleared >= excess) break;
clearTimeout(value);
map.delete(key);
cleared++;
}
};
const buildToolActivityTimerKey = (
teamName: string,
memberName: string,
@ -209,6 +222,7 @@ export function initializeNotificationListeners(): () => void {
cb();
}, delayMs);
toolActivityTimers.set(key, timer);
capTimerMap(toolActivityTimers, 200);
};
const clearToolActivityTimersForTeam = (teamName: string): void => {
for (const [key, timer] of toolActivityTimers.entries()) {
@ -318,6 +332,16 @@ export function initializeNotificationListeners(): () => void {
return;
}
// Cleanup cursors for teams that no longer exist (prevent unbounded growth)
if (inProgressChangePresenceCursorByTeam.size > 50) {
const teamNames = new Set(useStore.getState().teams.map((t) => t.teamName));
for (const key of inProgressChangePresenceCursorByTeam.keys()) {
if (!teamNames.has(key)) {
inProgressChangePresenceCursorByTeam.delete(key);
}
}
}
const candidateTasks = selectedTeamData.tasks.filter((task) => {
if (task.status !== 'in_progress') {
return false;
@ -376,6 +400,7 @@ export function initializeNotificationListeners(): () => void {
void state.refreshSessionInPlace(projectId, sessionId);
}, SESSION_REFRESH_DEBOUNCE_MS);
pendingSessionRefreshTimers.set(key, timer);
capTimerMap(pendingSessionRefreshTimers, 50);
};
const scheduleProjectRefresh = (projectId: string): void => {
@ -389,6 +414,7 @@ export function initializeNotificationListeners(): () => void {
void state.refreshSessionsInPlace(projectId);
}, PROJECT_REFRESH_DEBOUNCE_MS);
pendingProjectRefreshTimers.set(projectId, timer);
capTimerMap(pendingProjectRefreshTimers, 20);
};
// Listen for new notifications from main process
@ -933,6 +959,7 @@ export function initializeNotificationListeners(): () => void {
void current.refreshTeamData(event.teamName);
}, TEAM_REFRESH_THROTTLE_MS);
teamRefreshTimers.set(event.teamName, timer);
capTimerMap(teamRefreshTimers, 20);
return;
}
@ -949,6 +976,7 @@ export function initializeNotificationListeners(): () => void {
void current.refreshSelectedTeamChangePresence(event.teamName);
}, TEAM_PRESENCE_REFRESH_THROTTLE_MS);
teamPresenceRefreshTimers.set(event.teamName, timer);
capTimerMap(teamPresenceRefreshTimers, 20);
return;
}
@ -984,6 +1012,7 @@ export function initializeNotificationListeners(): () => void {
void current.refreshTeamData(event.teamName);
}, TEAM_REFRESH_THROTTLE_MS);
teamRefreshTimers.set(event.teamName, timer);
capTimerMap(teamRefreshTimers, 20);
});
if (typeof cleanup === 'function') {