- 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.
232 lines
8.1 KiB
TypeScript
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>
|
|
);
|
|
};
|