[dev]update notifCenter

This commit is contained in:
susie-laptop 2025-04-17 10:07:08 -04:00
parent c6368b1b4b
commit 97203b4bcd
4 changed files with 114 additions and 24 deletions

View File

@ -40,7 +40,7 @@ export default function NotificationCenter({ show, onClose }) {
// WebSocket // WebSocket
useEffect(() => { useEffect(() => {
// WebSocket // WebSocket
if (isAuthenticated && !isConnected) { if (isAuthenticated) {
initWebSocket() initWebSocket()
.then(() => { .then(() => {
dispatch(setWebSocketConnected(true)); dispatch(setWebSocketConnected(true));
@ -59,14 +59,7 @@ export default function NotificationCenter({ show, onClose }) {
}); });
} }
// WebSocket }, [isAuthenticated, dispatch]);
return () => {
if (isConnected) {
closeWebSocket();
dispatch(setWebSocketConnected(false));
}
};
}, [isAuthenticated, isConnected, dispatch]);
// //
useEffect(() => { useEffect(() => {
@ -125,6 +118,7 @@ export default function NotificationCenter({ show, onClose }) {
}; };
const handleViewDetail = (notification) => { const handleViewDetail = (notification) => {
onClose();
navigate('/permissions'); navigate('/permissions');
// setSelectedRequest(notification); // setSelectedRequest(notification);
// setShowSlideOver(true); // setShowSlideOver(true);

View File

@ -88,7 +88,6 @@ export default function HeaderWithNav() {
<button <button
className='btn btn-link text-dark p-0' className='btn btn-link text-dark p-0'
onClick={() => setShowNotifications(!showNotifications)} onClick={() => setShowNotifications(!showNotifications)}
title={isConnected ? '通知服务已连接' : '通知服务未连接'}
> >
<SvgIcon className={'bell'} /> <SvgIcon className={'bell'} />
{unreadCount > 0 && ( {unreadCount > 0 && (
@ -96,16 +95,6 @@ export default function HeaderWithNav() {
{unreadCount > 99 ? '99+' : unreadCount} {unreadCount > 99 ? '99+' : unreadCount}
</span> </span>
)} )}
{!isConnected && (
<span className='position-absolute bottom-0 end-0'>
<span
className='badge bg-secondary'
style={{ fontSize: '0.6rem', transform: 'translate(25%, 25%)' }}
>
<i className='bi bi-x-circle-fill'></i>
</span>
</span>
)}
</button> </button>
</div> </div>
<div className='flex-shrink-0 dropdown'> <div className='flex-shrink-0 dropdown'>

View File

@ -1,4 +1,8 @@
import { addNotification, markNotificationAsRead } from '../store/notificationCenter/notificationCenter.slice'; import {
addNotification,
markNotificationAsRead,
setWebSocketConnected,
} from '../store/notificationCenter/notificationCenter.slice';
import store from '../store/store'; // 修改为默认导出 import store from '../store/store'; // 修改为默认导出
import CryptoJS from 'crypto-js'; import CryptoJS from 'crypto-js';
@ -38,6 +42,7 @@ export const initWebSocket = () => {
let token = ''; let token = '';
if (!encryptedToken) { if (!encryptedToken) {
console.error('No token found, cannot connect to notification service'); console.error('No token found, cannot connect to notification service');
store.dispatch(setWebSocketConnected(false));
reject(new Error('No token found')); reject(new Error('No token found'));
return; return;
} }
@ -53,6 +58,9 @@ export const initWebSocket = () => {
console.log('WebSocket connection established'); console.log('WebSocket connection established');
reconnectAttempts = 0; // 连接成功后重置重连计数器 reconnectAttempts = 0; // 连接成功后重置重连计数器
// 更新Redux中的连接状态
store.dispatch(setWebSocketConnected(true));
// 订阅通知频道 // 订阅通知频道
subscribeToNotifications(); subscribeToNotifications();
@ -79,6 +87,8 @@ export const initWebSocket = () => {
// 错误处理 // 错误处理
socket.onerror = (error) => { socket.onerror = (error) => {
console.error('WebSocket error:', error); console.error('WebSocket error:', error);
// 更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
reject(error); reject(error);
}; };
@ -86,6 +96,9 @@ export const initWebSocket = () => {
socket.onclose = (event) => { socket.onclose = (event) => {
console.log(`WebSocket connection closed: ${event.code} ${event.reason}`); console.log(`WebSocket connection closed: ${event.code} ${event.reason}`);
// 更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
// 清除ping定时器 // 清除ping定时器
if (pingInterval) clearInterval(pingInterval); if (pingInterval) clearInterval(pingInterval);
@ -99,15 +112,21 @@ export const initWebSocket = () => {
console.log('Attempting to reconnect WebSocket...'); console.log('Attempting to reconnect WebSocket...');
initWebSocket().catch((err) => { initWebSocket().catch((err) => {
console.error('Failed to reconnect WebSocket:', err); console.error('Failed to reconnect WebSocket:', err);
// 重连失败时更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
}); });
}, RECONNECT_DELAY); }, RECONNECT_DELAY);
} else { } else {
console.log('Maximum reconnection attempts reached. Giving up.'); console.log('Maximum reconnection attempts reached. Giving up.');
// 达到最大重连次数时更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
} }
} }
}; };
} catch (error) { } catch (error) {
console.error('Error initializing WebSocket:', error); console.error('Error initializing WebSocket:', error);
// 更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
reject(error); reject(error);
} }
}); });
@ -173,6 +192,9 @@ export const closeWebSocket = () => {
clearInterval(pingInterval); clearInterval(pingInterval);
pingInterval = null; pingInterval = null;
} }
// 更新Redux中的连接状态
store.dispatch(setWebSocketConnected(false));
}; };
/** /**

View File

@ -9,7 +9,7 @@ export const fetchNotifications = createAsyncThunk(
async (_, { rejectWithValue }) => { async (_, { rejectWithValue }) => {
try { try {
const response = await get('/notifications/'); const response = await get('/notifications/');
return response; return processNotification(response);
} catch (error) { } catch (error) {
const errorMessage = error.response?.data?.message || 'Failed to fetch notifications'; const errorMessage = error.response?.data?.message || 'Failed to fetch notifications';
return rejectWithValue(errorMessage); return rejectWithValue(errorMessage);
@ -48,3 +48,88 @@ export const markAllNotificationsRead = createAsyncThunk(
} }
} }
); );
/**
* 处理通知数据转换为应用内通知格式
* @param {Object|Array} data 通知数据或通知数组
* @returns {Object|Array} 处理后的通知数据
*/
export const processNotification = (data) => {
// 处理数组类型的通知数据
if (Array.isArray(data)) {
return data.map((item) => processNotificationItem(item));
}
// 处理WebSocket格式的通知数据 (带有data字段的对象)
if (data && data.data) {
return processNotificationItem(data.data);
}
// 处理单个通知对象
return processNotificationItem(data);
};
/**
* 处理单个通知项
* @param {Object} notification 单个通知数据
* @returns {Object} 处理后的通知数据
*/
const processNotificationItem = (notification) => {
// 确保我们有一个有效的通知对象
if (!notification) return null;
// 提取通知数据兼容不同的API格式
const notificationData = notification.data || notification;
// 设置图标
let icon = 'bi-info-circle';
const type = notificationData.category || notificationData.type;
if (type === 'system') {
icon = 'bi-info-circle';
} else if (type === 'permission' || type === 'permission_request') {
icon = 'bi-shield';
}
// 计算时间显示
const createdAt = new Date(notificationData.created_at);
const now = new Date();
// 检查是否是今天
const isToday =
createdAt.getDate() === now.getDate() &&
createdAt.getMonth() === now.getMonth() &&
createdAt.getFullYear() === now.getFullYear();
// 如果是今天,只显示时间;否则显示年月日和时间
let timeDisplay;
if (isToday) {
timeDisplay = createdAt.toLocaleTimeString('zh-CN', {
hour: '2-digit',
minute: '2-digit',
});
} else {
timeDisplay = createdAt.toLocaleString('zh-CN', {
year: 'numeric',
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit',
});
}
return {
id: notificationData.id,
type: type,
icon,
title: notificationData.title,
content: notificationData.content,
time: timeDisplay,
hasDetail: true,
isRead: notificationData.is_read,
created_at: notificationData.created_at,
sender: notificationData.sender,
receiver: notificationData.receiver,
related_resource: notificationData.related_resource,
metadata: notificationData.metadata || {},
};
};