agent-ecosystem/src/renderer/components/settings/sections/NotificationsSection.tsx
iliya 6aec33ae33 feat: enhance notification and task management features
- Updated notification configuration to separate settings for lead and user inbox messages, improving user control over notifications.
- Enhanced the handling of inbox message notifications to respect individual inbox settings.
- Introduced new IPC methods for managing watched directories, allowing for more granular file monitoring.
- Improved task management by implementing work intervals for task status transitions, ensuring accurate tracking of task progress.
- Added validation for task creation and configuration, ensuring proper input handling for project paths and task properties.
2026-03-02 18:17:57 +02:00

232 lines
8.1 KiB
TypeScript

/**
* NotificationsSection - Notification settings including triggers and ignored repositories.
*/
import { api } from '@renderer/api';
import {
RepositoryDropdown,
SelectedRepositoryItem,
} from '@renderer/components/common/RepositoryDropdown';
import { ExternalLink } from 'lucide-react';
import { SettingRow, SettingsSectionHeader, SettingsSelect, SettingsToggle } from '../components';
import { NotificationTriggerSettings } from '../NotificationTriggerSettings';
import type { RepositoryDropdownItem, SafeConfig } from '../hooks/useSettingsConfig';
import type { NotificationTrigger } from '@renderer/types/data';
// Snooze duration options
const SNOOZE_OPTIONS = [
{ value: 15, label: '15 minutes' },
{ value: 30, label: '30 minutes' },
{ value: 60, label: '1 hour' },
{ value: 120, label: '2 hours' },
{ value: 240, label: '4 hours' },
{ value: -1, label: 'Until tomorrow' },
] as const;
interface NotificationsSectionProps {
readonly safeConfig: SafeConfig;
readonly saving: boolean;
readonly isSnoozed: boolean;
readonly ignoredRepositoryItems: RepositoryDropdownItem[];
readonly excludedRepositoryIds: string[];
readonly onNotificationToggle: (
key:
| 'enabled'
| 'soundEnabled'
| 'includeSubagentErrors'
| 'notifyOnLeadInbox'
| 'notifyOnUserInbox'
| 'notifyOnClarifications',
value: boolean
) => void;
readonly onSnooze: (minutes: number) => Promise<void>;
readonly onClearSnooze: () => Promise<void>;
readonly onAddIgnoredRepository: (item: RepositoryDropdownItem) => Promise<void>;
readonly onRemoveIgnoredRepository: (repositoryId: string) => Promise<void>;
readonly onAddTrigger: (trigger: Omit<NotificationTrigger, 'isBuiltin'>) => Promise<void>;
readonly onUpdateTrigger: (
triggerId: string,
updates: Partial<NotificationTrigger>
) => Promise<void>;
readonly onRemoveTrigger: (triggerId: string) => Promise<void>;
}
export const NotificationsSection = ({
safeConfig,
saving,
isSnoozed,
ignoredRepositoryItems,
excludedRepositoryIds,
onNotificationToggle,
onSnooze,
onClearSnooze,
onAddIgnoredRepository,
onRemoveIgnoredRepository,
onAddTrigger,
onUpdateTrigger,
onRemoveTrigger,
}: NotificationsSectionProps): React.JSX.Element => {
return (
<div>
{/* Task Completion Notifications */}
<SettingsSectionHeader title="Task Completion Notifications" />
<div
className="mb-4 rounded-lg border p-4"
style={{
borderColor: 'var(--color-border)',
backgroundColor: 'var(--color-surface-raised)',
}}
>
<p className="mb-3 text-sm" style={{ color: 'var(--color-text-secondary)' }}>
Get native OS notifications when Claude finishes tasks sounds, banners, and Dock/taskbar
badges. Works on macOS, Linux, and Windows.
</p>
<button
onClick={() =>
void api.openExternal('https://github.com/777genius/claude-notifications-go')
}
className="inline-flex items-center gap-1.5 rounded-md px-3 py-1.5 text-sm font-medium transition-colors hover:brightness-125"
style={{
backgroundColor: 'var(--color-border-emphasis)',
color: 'var(--color-text)',
}}
>
<ExternalLink className="size-3.5" />
Install claude-notifications-go plugin
</button>
</div>
{/* Notification Triggers */}
<NotificationTriggerSettings
triggers={safeConfig.notifications.triggers || []}
saving={saving}
onUpdateTrigger={onUpdateTrigger}
onAddTrigger={onAddTrigger}
onRemoveTrigger={onRemoveTrigger}
/>
{/* Notification Settings */}
<SettingsSectionHeader title="Notification Settings" />
<SettingRow
label="Enable System Notifications"
description="Show system notifications for errors and events"
>
<SettingsToggle
enabled={safeConfig.notifications.enabled}
onChange={(v) => onNotificationToggle('enabled', v)}
disabled={saving}
/>
</SettingRow>
<SettingRow label="Play sound" description="Play a sound when notifications appear">
<SettingsToggle
enabled={safeConfig.notifications.soundEnabled}
onChange={(v) => onNotificationToggle('soundEnabled', v)}
disabled={saving || !safeConfig.notifications.enabled}
/>
</SettingRow>
<SettingRow
label="Include subagent errors"
description="Detect and notify about errors in subagent sessions"
>
<SettingsToggle
enabled={safeConfig.notifications.includeSubagentErrors}
onChange={(v) => onNotificationToggle('includeSubagentErrors', v)}
disabled={saving || !safeConfig.notifications.enabled}
/>
</SettingRow>
<SettingRow
label="Lead inbox notifications"
description="Notify when teammates send messages to the team lead"
>
<SettingsToggle
enabled={safeConfig.notifications.notifyOnLeadInbox}
onChange={(v) => onNotificationToggle('notifyOnLeadInbox', v)}
disabled={saving || !safeConfig.notifications.enabled}
/>
</SettingRow>
<SettingRow
label="User inbox notifications"
description="Notify when teammates send messages to you"
>
<SettingsToggle
enabled={safeConfig.notifications.notifyOnUserInbox}
onChange={(v) => onNotificationToggle('notifyOnUserInbox', v)}
disabled={saving || !safeConfig.notifications.enabled}
/>
</SettingRow>
<SettingRow
label="Task clarification notifications"
description="Show native OS notifications when a task needs your input"
>
<SettingsToggle
enabled={safeConfig.notifications.notifyOnClarifications}
onChange={(v) => onNotificationToggle('notifyOnClarifications', v)}
disabled={saving || !safeConfig.notifications.enabled}
/>
</SettingRow>
<SettingRow
label="Snooze notifications"
description={
isSnoozed
? `Snoozed until ${new Date(safeConfig.notifications.snoozedUntil!).toLocaleTimeString()}`
: 'Temporarily pause notifications'
}
>
<div className="flex items-center gap-2">
{isSnoozed ? (
<button
onClick={onClearSnooze}
disabled={saving}
className={`rounded-md bg-red-500/10 px-3 py-1.5 text-sm font-medium text-red-400 transition-all duration-150 hover:bg-red-500/20 ${saving ? 'cursor-not-allowed opacity-50' : ''} `}
>
Clear Snooze
</button>
) : (
<SettingsSelect
value={0}
options={[{ value: 0, label: 'Select duration...' }, ...SNOOZE_OPTIONS]}
onChange={(v) => v !== 0 && onSnooze(v)}
disabled={saving || !safeConfig.notifications.enabled}
dropUp
/>
)}
</div>
</SettingRow>
<SettingsSectionHeader title="Ignored Repositories" />
<p className="mb-3 text-xs" style={{ color: 'var(--color-text-muted)' }}>
Notifications from these repositories will be ignored
</p>
{ignoredRepositoryItems.length > 0 ? (
<div className="mb-3">
{ignoredRepositoryItems.map((item) => (
<SelectedRepositoryItem
key={item.id}
item={item}
onRemove={() => onRemoveIgnoredRepository(item.id)}
disabled={saving}
/>
))}
</div>
) : (
<div
className="mb-3 rounded-md border border-dashed py-3 text-center"
style={{ borderColor: 'var(--color-border)' }}
>
<p className="text-sm" style={{ color: 'var(--color-text-muted)' }}>
No repositories ignored
</p>
</div>
)}
<RepositoryDropdown
onSelect={onAddIgnoredRepository}
excludeIds={excludedRepositoryIds}
placeholder="Select repository to ignore..."
disabled={saving}
dropUp
/>
</div>
);
};