文档简介:
单聊会话支持消息送达回执、会话已读回执和消息已读回执,发送方发送消息后可及时了解接收方是否及时收到并阅读了信息,也可以了解整个会话是否已读。
群聊会话只支持消息已读回执。群主和群管理员在发送消息时,可以设置该消息是否需要已读回执。仅旗舰版及以上版本支持群消息已读回执功能。若要使用该功能,需在环信即时通讯云控制台 (opens new window)开通。
本文介绍如何使用环信即时通讯 IM Android SDK 实现单聊和群聊的消息回执功能。
技术原理
实现消息送达回执和已读回执的逻辑如下:
单聊消息送达回执:
- SDK 初始化时,用户将 Connection 类中的 delivery 参数 (opens new window)设置为 true 开启消息送达回执。
- 发送方发送一条消息。
- 接收方收到消息后,SDK 会自动向发送方发送送达回执。
- 发送方通过监听 onDeliveredMessage 收到送达回执。
消息已读回执:
-
单聊会话及消息已读回执
- 发送方发送一条消息。
- 消息接收方收到消息后,调用 send 发送会话或消息已读回执;
- 消息发送方通过监听 onChannelMessage 或 onReadMessage 回调接收会话或消息已读回执。
-
群聊只支持消息已读回执:
- 发送方发送一条消息,消息中的 allowGroupAck 字段设置为 true,要求返回已读回执;
- 消息接收方收到消息后调用 send 方法发送群组消息的已读回执。
- 发送方在线监听 onReadMessage 回调或离线监听 onStatisticsMessage 回调接收消息回执。
- 发送方通过 getGroupMsgReadUser 获取阅读消息的用户的详情。
前提条件
开始前,请确保满足以下要求:
- 已经集成和初始化环信 IM SDK,并实现了注册账号和登录功能。详情请参见 快速开始。
- 了解 使用限制 中的 API 调用频率限制。
- 群消息已读回执功能仅在环信 IM 旗舰版及以上版本支持该功能。若要使用该功能,需在环信即时通讯云控制台 (opens new window)开通。
实现方法
发送单聊消息送达回执
发送消息送达回执,可参考以下步骤:
- 消息发送方在 SDK 初始化时将 options 中的 delivery 设置为 true。示例代码如下:
const conn = new websdk.connection({ appKey: "your appKey", delivery: true, });
- 接收方收到消息后,消息发送方会收到 onDeliveredMessage 回调,得知消息已送达接收方。
conn.addEventHandler("customEvent", { onReceivedMessage: function (message) {}, // 收到消息送达服务器回执。
onDeliveredMessage: function (message) {}, // 收到消息送达客户端回执。 });
发送消息已读回执
消息已读回执用于告知单聊或群聊中的用户接收方已阅读其发送的消息。为降低消息已读回执方法的调用次数,SDK 还支持在单聊中使用会话已读回执功能,用于获知接收方是否阅读了会话中的未读消息。
单聊
单聊既支持消息已读回执,也支持会话已读回执。我们建议你按照如下逻辑结合使用两种回执,减少发送消息已读回执数量。
- 聊天页面未打开时,若有未读消息,进入聊天页面,发送会话已读回执;
- 聊天页面打开时,若收到消息,发送消息已读回执。
会话已读回执
参考以下步骤在单聊中实现会话已读回执。
- 接收方发送会话已读回执。
消息接收方进入会话页面,查看会话中是否有未读消息。若有,调用 send 方法发送会话已读回执,没有则不再发送。
let option = { chatType: "singleChat", // 会话类型,设置为单聊。 type: "channel", // 消息类型。 to: "userId",
// 接收消息对象的用户 ID。 }; let msg = WebIM.message.create(option); conn.send(msg);
- 消息发送方在 onChannelMessage 回调中收到会话已读回执:
conn.addEventHandler("customEvent", { onChannelMessage: (message) => {}, });
同一用户 ID 登录多设备的情况下,用户在一台设备上发送会话已读回执,服务器会将会话的未读消息数置为 0,同时其他设备会收到 onChannelMessage 回调。
消息已读回执
参考如下步骤在单聊中实现消息已读回执。
- 接收方发送消息已读回执。
消息接收方进入会话时,发送会话已读回执。
let option = { chatType: "singleChat", // 会话类型,设置为单聊。 type: "channel", // 消息类型。 to: "userId",
// 接收消息对象的用户 ID。 }; let msg = WebIM.message.create(option); conn.send(msg);
在会话页面,接收到消息时发送消息已读回执,如下所示:
let option = { type: "read", // 消息是否已读。 chatType: "singleChat", // 会话类型,这里为单聊。 to: "userId",
// 消息接收方的用户 ID。 id: "id", // 需要发送已读回执的消息 ID。 }; let msg = WebIM.message.create(option); conn.send(msg);
- 消息发送方监听消息已读回调。
你可以调用接口监听指定消息是否已读,示例代码如下:
conn.addEventHandler("customEvent", { onReadMessage: (message) => {}, });
群聊
对于群聊,群主和群管理员发送消息时,可以设置该消息是否需要已读回执。若需要,每个群成员在阅读消息后,SDK 均会发送已读回执,即阅读该消息的群成员数量即为已读回执的数量。
仅旗舰版及以上版本支持群消息已读回执功能。若要使用该功能,需在环信即时通讯云控制台 (opens new window)开通。
- 群主或群管理员发送消息时若需已读回执,需设置 allowGroupAck 为 true:
sendGroupReadMsg = () => { let option = { type: 'txt', // 消息类型。 chatType: 'groupChat', // 会话类型,这里为群聊。
to: 'groupId', // 消息接收方,即群组 ID。 msg: 'message content' // 消息内容。 msgConfig: { allowGroupAck: true }
// 设置此消息需要已读回执。 } let msg = WebIM.message.create(option); conn.send(msg).then((res) => { console.log
('send message success'); }).catch((e) => { console.log("send message error"); }) }
- 阅读消息后,消息接收方发送群消息已读回执。
sendReadMsg = () => { let option = { type: "read", // 消息是否已读。 chatType: "groupChat", // 会话类型,这里为群聊。
id: "msgId", // 需要发送已读回执的消息 ID。 to: "groupId", // 群组 ID。 ackContent: JSON.stringify({}), // 回执内容。
}; let msg = WebIM.message.create(option); conn.send(msg); };
-
消息发送方通过监听以下任一回调接收消息已读回执:
- onReadMessage:消息发送方在线时监听该回调。
- onStatisticMessage:消息发送方离线时监听该回调。
// 在线时可以在 onReadMessage 里监听。 conn.addEventHandler("customEvent", { onReadMessage: (message) =>
-
{ let { mid } = message; let msg = { id: mid, }; if (message.groupReadCount) { // 消息已读数。
-
msg.groupReadCount = message.groupReadCount[message.mid]; } }, // 离线时收到回执,登录后会在这里监听到。
-
onStatisticMessage: (message) => { let statisticMsg = message.location && JSON.parse(message.location);
-
let groupAck = statisticMsg.group_ack || []; }, });
-
消息发送方收到群消息已读回执后,可以获取已阅读该消息的用户的详细信息:
conn .getGroupMsgReadUser({ msgId: "messageId", // 消息 ID。 groupId: "groupId", // 群组 ID。 }) .
-
then((res) => { console.log(res); });