====== 管理子区消息 Android ======
更新时间:2022-06-16
子区消息消息类型属于群聊消息类型,与普通群组消息的区别是需要添加 ''%%isChatThreadMessage%%'' 标记。本文介绍环信即时通讯 IM Android SDK 如何发送、接收以及撤回子区消息。
===== 技术原理 =====
环信即时通讯 IM Android SDK 提供 ''%%EMChatManager%%''、''%%EMMessage%%'' 和 ''%%EMChatThread%%'' 类,用于管理子区消息,支持你通过调用 API 在项目中实现如下功能:
* 发送子区消息
* 接收子区消息
* 撤回子区消息
* 获取子区消息
消息收发流程如下:
- 用户 A 发送一条消息到环信的消息服务器;
- 单聊时消息时,服务器投递消息给用户 B;对于群聊时消息,服务器投递给群内其他每一个成员;对于子区消息,服务器投递给子区内其他每一个成员;
- 用户收到消息。
[[https://docs-im.easemob.com/_detail/ccim/web/sendandreceivemsg.png?id=ccim%3Aandroid%3Athread|{{https://docs-im.easemob.com/_media/ccim/web/sendandreceivemsg.png?w=800&tok=54ca33|img}}]]
子区创建和查看如下图:
[[https://docs-im.easemob.com/_detail/ccim/ios/threads.png?id=ccim%3Aandroid%3Athread|{{https://docs-im.easemob.com/_media/ccim/ios/threads.png|img}}]]
===== 前提条件 =====
开始前,请确保满足以下条件:
* 已集成环信 IM ''%%(3.9.3 以上版本)%%'' 的基本功能,账户登录成功。
* 完成 SDK 初始化,详见 [[https://docs-im.easemob.com/ccim/android/quickstart|快速开始]]。
* 了解环信即时通讯 IM 的使用限制,详见 [[https://docs-im.easemob.com/ccim/limitation|使用限制]]。
* 联系商务开通子区功能。
===== 实现方法 =====
本节介绍如何使用环信即时通讯 IM Android SDK 提供的 API 实现上述功能。
==== 发送子区消息 ====
发送子区消息和发送群组消息的方法基本一致,详情请参考 [[https://docs-im.easemob.com/ccim/android/message2#发送文本消息|发送消息]]。唯一不同的是,发送子区消息需要指定标记 ''%%isChatThreadMessage%%'' 为 ''%%true%%''。
示例代码如下:
// 创建一条文本消息,`content` 为消息文字内容,`chatThreadId` 为子区 ID。
EMMessage message = EMMessage.createTxtSendMessage(content, chatThreadId);
// 设置消息类型,子区消息需要将 `ChatType` 设置为 `GroupChat`。
message.setChatType(ChatType.GroupChat);
// 设置消息标记 `isChatThreadMessage` 为 `true`。
message.setIsChatThreadMessage(true);
// 发送消息时可以设置 `EMCallBack` 的实例,获得消息发送的状态。可以在该回调中更新消息的显示状态。例如消息发送失败后的提示等等。
message.setMessageStatusCallback(new EMCallBack() {
@Override
public void onSuccess() {
}
@Override
public void onError(int code, String error) {
}
@Override
public void onProgress(int progress, String status) {
}
});
// 发送消息。
EMClient.getInstance().chatManager().sendMessage(message);
==== 接收子区消息 ====
接收消息的具体逻辑,请参考[[https://docs-im.easemob.com/ccim/android/message2#接收消息|接收消息]],此处只介绍子区消息和其他消息的区别。
子区有新增消息时,子区所属群组的所有成员收到 ''%%EMChatThreadChangeListener#onChatThreadUpdated%%'' 回调,子区成员收到 ''%%EMMessageListener#onMessageReceived%%'' 回调。
示例代码如下:
EMMessageListener msgListener = new EMMessageListener() {
// 收到消息,遍历消息队列,解析和显示。
@Override
public void onMessageReceived(List messages) {
for (EMMessage message : messages) {
if(message.isChatThreadMessage()) {
// 接收到子区消息,添加处理逻辑。
}
}
}
...// 其他回调,此处省略。
};
// 添加消息监听器。
EMClient.getInstance().chatManager().addMessageListener(msgListener);
// 移除消息监听器。
EMClient.getInstance().chatManager().removeMessageListener(msgListener);
==== 撤回子区消息 ====
接收消息的具体逻辑,请参考[[https://docs-im.easemob.com/ccim/android/message2#撤回消息|撤回消息]],此处只介绍子区消息和其他消息的区别。
子区有消息撤回时,子区所属群组的所有成员收到 ''%%EMChatThreadChangeListener#onChatThreadUpdated%%'' 回调,子区成员收到 ''%%EMMessageListener#onMessageRecalled%%'' 回调。
示例代码如下:
EMMessageListener msgListener = new EMMessageListener() {
// 收到撤回消息回调,遍历消息队列,解析和显示。
@Override
public void onMessageRecalled(List messages) {
for (EMMessage message : messages) {
if(message.isChatThreadMessage()) {
// 接收到子区消息被撤回,添加处理逻辑。
}
}
}
...// 其他回调,此处省略。
};
==== 获取子区消息 ====
进入单个子区会话后默认展示最早消息,用户可以从服务器获取子区历史消息;当你需要合并处理本地和服务器拉取到的消息(例如有用户撤回子区消息的提示是 SDK 在本地生成的一条消息)的时候,可以选择从本地获取子区消息。
=== 从服务器获取子区消息(消息漫游) ===
从服务器获取子区消息,请参考 [[https://docs-im.easemob.com/ccim/android/message4#消息管理_从服务器获取消息_消息漫游|从服务器获取消息 (消息漫游)]]。
=== 管理本地子区消息 ===
调用 ''%%EMChatManager#getAllConversations()%%'' 会返回单聊和群聊的会话,不会返回子区会话,你可以从本地数据库中读取指定会话的消息:
// 需要指定会话类型为 `EMConversationType.GroupChat`,且 `isChatThread` 设置为 `true`
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(chatThreadId, EMConversationType.GroupChat, createIfNotExists, isChatThread);
// 获取此会话的所有内存中的消息
List messages = conversation.getAllMessages();
// 如需处理本地数据库中消息,用以下方法到数据库中获取,SDK 会将这些消息自动存入此会话
List messages = conversation.loadMoreMsgFromDB(startMsgId, pagesize, searchDirection);
**注意:**
* 判断当前会话是否是子区会话,可以通过 ''%%EMConversation#isChatThread()%%'' 进行判断。