feat(frontend): add loading empty and error states for chat operations
This commit is contained in:
@@ -26,6 +26,8 @@ export default function ChatPage() {
|
|||||||
const [socketState, setSocketState] = useState<'offline' | 'online'>('offline')
|
const [socketState, setSocketState] = useState<'offline' | 'online'>('offline')
|
||||||
const [onlineCount, setOnlineCount] = useState(0)
|
const [onlineCount, setOnlineCount] = useState(0)
|
||||||
const [typingUsers, setTypingUsers] = useState<string[]>([])
|
const [typingUsers, setTypingUsers] = useState<string[]>([])
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [error, setError] = useState('')
|
||||||
const [seqFrom, setSeqFrom] = useState('1')
|
const [seqFrom, setSeqFrom] = useState('1')
|
||||||
const [seqTo, setSeqTo] = useState('999999')
|
const [seqTo, setSeqTo] = useState('999999')
|
||||||
const [limit, setLimit] = useState('50')
|
const [limit, setLimit] = useState('50')
|
||||||
@@ -86,6 +88,9 @@ export default function ChatPage() {
|
|||||||
|
|
||||||
async function pullMessages() {
|
async function pullMessages() {
|
||||||
if (!channelId) return
|
if (!channelId) return
|
||||||
|
setLoading(true)
|
||||||
|
setError('')
|
||||||
|
try {
|
||||||
const res = await getApiClient().get(`/channels/${channelId}/messages`, {
|
const res = await getApiClient().get(`/channels/${channelId}/messages`, {
|
||||||
params: {
|
params: {
|
||||||
seq_from: Number(seqFrom || '1'),
|
seq_from: Number(seqFrom || '1'),
|
||||||
@@ -95,31 +100,51 @@ export default function ChatPage() {
|
|||||||
})
|
})
|
||||||
setMessages(res.data.items ?? [])
|
setMessages(res.data.items ?? [])
|
||||||
setPageInfo(res.data.page ?? null)
|
setPageInfo(res.data.page ?? null)
|
||||||
|
} catch {
|
||||||
|
setError('消息拉取失败')
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function sendMessage() {
|
async function sendMessage() {
|
||||||
if (!channelId || !content.trim()) return
|
if (!channelId || !content.trim()) return
|
||||||
|
setError('')
|
||||||
|
try {
|
||||||
await getApiClient().post(`/channels/${channelId}/messages`, {
|
await getApiClient().post(`/channels/${channelId}/messages`, {
|
||||||
content,
|
content,
|
||||||
authorUserId: 'frontend-user',
|
authorUserId: 'frontend-user',
|
||||||
})
|
})
|
||||||
socket.emit('typing.stop', { channelId })
|
socket.emit('typing.stop', { channelId })
|
||||||
setContent('')
|
setContent('')
|
||||||
|
} catch {
|
||||||
|
setError('发送失败')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function editMessage() {
|
async function editMessage() {
|
||||||
if (!channelId || !editingMessageId || !editingContent.trim()) return
|
if (!channelId || !editingMessageId || !editingContent.trim()) return
|
||||||
|
setError('')
|
||||||
|
try {
|
||||||
await getApiClient().patch(`/channels/${channelId}/messages/${editingMessageId}`, {
|
await getApiClient().patch(`/channels/${channelId}/messages/${editingMessageId}`, {
|
||||||
content: editingContent,
|
content: editingContent,
|
||||||
})
|
})
|
||||||
setEditingContent('')
|
setEditingContent('')
|
||||||
await pullMessages()
|
await pullMessages()
|
||||||
|
} catch {
|
||||||
|
setError('编辑失败')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function deleteMessage(messageId: string) {
|
async function deleteMessage(messageId: string) {
|
||||||
if (!channelId || !messageId) return
|
if (!channelId || !messageId) return
|
||||||
|
setError('')
|
||||||
|
try {
|
||||||
await getApiClient().delete(`/channels/${channelId}/messages/${messageId}`)
|
await getApiClient().delete(`/channels/${channelId}/messages/${messageId}`)
|
||||||
await pullMessages()
|
await pullMessages()
|
||||||
|
} catch {
|
||||||
|
setError('删除失败')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function onChangeChannel(value: string) {
|
function onChangeChannel(value: string) {
|
||||||
@@ -155,6 +180,8 @@ export default function ChatPage() {
|
|||||||
<p>Guild: {guildId || '-'}</p>
|
<p>Guild: {guildId || '-'}</p>
|
||||||
<p>Socket: {socketState}</p>
|
<p>Socket: {socketState}</p>
|
||||||
<p>在线人数: {onlineCount}</p>
|
<p>在线人数: {onlineCount}</p>
|
||||||
|
{loading ? <p>加载中...</p> : null}
|
||||||
|
{error ? <p style={{ color: 'crimson' }}>{error}</p> : null}
|
||||||
{typingUsers.length ? <p>正在输入: {typingUsers.join(', ')}</p> : null}
|
{typingUsers.length ? <p>正在输入: {typingUsers.join(', ')}</p> : null}
|
||||||
<div style={{ display: 'flex', gap: 8, marginBottom: 8 }}>
|
<div style={{ display: 'flex', gap: 8, marginBottom: 8 }}>
|
||||||
<input value={channelId} onChange={(e) => onChangeChannel(e.target.value)} placeholder="Channel ID" />
|
<input value={channelId} onChange={(e) => onChangeChannel(e.target.value)} placeholder="Channel ID" />
|
||||||
@@ -186,6 +213,7 @@ export default function ChatPage() {
|
|||||||
</li>
|
</li>
|
||||||
))}
|
))}
|
||||||
</ul>
|
</ul>
|
||||||
|
{!loading && !messages.length ? <p>暂无消息</p> : null}
|
||||||
{pageInfo ? (
|
{pageInfo ? (
|
||||||
<p>
|
<p>
|
||||||
next_expected_seq: {pageInfo.nextExpectedSeq ?? '-'} | highest_committed_seq:{' '}
|
next_expected_seq: {pageInfo.nextExpectedSeq ?? '-'} | highest_committed_seq:{' '}
|
||||||
|
|||||||
Reference in New Issue
Block a user