接收消息
接收消息
环信即时通讯 IM HarmonyOS SDK 通过 ChatMessageListener 类实现文本、图片、音频、视频和文件等类型的消息的接收。
前提条件
开始前,请确保满足以下条件:
接收文本消息
你可以用注册监听 ChatMessageListener
接收消息。该 ChatMessageListener
可以多次添加,请记得在不需要的时候移除 listener
。
在新消息到来时,你会收到 onMessageReceived
的回调,消息接收时可能是一条,也可能是多条。你可以在该回调里遍历消息队列,解析并显示收到的消息。
对于聊天室消息,你可以通过消息的 ChatMessage#isBroadcast
属性判断该消息是否为 通过 REST API 发送的聊天室全局广播消息。
let msgListener: ChatMessageListener = {
onMessageReceived: (messages: ChatMessage[]): void => {
// 收到消息,遍历消息队列,解析和显示。
}
}
// 注册消息监听
ChatClient.getInstance().chatManager()?.addMessageListener(msgListener);
// 解注册消息监听
ChatClient.getInstance().chatManager()?.removeMessageListener(msgListener);
接收附件消息
除文本消息外,SDK 还支持接收附件类型消息,包括语音、图片、视频和文件消息。
附件消息的接收过程如下:
- 接收附件消息。SDK 自动下载语音消息,默认自动下载图片和视频的缩略图。若下载原图、视频和文件,需调用
downloadAttachment
方法。 - 获取附件的服务器地址和本地路径。
自 1.7.0 版本开始,即时通讯 IM 支持消息附件下载鉴权功能。该功能默认关闭,如要开通需联系环信商务。该功能开通后,用户必须调用 SDK 的 downloadAttachment
方法下载消息附件。
接收语音消息
接收方收到语音消息时,自动下载语音文件。
接收方收到 onMessageReceived 回调,调用
getRemoteUrl
或getLocalPath
方法获取语音文件的服务器地址或本地路径,从而获取语音文件。
let voiceBody = message.getBody() as VoiceMessageBody;
// 获取语音文件在服务器的地址。
let voiceRemoteUrl = voiceBody.getRemoteUrl();
// 本地语音文件的本地路径。
let voiceLocalPath = voiceBody.getLocalPath();
接收图片消息
- 接收方收到图片消息,自动下载图片缩略图。
- 默认情况下,SDK 自动下载缩略图,即
ChatOptions.setAutoDownloadThumbnail
设置为true
。 - 若设置为手动下载缩略图,即
ChatOptions.setAutoDownloadThumbnail
设置为false
,需调用ChatClient.getInstance().chatManager()?.downloadThumbnail(message)
下载。
- 接收方收到 onMessageReceived 回调,调用
downloadAttachment
下载原图。
let msgListener: ChatMessageListener = {
onMessageReceived: (messages: ChatMessage[]): void => {
messages.forEach( message => {
if (message.getType() === ContentType.IMAGE) {
let callback: ChatCallback = {
onSuccess: (): void => {
// 附件下载成功
},
onError: (code: number, error: string): void => {
// 附件下载失败
},
onProgress: (progress: number): void => {
// 附件下载进度
}
}
message.setMessageStatusCallback(callback);
// 下载附件
ChatClient.getInstance().chatManager()?.downloadAttachment(message);
}
})
}
}
- 获取图片消息的缩略图和附件。
let imgBody = message.getBody() as ImageMessageBody;
// 从服务器端获取图片文件。
let imgRemoteUrl = imgBody.getRemoteUrl();
// 从服务器端获取图片缩略图。
let thumbnailUrl = imgBody.getThumbnailRemoteUrl();
// 从本地获取图片文件。
let imgLocalPath = imgBody.getLocalPath();
// 从本地获取图片缩略图。
let thumbnailLocalPath = imgBody.getThumbnailLocalPath();
接收 GIF 图片消息
自 HarmonyOS SDK 1.7.0 开始,支持接收 GIF 图片消息。
GIF 图片缩略图的下载与普通图片消息相同,详见 接收图片消息。
与普通消息相同,接收 GIF 图片消息时,接收方会收到 onMessageReceived 回调方法。接收方判断为图片消息后,读取消息体的 isGif
属性,若值是 YES
, 则为 GIF 图片消息。
ChatClient.getInstance().chatManager()?.addMessageListener({
onMessageReceived: (messages) => {
messages.forEach(message => {
if (message.getType() === ContentType.IMAGE) {
let body = message.getBody() as ImageMessageBody;
if (body.isGif()) {
// 根据业务情况处理gif message, 例如下载展示该消息
}
}
})
}
});
接收视频消息
接收方收到视频消息时,自动下载视频缩略图。你可以设置自动或手动下载视频缩略图,该设置与图片缩略图相同,详见设置图片缩略图自动下载。
接收方收到 onMessageReceived 回调,可以调用
ChatClient.getInstance().chatManager()?.downloadAttachment(message)
方法下载视频原文件。
let msgListener: ChatMessageListener = {
onMessageReceived: (messages: ChatMessage[]): void => {
messages.forEach( message => {
if (message.getType() === ContentType.VIDEO) {
let callback: ChatCallback = {
onSuccess: (): void => {
// 附件下载成功
},
onError: (code: number, error: string): void => {
// 附件下载失败
},
onProgress: (progress: number): void => {
// 附件下载进度
}
}
message.setMessageStatusCallback(callback);
// 下载附件
ChatClient.getInstance().chatManager()?.downloadAttachment(message);
}
})
}
}
- 获取视频缩略图和视频原文件。
let body = message.getBody() as VideoMessageBody;
// 从服务器端获取视频文件。
let imgRemoteUrl = body.getRemoteUrl();
// 从服务器获取视频缩略图文件。
let thumbnailUrl = body.getThumbnailRemoteUrl();
// 从本地获取视频文件文件。
let localPath = body.getLocalPath();
// 从本地获取视频缩略图文件。
let localThumbPath = body.getThumbnailLocalPath();
接收文件消息
- 接收方收到 onMessageReceived 回调,调用
downloadAttachment
方法下载文件。
/**
* 下载文件。
*/
private downloadFile(message: ChatMessage) {
let callback: ChatCallback = {
onSuccess: (): void => {
// 附件下载成功
},
onError: (code: number, error: string): void => {
// 附件下载失败
},
onProgress: (progress: number): void => {
// 附件下载进度
}
}
message.setMessageStatusCallback(callback);
// 下载附件
ChatClient.getInstance().chatManager()?.downloadAttachment(message);
}
- 调用以下方法从服务器或本地获取文件附件:
let fileMessageBody = message.getBody() as FileMessageBody;
// 从服务器获取文件。
let fileRemoteUrl = fileMessageBody.getRemoteUrl();
// 从本地获取文件。
let fileLocalPath = fileMessageBody.getLocalPath();
接收位置消息
接收位置消息与文本消息一致,详见 接收文本消息。
接收方接收到位置消息时,需要将该位置的经纬度,借由第三方的地图服务,将位置在地图上显示出来。
接收透传消息
透传消息可视为命令消息,通过发送这条命令给对方,通知对方要进行的操作,收到消息可以自定义处理。
具体功能可以根据自身业务需求自定义,例如实现头像、昵称的更新等。另外,以 em_
和 easemob::
开头的 action 为内部保留字段,注意不要使用。
提示
- 透传消息发送后,不支持撤回。
- 透传消息不会存入本地数据库中,所以在 UI 上不会显示。
接收方通过 onMessageReceived 和 onCmdMessageReceived
回调接收透传消息,方便用户进行不同的处理。
let msgListener: ChatMessageListener = {
onMessageReceived: (messages: ChatMessage[]): void => {
// 接收到消息
},
onCmdMessageReceived: (messages: ChatMessage[]): void => {
// 接收到透传消息
}
}
接收自定义类型消息
你可以自己定义消息类型,方便业务处理,即首先设置一个消息类型名称,然后可添加多种自定义消息。
接收自定义消息与其他类型消息一致,详见 接收文本消息。
接收合并消息
为了方便消息互动,SDK 支持将多个消息合并在一起进行转发。
接收合并消息与接收普通消息的操作相同,详见 接收文本消息。
- 对于不支持合并转发消息的 SDK 版本,该类消息会被解析为文本消息,消息内容为
compatibleText
携带的内容,其他字段会被忽略。 - 合并消息实际上是一种附件消息。收到合并消息后,你可以调用
downloadAndParseCombineMessage
方法下载合并消息附件并解析出原始消息列表。 - 对于一条合并消息,首次调用该方法会下载和解析合并消息附件,然后返回原始消息列表,而后续调用会存在以下情况:
- 若附件已存在,该方法会直接解析附件并返回原始消息列表。
- 若附件不存在,该方法首先下载附件,然后解析附件并返回原始消息列表。
ChatClient.getInstance().chatManager()?.downloadAndParseCombineMessage(message).then((result) => {
// 处理并展示消息列表
}).catch((e: ChatError) => {
// 处理出错信息
});