- 新增 `how-to-use-messageBlock.md`,详细介绍 `messageBlock.ts` 的 Redux Slice 及其状态管理、actions 和 selectors。 - 新增 `how-to-use-messageThunk.md`,概述 `messageThunk.ts` 的核心功能和主要 Thunks 的使用。 - 新增 `how-to-use-useMessageOperations.md`,提供 `useMessageOperations` Hook 的使用示例和功能说明,简化组件与消息数据的交互。
6.3 KiB
messageBlock.ts 使用指南
该文件定义了用于管理应用程序中所有 MessageBlock 实体的 Redux Slice。它使用 Redux Toolkit 的 createSlice 和 createEntityAdapter 来高效地处理规范化的状态,并提供了一系列 actions 和 selectors 用于与消息块数据交互。
核心目标
- 状态管理: 集中管理所有
MessageBlock的状态。MessageBlock代表消息中的不同内容单元(如文本、代码、图片、引用等)。 - 规范化: 使用
createEntityAdapter将MessageBlock数据存储在规范化的结构中({ ids: [], entities: {} }),这有助于提高性能和简化更新逻辑。 - 可预测性: 提供明确的 actions 来修改状态,并通过 selectors 安全地访问状态。
关键概念
- Slice (
createSlice): Redux Toolkit 的核心 API,用于创建包含 reducer 逻辑、action creators 和初始状态的 Redux 模块。 - Entity Adapter (
createEntityAdapter): Redux Toolkit 提供的工具,用于简化对规范化数据的 CRUD(创建、读取、更新、删除)操作。它会自动生成 reducer 函数和 selectors。 - Selectors: 用于从 Redux store 中派生和计算数据的函数。Selectors 可以被记忆化(memoized),以提高性能。
State 结构
messageBlocks slice 的状态结构由 createEntityAdapter 定义,大致如下:
{
ids: string[]; // 存储所有 MessageBlock ID 的有序列表
entities: { [id: string]: MessageBlock }; // 按 ID 存储 MessageBlock 对象的字典
loadingState: 'idle' | 'loading' | 'succeeded' | 'failed'; // (可选) 其他状态,如加载状态
error: string | null; // (可选) 错误信息
}
Actions
该 slice 导出以下 actions (由 createSlice 和 createEntityAdapter 自动生成或自定义):
-
upsertOneBlock(payload: MessageBlock):- 添加一个新的
MessageBlock或更新一个已存在的MessageBlock。如果 payload 中的id已存在,则执行更新;否则执行插入。
- 添加一个新的
-
upsertManyBlocks(payload: MessageBlock[]):- 添加或更新多个
MessageBlock。常用于批量加载数据(例如,加载一个 Topic 的所有消息块)。
- 添加或更新多个
-
removeOneBlock(payload: string):- 根据提供的
id(payload) 移除单个MessageBlock。
- 根据提供的
-
removeManyBlocks(payload: string[]):- 根据提供的
id数组 (payload) 移除多个MessageBlock。常用于删除消息或清空 Topic 时清理相关的块。
- 根据提供的
-
removeAllBlocks():- 移除 state 中的所有
MessageBlock实体。
- 移除 state 中的所有
-
updateOneBlock(payload: { id: string; changes: Partial<MessageBlock> }):- 更新一个已存在的
MessageBlock。payload需要包含块的id和一个包含要更改的字段的changes对象。
- 更新一个已存在的
-
setMessageBlocksLoading(payload: 'idle' | 'loading'):- (自定义) 设置
loadingState属性。
- (自定义) 设置
-
setMessageBlocksError(payload: string):- (自定义) 设置
loadingState为'failed'并记录错误信息。
- (自定义) 设置
使用示例 (在 Thunk 或其他 Dispatch 的地方):
import { upsertOneBlock, removeManyBlocks, updateOneBlock } from './messageBlock'
import store from './store' // 假设这是你的 Redux store 实例
// 添加或更新一个块
const newBlock: MessageBlock = {
/* ... block data ... */
}
store.dispatch(upsertOneBlock(newBlock))
// 更新一个块的内容
store.dispatch(updateOneBlock({ id: blockId, changes: { content: 'New content' } }))
// 删除多个块
const blockIdsToRemove = ['id1', 'id2']
store.dispatch(removeManyBlocks(blockIdsToRemove))
Selectors
该 slice 导出由 createEntityAdapter 生成的基础 selectors,并通过 messageBlocksSelectors 对象访问:
messageBlocksSelectors.selectIds(state: RootState): string[]: 返回包含所有块 ID 的数组。messageBlocksSelectors.selectEntities(state: RootState): { [id: string]: MessageBlock }: 返回块 ID 到块对象的映射字典。messageBlocksSelectors.selectAll(state: RootState): MessageBlock[]: 返回包含所有块对象的数组。messageBlocksSelectors.selectTotal(state: RootState): number: 返回块的总数。messageBlocksSelectors.selectById(state: RootState, id: string): MessageBlock | undefined: 根据 ID 返回单个块对象,如果找不到则返回undefined。
此外,还提供了一个自定义的、记忆化的 selector:
selectFormattedCitationsByBlockId(state: RootState, blockId: string | undefined): Citation[]:- 接收一个
blockId。 - 如果该 ID 对应的块是
CITATION类型,则提取并格式化其包含的引用信息(来自网页搜索、知识库等),进行去重和重新编号,最后返回一个Citation[]数组,用于在 UI 中显示。 - 如果块不存在或类型不匹配,返回空数组
[]。 - 这个 selector 封装了处理不同引用来源(Gemini, OpenAI, OpenRouter, Zhipu 等)的复杂逻辑。
- 接收一个
使用示例 (在 React 组件或 useSelector 中):
import { useSelector } from 'react-redux'
import { messageBlocksSelectors, selectFormattedCitationsByBlockId } from './messageBlock'
import type { RootState } from './store'
// 获取所有块
const allBlocks = useSelector(messageBlocksSelectors.selectAll)
// 获取特定 ID 的块
const specificBlock = useSelector((state: RootState) => messageBlocksSelectors.selectById(state, someBlockId))
// 获取特定引用块格式化后的引用列表
const formattedCitations = useSelector((state: RootState) => selectFormattedCitationsByBlockId(state, citationBlockId))
// 在组件中使用引用数据
// {formattedCitations.map(citation => ...)}
集成
messageBlock.ts slice 通常与 messageThunk.ts 中的 Thunks 紧密协作。Thunks 负责处理异步逻辑(如 API 调用、数据库操作),并在需要时 dispatch messageBlock slice 的 actions 来更新状态。例如,当 messageThunk 接收到流式响应时,它会 dispatch upsertOneBlock 或 updateOneBlock 来实时更新对应的 MessageBlock。同样,删除消息的 Thunk 会 dispatch removeManyBlocks。
理解 messageBlock.ts 的职责是管理状态本身,而 messageThunk.ts 负责触发状态变更的异步流程,这对于维护清晰的应用架构至关重要。