mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-06-13 09:42:02 +08:00
* feat: introduce notification system and claude notifications * fix(sw): prevent caching of API requests and WebSocket upgrades * default to false for webpush notifications and translations for the button * fix: notifications orchestrator and add a notification when first enabled * fix: remove unused state update and dependency in settings controller hook * fix: show notifications settings tab * fix: add notifications for response completion for all providers * feat: show session name in notification and don't reload tab on clicking --- the notification --------- Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com> Co-authored-by: Haileyesus <something@gmail.com>
198 lines
6.2 KiB
TypeScript
198 lines
6.2 KiB
TypeScript
import { useEffect, useRef } from 'react';
|
|
import { useNavigate, useParams } from 'react-router-dom';
|
|
import { useTranslation } from 'react-i18next';
|
|
import Sidebar from '../sidebar/view/Sidebar';
|
|
import MainContent from '../main-content/view/MainContent';
|
|
import { useWebSocket } from '../../contexts/WebSocketContext';
|
|
import { useDeviceSettings } from '../../hooks/useDeviceSettings';
|
|
import { useSessionProtection } from '../../hooks/useSessionProtection';
|
|
import { useProjectsState } from '../../hooks/useProjectsState';
|
|
import MobileNav from './MobileNav';
|
|
|
|
export default function AppContent() {
|
|
const navigate = useNavigate();
|
|
const { sessionId } = useParams<{ sessionId?: string }>();
|
|
const { t } = useTranslation('common');
|
|
const { isMobile } = useDeviceSettings({ trackPWA: false });
|
|
const { ws, sendMessage, latestMessage, isConnected } = useWebSocket();
|
|
const wasConnectedRef = useRef(false);
|
|
|
|
const {
|
|
activeSessions,
|
|
processingSessions,
|
|
markSessionAsActive,
|
|
markSessionAsInactive,
|
|
markSessionAsProcessing,
|
|
markSessionAsNotProcessing,
|
|
replaceTemporarySession,
|
|
} = useSessionProtection();
|
|
|
|
const {
|
|
selectedProject,
|
|
selectedSession,
|
|
activeTab,
|
|
sidebarOpen,
|
|
isLoadingProjects,
|
|
isInputFocused,
|
|
externalMessageUpdate,
|
|
setActiveTab,
|
|
setSidebarOpen,
|
|
setIsInputFocused,
|
|
setShowSettings,
|
|
openSettings,
|
|
refreshProjectsSilently,
|
|
sidebarSharedProps,
|
|
} = useProjectsState({
|
|
sessionId,
|
|
navigate,
|
|
latestMessage,
|
|
isMobile,
|
|
activeSessions,
|
|
});
|
|
|
|
useEffect(() => {
|
|
// Expose a non-blocking refresh for chat/session flows.
|
|
// Full loading refreshes are still available through direct fetchProjects calls.
|
|
window.refreshProjects = refreshProjectsSilently;
|
|
|
|
return () => {
|
|
if (window.refreshProjects === refreshProjectsSilently) {
|
|
delete window.refreshProjects;
|
|
}
|
|
};
|
|
}, [refreshProjectsSilently]);
|
|
|
|
useEffect(() => {
|
|
window.openSettings = openSettings;
|
|
|
|
return () => {
|
|
if (window.openSettings === openSettings) {
|
|
delete window.openSettings;
|
|
}
|
|
};
|
|
}, [openSettings]);
|
|
|
|
useEffect(() => {
|
|
if (typeof navigator === 'undefined' || !('serviceWorker' in navigator)) {
|
|
return undefined;
|
|
}
|
|
|
|
const handleServiceWorkerMessage = (event: MessageEvent) => {
|
|
const message = event.data;
|
|
if (!message || message.type !== 'notification:navigate') {
|
|
return;
|
|
}
|
|
|
|
if (typeof message.provider === 'string' && message.provider.trim()) {
|
|
localStorage.setItem('selected-provider', message.provider);
|
|
}
|
|
|
|
setActiveTab('chat');
|
|
setSidebarOpen(false);
|
|
void refreshProjectsSilently();
|
|
|
|
if (typeof message.sessionId === 'string' && message.sessionId) {
|
|
navigate(`/session/${message.sessionId}`);
|
|
return;
|
|
}
|
|
|
|
navigate('/');
|
|
};
|
|
|
|
navigator.serviceWorker.addEventListener('message', handleServiceWorkerMessage);
|
|
|
|
return () => {
|
|
navigator.serviceWorker.removeEventListener('message', handleServiceWorkerMessage);
|
|
};
|
|
}, [navigate, refreshProjectsSilently, setActiveTab, setSidebarOpen]);
|
|
|
|
// Permission recovery: query pending permissions on WebSocket reconnect or session change
|
|
useEffect(() => {
|
|
const isReconnect = isConnected && !wasConnectedRef.current;
|
|
|
|
if (isReconnect) {
|
|
wasConnectedRef.current = true;
|
|
} else if (!isConnected) {
|
|
wasConnectedRef.current = false;
|
|
}
|
|
|
|
if (isConnected && selectedSession?.id) {
|
|
sendMessage({
|
|
type: 'get-pending-permissions',
|
|
sessionId: selectedSession.id
|
|
});
|
|
}
|
|
}, [isConnected, selectedSession?.id, sendMessage]);
|
|
|
|
return (
|
|
<div className="fixed inset-0 flex bg-background">
|
|
{!isMobile ? (
|
|
<div className="h-full flex-shrink-0 border-r border-border/50">
|
|
<Sidebar {...sidebarSharedProps} />
|
|
</div>
|
|
) : (
|
|
<div
|
|
className={`fixed inset-0 z-50 flex transition-all duration-150 ease-out ${sidebarOpen ? 'visible opacity-100' : 'invisible opacity-0'
|
|
}`}
|
|
>
|
|
<button
|
|
className="fixed inset-0 bg-background/60 backdrop-blur-sm transition-opacity duration-150 ease-out"
|
|
onClick={(event) => {
|
|
event.stopPropagation();
|
|
setSidebarOpen(false);
|
|
}}
|
|
onTouchStart={(event) => {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
setSidebarOpen(false);
|
|
}}
|
|
aria-label={t('versionUpdate.ariaLabels.closeSidebar')}
|
|
/>
|
|
<div
|
|
className={`relative h-full w-[85vw] max-w-sm transform border-r border-border/40 bg-card transition-transform duration-150 ease-out sm:w-80 ${sidebarOpen ? 'translate-x-0' : '-translate-x-full'
|
|
}`}
|
|
onClick={(event) => event.stopPropagation()}
|
|
onTouchStart={(event) => event.stopPropagation()}
|
|
>
|
|
<Sidebar {...sidebarSharedProps} />
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
<div className={`flex min-w-0 flex-1 flex-col ${isMobile ? 'pb-mobile-nav' : ''}`}>
|
|
<MainContent
|
|
selectedProject={selectedProject}
|
|
selectedSession={selectedSession}
|
|
activeTab={activeTab}
|
|
setActiveTab={setActiveTab}
|
|
ws={ws}
|
|
sendMessage={sendMessage}
|
|
latestMessage={latestMessage}
|
|
isMobile={isMobile}
|
|
onMenuClick={() => setSidebarOpen(true)}
|
|
isLoading={isLoadingProjects}
|
|
onInputFocusChange={setIsInputFocused}
|
|
onSessionActive={markSessionAsActive}
|
|
onSessionInactive={markSessionAsInactive}
|
|
onSessionProcessing={markSessionAsProcessing}
|
|
onSessionNotProcessing={markSessionAsNotProcessing}
|
|
processingSessions={processingSessions}
|
|
onReplaceTemporarySession={replaceTemporarySession}
|
|
onNavigateToSession={(targetSessionId: string) => navigate(`/session/${targetSessionId}`)}
|
|
onShowSettings={() => setShowSettings(true)}
|
|
externalMessageUpdate={externalMessageUpdate}
|
|
/>
|
|
</div>
|
|
|
|
{isMobile && (
|
|
<MobileNav
|
|
activeTab={activeTab}
|
|
setActiveTab={setActiveTab}
|
|
isInputFocused={isInputFocused}
|
|
/>
|
|
)}
|
|
|
|
</div>
|
|
);
|
|
}
|