发送消息
发送消息
环信即时通讯 IM Flutter SDK 通过 EMChatManager
和 EMMessage
类实现文本、图片、音频、视频和文件等类型的消息的发送和接收。
- 对于单聊,环信即时通信 IM 默认支持陌生人之间发送消息,即无需添加好友即可聊天。若仅允许好友之间发送单聊消息,你需要 开启好友关系检查。
- 对于群组和聊天室,用户每次只能向所属的单个群组和聊天室发送消息。
- 关于消息发送控制,详见 单聊、群组聊天 和 聊天室 的 相关文档。
前提条件
开始前,请确保满足以下条件:
发送文本消息
- 调用
EMMessage#createTxtSendMessage
方法构造一条消息。 - 调用
EMChatManager#sendMessage
方法发送这条消息。
默认情况下,SDK 对单个用户发送消息的频率未做限制。如果你联系了环信商务设置了该限制,一旦在单聊、群聊或聊天室中单个用户的消息发送频率超过设定的上限,SDK 会上报错误,即错误码 509 MESSAGE_CURRENT_LIMITING
。
// 创建一条文本消息。
final msg = EMMessage.createTxtSendMessage(
// `targetId` 为接收方,单聊为对端用户 ID、群聊为群组 ID,聊天室为聊天室 ID。
targetId: conversationId,
// `content` 为消息文字内容。
content: 'hello',
// 会话类型:单聊为 `Chat`,群聊为 `GroupChat`, 聊天室为 `ChatRoom`,默认为单聊。
chatType: ChatType.Chat,
);
// 发送消息。
EMClient.getInstance.chatManager.sendMessage(msg);
发送附件消息
除文本消息外,SDK 还支持发送附件类型消息,包括语音、图片、视频和文件消息。
发送附件消息分为以下两步:
- 创建和发送附件类型消息。
- SDK 将附件上传到环信服务器。
发送语音消息
- 发送语音消息前,在应用层录制语音文件。
- 发送方调用
EMMessage#createVoiceSendMessage
方法传入接收方的用户 ID(群聊或聊天室分别为群组 ID 或聊天室 ID),语音文件的filePath
和语音时长创建语音消息。 - 发送方调用
sendMessage
方法发送消息。SDK 会将语音文件上传至环信服务器。
final voiceMsg = EMMessage.createVoiceSendMessage(
targetId: targetId,
filePath: filePath,
duration: 30,
);
EMClient.getInstance.chatManager.sendMessage(msg);
发送图片消息
- 发送方调用
EMMessage#createImageSendMessage
方法传入接收方的用户 ID(群聊或聊天室分别为群组 ID 或聊天室 ID)和图片文件的filePath
,创建图片消息。 - 发送方调用
sendMessage
方法发送该消息。SDK 会将图片上传至环信服务器,服务器自动生成图片缩略图。
final imgMsg = EMMessage.createImageSendMessage(
targetId: targetId,
filePath: filePath,
);
EMClient.getInstance.chatManager.sendMessage(imgMsg);
发送 GIF 图片消息
自 Flutter SDK 4.15.0 开始,支持发送 GIF 图片消息。
GIF 图片消息是一种特殊的图片消息,与普通图片消息不同,GIF 图片发送时不能压缩。
图片缩略图的生成和下载与普通图片消息相同,详见 发送图片消息。
使用 EMMessage#createImageSendMessage
方法构造 GIF 图片消息体。
final msg = EMMessage.createImageSendMessage(
targetId: 'targetId',
filePath: 'filePath',
isGif: true,
);
EMClient.getInstance.chatManager.sendMessage(msg);
发送视频消息
- 发送视频消息前,在应用层完成视频文件的选取或者录制。 你可以设置发送消息结果回调,用于接收消息发送进度或者发送结果,如发送成功或失败。为此,需实现
EMChatManager#addMessageEvent
接口。 - 发送方调用
EMMessage#createVideoSendMessage
方法传入接收方的用户 ID(群聊或聊天室分别为群组 ID 或聊天室 ID)、图片文件的filePath
、创建视频消息。 - 发送方调用
sendMessage
方法发送消息。SDK 会将视频文件上传至消息服务器。若需要视频缩略图,你需自行获取视频首帧的路径,将该路径传入createVideoSendMessage
方法。
final videoMsg = EMMessage.createVideoSendMessage(
targetId: targetId,
filePath: filePath,
thumbnailLocalPath: thumbnailLocalPath,
duration: 30,
);
EMClient.getInstance.chatManager.sendMessage(videoMsg);
发送文件消息
- 发送方调用
EMMessage#createImageSendMessage
方法传入接收方的用户 ID(群聊或聊天室分别为群组 ID 或聊天室 ID)和文件的filePath
,创建文件消息。 - 发送方调用
sendMessage
方法发送该消息。SDK 将文件上传至环信服务器。
final fileMsg = EMMessage.createFileSendMessage(
targetId: targetId,
filePath: filePath,
);
EMClient.getInstance.chatManager.sendMessage(fileMsg);
发送位置消息
- 发送方调用
EMMessage#createLocationSendMessage
方法创建位置消息。 - 发送方调用
EMChatManager#sendMessage
方法发送位置消息。
发送位置时,需要集成第三方的地图服务,获取到位置点的经纬度信息。
final localMsg = EMMessage.createLocationSendMessage(
targetId: targetId,
latitude: 0,
longitude: 0,
address: 'address',
);
EMClient.getInstance.chatManager.sendMessage(localMsg);
发送和接收透传消息
透传消息可视为命令消息,通过发送这条命令给对方,通知对方要进行的操作,收到消息可以自定义处理。
具体功能可以根据自身业务需求自定义,例如实现头像、昵称的更新等。另外,以 em_
和 easemob::
开头的 action 为内部保留字段,注意不要使用。
提示
- 透传消息发送后,不支持撤回。
- 透传消息不会存入本地数据库中,所以在 UI 上不会显示。
发送透传消息的过程如下:
- 发送方调用
EMMessage#createCmdSendMessage
方法创建和发送透传消息。 - 发送方调用
sendMessage
方法发送消息。
final cmdMsg = EMMessage.createCmdSendMessage(
targetId: targetId,
// `action` 可以自定义。
action: action,
);
EMClient.getInstance.chatManager.sendMessage(cmdMsg);
发送自定义类型消息
除了几种消息之外,你可以自己定义消息类型,方便业务处理,即首先设置一个消息类型名称,然后可添加多种自定义消息。
final customMsg = EMMessage.createCustomSendMessage(
targetId: targetId,
// `event` 为需要传递的自定义消息事件,比如礼物消息,可以设置:
event: 'gift',
// `params` 类型为 `Map<String, String>`。
params: {'k': 'v'},
);
EMClient.getInstance.chatManager.sendMessage(customMsg);
发送合并消息
为了方便消息互动,即时通讯 IM 自 4.1.0 版本开始支持将多个消息合并在一起进行转发。你可以采取以下步骤进行消息的合并转发:
- 利用原始消息列表创建一条合并消息。
- 发送合并消息。
你可以调用 createCombineSendMessage
方法创建一条合并消息,然后调用 sendMessage
方法发送该条消息。
创建合并消息时,需要设置以下参数:
属性 | 类型 | 描述 |
---|---|---|
title | String | 合并消息的标题。 |
summary | String | 合并消息的概要。 |
compatibleText | String | 合并消息的兼容文本。 兼容文本起向下兼容不支持消息合并转发的版本的作用。当支持合并消息的 SDK 向不支持合并消息的低版本 SDK 发送消息时,低版本的 SDK 会将该属性解析为文本消息的消息内容。 |
msgIds | List | 合并消息的原始消息 ID 列表。该列表最多包含 300 个消息 ID。 |
targetId | String | 消息接收方。该字段的设置取决于会话类型: - 单聊:对方用户 ID; - 群聊:群组 ID; - 子区会话:子区 ID; - 聊天室聊天:聊天室 ID。 |
提示
- 合并转发支持嵌套,最多支持 10 层嵌套,每层最多 300 条消息。
- 不论
EMOptions#serverTransfer
设置为false
或true
,SDK 都会将合并消息附件上传到环信服务器。 - 合并消息不支持搜索。
- 对于转发合并消息,例如,用户 A 向用户 B 发送了合并消息,用户 B 将该合并消息转发给用户 C,需要调用转发单条合并消息的 API。详见 转发单条消息。
示例代码如下:
final combineMsg = EMMessage.createCombineSendMessage(
targetId: targetId,
title: 'A和B的聊天记录',
summary: 'A:这是A的消息内容\nB:这是B的消息内容',
compatibleText: compatibleText,
msgIds: msgIds,
);
EMClient.getInstance.chatManager.sendMessage(combineMsg);
更多
聊天室消息优先级与消息丢弃逻辑
消息优先级:对于聊天室消息,环信即时通讯提供消息分级功能,支持高、普通和低三种优先级,高优先级的消息会优先送达。你可以在创建消息时对指定消息类型或指定成员的消息设置为高优先级,确保这些消息优先送达。这种方式可以确保在聊天室内消息并发量较大或消息发送频率过高的情况下,服务器首先丢弃低优先级消息,将资源留给高优先级消息,确保重要消息(如打赏、公告等)优先送达,以此提升重要消息的可靠性。请注意,该功能并不保证高优先级消息必达。在聊天室内消息并发量过大的情况下,为保证用户实时互动的流畅性,即使是高优先级消息仍然会被丢弃。
消息丢弃逻辑:对于单个聊天室,每秒发送的消息数量默认超过 20 条,则会触发消息丢弃逻辑,即首先丢弃低优先级的消息,优先保留高优先级的消息。若带有优先级的消息超过了 20 条/秒,则按照消息发送时间顺序处理,丢弃后发送的消息。
final msg = EMMessage.createTxtSendMessage(
targetId: conversationId,
content: 'hello',
chatType: ChatType.ChatRoom,
);
// 聊天室消息的优先级。如果不设置,默认值为 `Normal`,即“普通”优先级。
msg.chatroomMessagePriority = ChatRoomMessagePriority.High;
EMClient.getInstance.chatManager.sendMessage(msg);
获取发送附件消息的进度
发送附件类型消息时,可以在 ChatMessageEvent#onProgress
回调中获取附件上传的进度(百分比),以 int 表示,范围为 [0, 100],示例代码如下:
final handler = ChatMessageEvent(
// 消息发送成功回调,msgId 为消息原始 ID,msg 为发送完成后的消息。
onSuccess: (msgId, msg) {},
// 附件上传进度回调,msgId 为消息原始 ID,progress 为消息发送进度(百分比), 范围[0, 100]。
onProgress: (msgId, progress) {},
// 消息发送失败回调,msgId 为消息原始 ID,msg 为发送完成后的消息,error 为错误原因。
onError: (msgId, msg, error) {},
);
/// 添加监听
EMClient.getInstance.chatManager.addMessageEvent(
'UNIQUE_HANDLER_ID',
handler,
);
/// 移除监听
EMClient.getInstance.chatManager.removeMessageEvent('UNIQUE_HANDLER_ID');
发送消息前的内容审核
- 内容审核关注消息 body
内容审核服务会关注消息 body 中指定字段的内容,不同类型的消息审核不同的字段,若创建消息时在这些字段中传入了很多业务信息,可能会影响审核效果。因此,创建消息时需要注意内容审核的字段不涉及业务信息,建议业务信息放在扩展字段中。
- 设置发送方收到内容审核替换后的内容
若初始化时打开了 ChatOptions#useReplacedMessageContents
开关,发送文本消息时如果被内容审核(Moderation)进行了内容替换,发送方会收到替换后的内容。若该开关为关闭状态,则发送方不会收到替换后的内容。