|
| 1 | +# 接收消息 |
| 2 | + |
| 3 | +环信即时通讯 IM Android SDK 通过 [EMMessageListener](https://sdkdocs.easemob.com/apidoc/android/chat3.0/interfacecom_1_1hyphenate_1_1_e_m_message_listener.html) 类实现文本、图片、音频、视频和文件等类型的消息的接收。 |
| 4 | + |
| 5 | +## 前提条件 |
| 6 | + |
| 7 | +开始前,请确保满足以下条件: |
| 8 | + |
| 9 | +- 完成 SDK 初始化,详见 [初始化文档](initialization.html)。 |
| 10 | +- 了解环信即时通讯 IM 的使用限制,详见 [使用限制](/product/limitation.html)。 |
| 11 | + |
| 12 | +## 接收文本消息 |
| 13 | + |
| 14 | +- 你可以用注册监听 [EMMessageListener](https://sdkdocs.easemob.com/apidoc/android/chat3.0/interfacecom_1_1hyphenate_1_1_e_m_message_listener.html) 接收消息。该 [EMMessageListener](https://sdkdocs.easemob.com/apidoc/android/chat3.0/interfacecom_1_1hyphenate_1_1_e_m_message_listener.html) 可以多次添加,请记得在不需要的时候移除 `listener`,如在 `activity` 的 `onDestroy()` 时。 |
| 15 | +- 在新消息到来时,你会收到 `onMessageReceived` 的回调,消息接收时可能是一条,也可能是多条。你可以在该回调里遍历消息队列,解析并显示收到的消息。若在初始化时打开了 `EMOptions#setIncludeSendMessageInMessageListener` 开关,则该回调中会返回发送成功的消息。 |
| 16 | +- 对于聊天室消息,你可以通过消息的 `EMMessage#isBroadcast` 属性判断该消息是否为 [通过 REST API 发送的聊天室全局广播消息](/document/server-side/message_broadcast.html#发送聊天室全局广播消息)。 |
| 17 | + |
| 18 | +```java |
| 19 | +EMMessageListener msgListener = new EMMessageListener() { |
| 20 | + |
| 21 | + // 收到消息,遍历消息队列,解析和显示。 |
| 22 | + @Override |
| 23 | + public void onMessageReceived(List<EMMessage> messages) { |
| 24 | + |
| 25 | + } |
| 26 | +}; |
| 27 | +// 注册消息监听 |
| 28 | +EMClient.getInstance().chatManager().addMessageListener(msgListener); |
| 29 | +// 解注册消息监听 |
| 30 | +EMClient.getInstance().chatManager().removeMessageListener(msgListener); |
| 31 | +``` |
| 32 | + |
| 33 | +## 接收附件消息 |
| 34 | + |
| 35 | +除文本消息外,SDK 还支持接收附件类型消息,包括语音、图片、视频和文件消息。 |
| 36 | + |
| 37 | +附件消息的接收过程如下: |
| 38 | + |
| 39 | +1. 接收附件消息。SDK 自动下载语音消息,默认自动下载图片和视频的缩略图。若下载原图、视频和文件,需调用 `downloadAttachment` 方法。 |
| 40 | +2. 获取附件的服务器地址和本地路径。 |
| 41 | + |
| 42 | +自 4.14.0 版本开始,即时通讯 IM 支持消息附件下载鉴权功能。该功能默认关闭,如要开通需联系环信商务。该功能开通后,用户必须调用 SDK 的 `downloadAttachment` 方法下载消息附件。 |
| 43 | + |
| 44 | +### 接收语音消息 |
| 45 | + |
| 46 | +1. 接收方收到语音消息时,自动下载语音文件。 |
| 47 | +2. 接收方收到 [onMessageReceived 回调](#接收文本消息),调用 `getRemoteUrl` 或 `getLocalUri` 方法获取语音文件的服务器地址或本地路径,从而获取语音文件。 |
| 48 | + |
| 49 | +```java |
| 50 | +EMVoiceMessageBody voiceBody = (EMVoiceMessageBody) msg.getBody(); |
| 51 | +// 获取语音文件在服务器的地址。 |
| 52 | +String voiceRemoteUrl = voiceBody.getRemoteUrl(); |
| 53 | +// 本地语音文件的资源路径。 |
| 54 | +Uri voiceLocalUri = voiceBody.getLocalUri(); |
| 55 | +``` |
| 56 | + |
| 57 | +### 接收图片消息 |
| 58 | + |
| 59 | +1. 接收方收到图片消息,自动下载图片缩略图。 |
| 60 | + |
| 61 | +- 默认情况下,SDK 自动下载缩略图,即 `EMClient.getInstance().getOptions().setAutoDownloadThumbnail(true)`。 |
| 62 | +- 若设置为手动下载缩略图,即 `EMClient.getInstance().getOptions().setAutoDownloadThumbnail(false)`,需调用 `EMClient.getInstance().chatManager().downloadThumbnail(message)` 下载。 |
| 63 | + |
| 64 | +2. 接收方收到 [onMessageReceived 回调](#接收文本消息),调用 `downloadAttachment` 下载原图。 |
| 65 | + |
| 66 | +```java |
| 67 | +@Override |
| 68 | +public void onMessageReceived(List<EMMessage> messages) { |
| 69 | + for(EMMessage message : messages) { |
| 70 | + if (message.getType() == Type.IMAGE) { |
| 71 | + message.setMessageStatusCallback(new EMCallBack() { |
| 72 | + @Override |
| 73 | + public void onSuccess() { |
| 74 | + // 附件下载成功 |
| 75 | + } |
| 76 | + @Override |
| 77 | + public void onError(int code, String error) { |
| 78 | + // 附件下载失败 |
| 79 | + } |
| 80 | + |
| 81 | + @Override |
| 82 | + public void onProgress(int progress, String status) { |
| 83 | + // 附件下载进度 |
| 84 | + } |
| 85 | + |
| 86 | + }); |
| 87 | + // 下载附件 |
| 88 | + EMClient.getInstance().chatManager().downloadAttachment(message); |
| 89 | + } |
| 90 | + } |
| 91 | +} |
| 92 | +``` |
| 93 | + |
| 94 | +3. 获取图片消息的缩略图和附件。 |
| 95 | + |
| 96 | +```java |
| 97 | +EMImageMessageBody imgBody = (EMImageMessageBody) message.getBody(); |
| 98 | +// 从服务器端获取图片文件。 |
| 99 | +String imgRemoteUrl = imgBody.getRemoteUrl(); |
| 100 | +// 从服务器端获取图片缩略图。 |
| 101 | +String thumbnailUrl = imgBody.getThumbnailUrl(); |
| 102 | +// 从本地获取图片文件。 |
| 103 | +Uri imgLocalUri = imgBody.getLocalUri(); |
| 104 | +// 从本地获取图片缩略图。 |
| 105 | +Uri thumbnailLocalUri = imgBody.thumbnailLocalUri(); |
| 106 | +``` |
| 107 | + |
| 108 | +### 接收 GIF 图片消息 |
| 109 | + |
| 110 | +自 Android SDK 4.14.0 开始,支持接收 GIF 图片消息。 |
| 111 | + |
| 112 | +图片缩略图的下载与普通图片消息相同,详见 [接收图片消息](#接收图片消息)。 |
| 113 | + |
| 114 | +与普通消息相同,接收 GIF 图片消息时,接收方会收到 `onMessageReceived` 回调方法。接收方判断为图片消息后,读取消息体的 `isGif` 属性,若值是 `YES`, 则为 GIF 图片消息。 |
| 115 | + |
| 116 | +```java |
| 117 | +public void onMessageReceived(List<EMMessage> messages) { |
| 118 | + for(EMMessage message : messages) { |
| 119 | + if (message.getType() == Type.IMAGE && ) { |
| 120 | + EMImageMessageBody body = (EMImageMessageBody) msg.getBody(); |
| 121 | + if(body.isGif()) { |
| 122 | + // 根据业务情况处理gif message, 例如下载展示该消息 |
| 123 | + } |
| 124 | + } |
| 125 | + } |
| 126 | + |
| 127 | +} |
| 128 | +``` |
| 129 | + |
| 130 | +### 接收视频消息 |
| 131 | + |
| 132 | +1. 接收方收到视频消息时,自动下载视频缩略图。你可以设置自动或手动下载视频缩略图,该设置与图片缩略图相同,详见 [设置图片缩略图自动下载](#接收图片消息)。 |
| 133 | +2. 接收方收到 [onMessageReceived 回调](#接收文本消息),可以调用 `EMClient.getInstance().chatManager().downloadAttachment(message)` 方法下载视频原文件。 |
| 134 | + |
| 135 | +```java |
| 136 | +/** |
| 137 | + * 下载视频文件。 |
| 138 | + */ |
| 139 | +private void downloadVideo(final EMMessage message) { |
| 140 | + message.setMessageStatusCallback(new EMCallBack() { |
| 141 | + @Override |
| 142 | + public void onSuccess() { |
| 143 | + } |
| 144 | + |
| 145 | + @Override |
| 146 | + public void onProgress(final int progress,String status) { |
| 147 | + } |
| 148 | + |
| 149 | + @Override |
| 150 | + public void onError(final int error, String msg) { |
| 151 | + } |
| 152 | + }); |
| 153 | + // 下载附件 |
| 154 | + EMClient.getInstance().chatManager().downloadAttachment(message); |
| 155 | +} |
| 156 | +``` |
| 157 | + |
| 158 | +3. 获取视频缩略图和视频原文件。 |
| 159 | + |
| 160 | +```java |
| 161 | +// 从服务器端获取视频文件。 |
| 162 | +String imgRemoteUrl = ((EMVideoMessageBody) body).getRemoteUrl(); |
| 163 | +// 从服务器获取视频缩略图文件。 |
| 164 | +String thumbnailUrl = ((EMVideoMessageBody) body).getThumbnailUrl(); |
| 165 | +// 从本地获取视频文件文件。 |
| 166 | +Uri localUri = ((EMVideoMessageBody) body).getLocalUri(); |
| 167 | +// 从本地获取视频缩略图文件。 |
| 168 | +Uri localThumbUri = ((EMVideoMessageBody) body).thumbnailLocalUri(); |
| 169 | +``` |
| 170 | + |
| 171 | +### 接收文件消息 |
| 172 | + |
| 173 | +1. 接收方收到 [onMessageReceived 回调](#接收文本消息),调用 `downloadAttachment` 方法下载文件。 |
| 174 | + |
| 175 | +```java |
| 176 | +/** |
| 177 | + * 下载文件。 |
| 178 | + */ |
| 179 | +private void downloadFile(final EMMessage message) { |
| 180 | + message.setMessageStatusCallback(new CallBack() { |
| 181 | + @Override |
| 182 | + public void onSuccess() { |
| 183 | + } |
| 184 | + |
| 185 | + @Override |
| 186 | + public void onProgress(final int progress,String status) { |
| 187 | + } |
| 188 | + |
| 189 | + @Override |
| 190 | + public void onError(final int error, String msg) { |
| 191 | + } |
| 192 | + }); |
| 193 | + // 下载附件 |
| 194 | + EMClient.getInstance().chatManager().downloadAttachment(message); |
| 195 | +} |
| 196 | +``` |
| 197 | + |
| 198 | +2. 调用以下方法从服务器或本地获取文件附件: |
| 199 | + |
| 200 | +```java |
| 201 | +EMNormalFileMessageBody fileMessageBody = (EMNormalFileMessageBody) message.getBody(); |
| 202 | +// 从服务器获取文件。 |
| 203 | +String fileRemoteUrl = fileMessageBody.getRemoteUrl(); |
| 204 | +// 从本地获取文件。 |
| 205 | +Uri fileLocalUri = fileMessageBody.getLocalUri(); |
| 206 | +``` |
| 207 | + |
| 208 | +## 接收位置消息 |
| 209 | + |
| 210 | +接收位置消息与文本消息一致,详见[接收文本消息](#接收文本消息)。 |
| 211 | + |
| 212 | +接收方接收到位置消息时,需要将该位置的经纬度,借由第三方的地图服务,将位置在地图上显示出来。 |
| 213 | + |
| 214 | +## 接收透传消息 |
| 215 | + |
| 216 | +透传消息可视为命令消息,通过发送这条命令给对方,通知对方要进行的操作,收到消息可以自定义处理。 |
| 217 | + |
| 218 | +具体功能可以根据自身业务需求自定义,例如实现头像、昵称的更新等。另外,以 `em_` 和 `easemob::` 开头的 action 为内部保留字段,注意不要使用。 |
| 219 | + |
| 220 | +:::tip |
| 221 | +- 透传消息发送后,不支持撤回。 |
| 222 | +- 透传消息不会存入本地数据库中,所以在 UI 上不会显示。 |
| 223 | +::: |
| 224 | + |
| 225 | +接收方通过 `onMessageReceived` 和 `onCmdMessageReceived` 回调接收透传消息,方便用户进行不同的处理。 |
| 226 | + |
| 227 | +```java |
| 228 | +EMMessageListener msgListener = new EMMessageListener(){ |
| 229 | + // 收到消息。 |
| 230 | + @Override |
| 231 | + public void onMessageReceived(List<EMMessage> messages) { |
| 232 | + } |
| 233 | + // 收到透传消息。 |
| 234 | + @Override |
| 235 | + public void onCmdMessageReceived(List<EMMessage> messages) { |
| 236 | + } |
| 237 | +} |
| 238 | +``` |
| 239 | + |
| 240 | +## 接收自定义类型消息 |
| 241 | + |
| 242 | +你可以自己定义消息类型,方便业务处理,即首先设置一个消息类型名称,然后可添加多种自定义消息。 |
| 243 | + |
| 244 | +接收自定义消息与其他类型消息一致,详见[接收文本消息](#接收文本消息)。 |
| 245 | + |
| 246 | +## 接收合并消息 |
| 247 | + |
| 248 | +为了方便消息互动,即时通讯 IM 自 4.1.0 版本开始支持将多个消息合并在一起进行转发。 |
| 249 | + |
| 250 | +接收合并消息与接收普通消息的操作相同,详见[接收文本消息](#接收文本消息)。 |
| 251 | + |
| 252 | +- 对于不支持合并转发消息的 SDK 版本,该类消息会被解析为文本消息,消息内容为 `compatibleText` 携带的内容,其他字段会被忽略。 |
| 253 | +- 合并消息实际上是一种附件消息。收到合并消息后,你可以调用 `downloadAndParseCombineMessage` 方法下载合并消息附件并解析出原始消息列表。 |
| 254 | +- 对于一条合并消息,首次调用该方法会下载和解析合并消息附件,然后返回原始消息列表,而后续调用会存在以下情况: |
| 255 | + - 若附件已存在,该方法会直接解析附件并返回原始消息列表。 |
| 256 | + - 若附件不存在,该方法首先下载附件,然后解析附件并返回原始消息列表。 |
| 257 | + |
| 258 | +```java |
| 259 | +EMClient.getInstance().chatManager().downloadAndParseCombineMessage(combineMessage, new EMValueCallBack<List<EMMessage>>() { |
| 260 | + @Override |
| 261 | + public void onSuccess(List<EMMessage> value) { |
| 262 | + // 处理并展示消息列表 |
| 263 | + } |
| 264 | + |
| 265 | + @Override |
| 266 | + public void onError(int error, String errorMsg) { |
| 267 | + // 处理出错信息 |
| 268 | + } |
| 269 | +}); |
| 270 | +``` |
| 271 | + |
0 commit comments