Compare commits

..

3 Commits

Author SHA1 Message Date
Tim
cf4ca89e19 feat: show channel unread indicator 2025-08-23 02:05:39 +08:00
tim
15cba0c96e fix: 支持显示最后一条消息 2025-08-23 01:57:33 +08:00
Tim
98a79acad9 Merge pull request #702 from nagisa77/codex/add-multi-tab-support-to-message-box
feat: add channel support
2025-08-23 01:31:26 +08:00
7 changed files with 138 additions and 65 deletions

View File

@@ -18,14 +18,14 @@ public class ChannelInitializer implements CommandLineRunner {
chat.setChannel(true); chat.setChannel(true);
chat.setName("吹水群"); chat.setName("吹水群");
chat.setDescription("吹水聊天"); chat.setDescription("吹水聊天");
chat.setAvatar("/default-avatar.svg"); chat.setAvatar("https://openisle-1307107697.cos.accelerate.myqcloud.com/dynamic_assert/32647273e2334d14adfd4a6ce9db0643.jpeg");
conversationRepository.save(chat); conversationRepository.save(chat);
MessageConversation tech = new MessageConversation(); MessageConversation tech = new MessageConversation();
tech.setChannel(true); tech.setChannel(true);
tech.setName("技术讨论群"); tech.setName("技术讨论群");
tech.setDescription("讨论技术相关话题"); tech.setDescription("讨论技术相关话题");
tech.setAvatar("/default-avatar.svg"); tech.setAvatar("https://openisle-1307107697.cos.accelerate.myqcloud.com/dynamic_assert/5edde9a5864e471caa32491dbcdaa8b2.png");
conversationRepository.save(tech); conversationRepository.save(tech);
} }
} }

View File

@@ -121,6 +121,7 @@ public class SecurityConfig {
.requestMatchers(HttpMethod.GET, "/api/reaction-types").permitAll() .requestMatchers(HttpMethod.GET, "/api/reaction-types").permitAll()
.requestMatchers(HttpMethod.GET, "/api/activities/**").permitAll() .requestMatchers(HttpMethod.GET, "/api/activities/**").permitAll()
.requestMatchers(HttpMethod.GET, "/api/sitemap.xml").permitAll() .requestMatchers(HttpMethod.GET, "/api/sitemap.xml").permitAll()
.requestMatchers(HttpMethod.GET, "/api/channels").permitAll()
.requestMatchers(HttpMethod.GET, "/api/rss").permitAll() .requestMatchers(HttpMethod.GET, "/api/rss").permitAll()
.requestMatchers(HttpMethod.GET, "/api/point-goods").permitAll() .requestMatchers(HttpMethod.GET, "/api/point-goods").permitAll()
.requestMatchers(HttpMethod.POST, "/api/point-goods").permitAll() .requestMatchers(HttpMethod.POST, "/api/point-goods").permitAll()
@@ -156,7 +157,7 @@ public class SecurityConfig {
uri.startsWith("/api/search") || uri.startsWith("/api/users") || uri.startsWith("/api/search") || uri.startsWith("/api/users") ||
uri.startsWith("/api/reaction-types") || uri.startsWith("/api/config") || uri.startsWith("/api/reaction-types") || uri.startsWith("/api/config") ||
uri.startsWith("/api/activities") || uri.startsWith("/api/push/public-key") || uri.startsWith("/api/activities") || uri.startsWith("/api/push/public-key") ||
uri.startsWith("/api/point-goods") || uri.startsWith("/api/point-goods") || uri.startsWith("/api/channels") ||
uri.startsWith("/api/sitemap.xml") || uri.startsWith("/api/medals") || uri.startsWith("/api/sitemap.xml") || uri.startsWith("/api/medals") ||
uri.startsWith("/api/rss")); uri.startsWith("/api/rss"));

View File

@@ -10,6 +10,7 @@ public class ChannelDto {
private String name; private String name;
private String description; private String description;
private String avatar; private String avatar;
private MessageDto lastMessage;
private long memberCount; private long memberCount;
private boolean joined; private boolean joined;
private long unreadCount; private long unreadCount;

View File

@@ -6,7 +6,10 @@
<button class="menu-btn" ref="menuBtn" @click="$emit('toggle-menu')"> <button class="menu-btn" ref="menuBtn" @click="$emit('toggle-menu')">
<i class="fas fa-bars"></i> <i class="fas fa-bars"></i>
</button> </button>
<span v-if="isMobile && unreadMessageCount > 0" class="menu-unread-dot"></span> <span
v-if="isMobile && (messageUnreadCount > 0 || channelUnreadCount > 0)"
class="menu-unread-dot"
></span>
</div> </div>
<NuxtLink class="logo-container" :to="`/`" @click="refrechData"> <NuxtLink class="logo-container" :to="`/`" @click="refrechData">
<img <img
@@ -50,9 +53,10 @@
<ToolTip v-if="isLogin" content="站内信和频道" placement="bottom"> <ToolTip v-if="isLogin" content="站内信和频道" placement="bottom">
<div class="messages-icon" @click="goToMessages"> <div class="messages-icon" @click="goToMessages">
<i class="fas fa-comments"></i> <i class="fas fa-comments"></i>
<span v-if="unreadMessageCount > 0" class="unread-badge">{{ <span v-if="messageUnreadCount > 0" class="unread-badge">
unreadMessageCount {{ messageUnreadCount }}
}}</span> </span>
<span v-else-if="channelUnreadCount > 0" class="messages-unread-dot"></span>
</div> </div>
</ToolTip> </ToolTip>
@@ -102,7 +106,10 @@ const props = defineProps({
const isLogin = computed(() => authState.loggedIn) const isLogin = computed(() => authState.loggedIn)
const isMobile = useIsMobile() const isMobile = useIsMobile()
const { count: unreadMessageCount, fetchUnreadCount } = useUnreadCount() const { count: totalUnreadCount, channelUnreadCount, fetchUnreadCount } = useUnreadCount()
const messageUnreadCount = computed(() =>
Math.max(totalUnreadCount.value - channelUnreadCount.value, 0),
)
const avatar = ref('') const avatar = ref('')
const showSearch = ref(false) const showSearch = ref(false)
const searchDropdown = ref(null) const searchDropdown = ref(null)
@@ -413,6 +420,16 @@ onMounted(async () => {
box-sizing: border-box; box-sizing: border-box;
} }
.messages-unread-dot {
position: absolute;
top: -2px;
right: -4px;
width: 8px;
height: 8px;
border-radius: 50%;
background-color: #ff4d4f;
}
.rss-icon { .rss-icon {
animation: rss-glow 2s 3; animation: rss-glow 2s 3;
} }

View File

@@ -1,93 +1,123 @@
import { ref, watch, onMounted } from 'vue'; import { ref, watch } from 'vue'
import { useWebSocket } from './useWebSocket'; import { useWebSocket } from './useWebSocket'
import { getToken } from '~/utils/auth'; import { getToken } from '~/utils/auth'
const count = ref(0); const count = ref(0)
let isInitialized = false; const channelUnreadCount = ref(0)
let wsSubscription = null; let isInitialized = false
let wsSubscription = null
export function useUnreadCount() { export function useUnreadCount() {
const config = useRuntimeConfig(); const config = useRuntimeConfig()
const API_BASE_URL = config.public.apiBaseUrl; const API_BASE_URL = config.public.apiBaseUrl
const { subscribe, isConnected, connect } = useWebSocket(); const { subscribe, isConnected, connect } = useWebSocket()
const fetchUnreadCount = async () => { const fetchTotalUnreadCount = async () => {
const token = getToken(); const token = getToken()
if (!token) { if (!token) {
count.value = 0; count.value = 0
return; return
} }
try { try {
const response = await fetch(`${API_BASE_URL}/api/messages/unread-count`, { const response = await fetch(`${API_BASE_URL}/api/messages/unread-count`, {
headers: { Authorization: `Bearer ${token}` }, headers: { Authorization: `Bearer ${token}` },
}); })
if (response.ok) { if (response.ok) {
const data = await response.json(); const data = await response.json()
count.value = data; count.value = data
} }
} catch (error) { } catch (error) {
console.error('Failed to fetch unread count:', error); console.error('Failed to fetch unread count:', error)
} }
}; }
const fetchChannelUnreadCount = async () => {
const token = getToken()
if (!token) {
channelUnreadCount.value = 0
return
}
try {
const response = await fetch(`${API_BASE_URL}/api/channels`, {
headers: { Authorization: `Bearer ${token}` },
})
if (response.ok) {
const channels = await response.json()
channelUnreadCount.value = channels.reduce((sum, ch) => sum + (ch.unreadCount || 0), 0)
}
} catch (error) {
console.error('Failed to fetch channel unread count:', error)
}
}
const fetchUnreadCount = async () => {
await Promise.all([fetchTotalUnreadCount(), fetchChannelUnreadCount()])
}
const initialize = async () => { const initialize = async () => {
const token = getToken(); const token = getToken()
if (!token) { if (!token) {
count.value = 0; count.value = 0
return; channelUnreadCount.value = 0
return
} }
// 总是获取最新的未读数量 // 总是获取最新的未读数量
fetchUnreadCount(); fetchUnreadCount()
// 确保WebSocket连接 // 确保WebSocket连接
if (!isConnected.value) { if (!isConnected.value) {
connect(token); connect(token)
} }
// 设置WebSocket监听 // 设置WebSocket监听
await setupWebSocketListener(); await setupWebSocketListener()
}; }
const setupWebSocketListener = async () => { const setupWebSocketListener = async () => {
// 只有在还没有订阅的情况下才设置监听 // 只有在还没有订阅的情况下才设置监听
if (!wsSubscription) { if (!wsSubscription) {
watch(
watch(isConnected, (newValue) => { isConnected,
if (newValue && !wsSubscription) { (newValue) => {
const destination = `/user/queue/unread-count`; if (newValue && !wsSubscription) {
wsSubscription = subscribe(destination, (message) => { const destination = `/user/queue/unread-count`
const unreadCount = parseInt(message.body, 10); wsSubscription = subscribe(destination, (message) => {
if (!isNaN(unreadCount)) { const unreadCount = parseInt(message.body, 10)
count.value = unreadCount; if (!isNaN(unreadCount)) {
} count.value = unreadCount
}); fetchChannelUnreadCount()
} }
}, { immediate: true }); })
}
},
{ immediate: true },
)
} }
}; }
// 自动初始化逻辑 - 确保每次调用都能获取到未读数量并设置监听 // 自动初始化逻辑 - 确保每次调用都能获取到未读数量并设置监听
const token = getToken(); const token = getToken()
if (token) { if (token) {
if (!isInitialized) { if (!isInitialized) {
isInitialized = true; isInitialized = true
initialize(); // 完整初始化包括WebSocket监听 initialize() // 完整初始化包括WebSocket监听
} else { } else {
// 即使已经初始化也要确保获取最新的未读数量并确保WebSocket监听存在 // 即使已经初始化也要确保获取最新的未读数量并确保WebSocket监听存在
fetchUnreadCount(); fetchUnreadCount()
// 确保WebSocket连接和监听都存在 // 确保WebSocket连接和监听都存在
if (!isConnected.value) { if (!isConnected.value) {
connect(token); connect(token)
} }
setupWebSocketListener(); setupWebSocketListener()
} }
} }
return { return {
count, count,
channelUnreadCount,
fetchUnreadCount, fetchUnreadCount,
initialize, initialize,
}; }
} }

View File

@@ -22,8 +22,13 @@
</div> </div>
<BaseTimeline :items="messages"> <BaseTimeline :items="messages">
<template #item="{ item }"> <template #item="{ item }">
<div class="message-timestamp"> <div class="message-header">
{{ TimeManager.format(item.createdAt) }} <div class="user-name">
{{ item.sender.username }}
</div>
<div class="message-timestamp">
{{ TimeManager.format(item.createdAt) }}
</div>
</div> </div>
<div class="message-content"> <div class="message-content">
<div class="info-content-text" v-html="renderMarkdown(item.content)"></div> <div class="info-content-text" v-html="renderMarkdown(item.content)"></div>
@@ -448,10 +453,22 @@ onUnmounted(() => {
.message-timestamp { .message-timestamp {
font-size: 11px; font-size: 11px;
color: var(--text-color-secondary); color: var(--text-color-secondary);
margin-top: 5px;
opacity: 0.6; opacity: 0.6;
} }
.message-header {
display: flex;
flex-direction: row;
align-items: center;
gap: 10px;
}
.user-name {
font-size: 14px;
font-weight: 600;
color: var(--text-color);
}
.message-item.sent { .message-item.sent {
align-self: flex-end; align-self: flex-end;
flex-direction: row-reverse; flex-direction: row-reverse;

View File

@@ -97,7 +97,11 @@
<div class="message-time">成员 {{ ch.memberCount }}</div> <div class="message-time">成员 {{ ch.memberCount }}</div>
</div> </div>
<div class="last-message-row"> <div class="last-message-row">
<div class="last-message">{{ ch.description }}</div> <div class="last-message">
{{
ch.lastMessage ? stripMarkdownLength(ch.lastMessage.content, 100) : ch.description
}}
</div>
</div> </div>
</div> </div>
</div> </div>
@@ -262,8 +266,6 @@ function goToConversation(id) {
<style scoped> <style scoped>
.messages-container { .messages-container {
margin: 0 auto;
padding: 20px;
} }
.tabs { .tabs {
@@ -291,6 +293,8 @@ function goToConversation(id) {
.search-container { .search-container {
margin-bottom: 24px; margin-bottom: 24px;
margin-left: 20px;
margin-right: 20px;
} }
.messages-header { .messages-header {
@@ -330,6 +334,8 @@ function goToConversation(id) {
.conversation-item { .conversation-item {
display: flex; display: flex;
align-items: center; align-items: center;
margin-left: 20px;
margin-right: 20px;
padding: 8px 10px; padding: 8px 10px;
cursor: pointer; cursor: pointer;
transition: background-color 0.2s ease; transition: background-color 0.2s ease;
@@ -415,8 +421,9 @@ function goToConversation(id) {
/* 响应式设计 */ /* 响应式设计 */
@media (max-width: 768px) { @media (max-width: 768px) {
.messages-container { .conversation-item {
padding: 10px 10px; margin-left: 10px;
margin-right: 10px;
} }
.messages-title { .messages-title {