# Mirai - Messages ## 目录 - [消息系统](#消息系统) - [消息类型](#消息类型) - [消息元素](#消息元素) - [消息链](#消息链) - [发送消息](#发送消息) - [构造消息链](#构造消息链) - [元素唯一性](#元素唯一性) - [获取消息链中的消息元素](#获取消息链中的消息元素) - [Mirai 码](#mirai-码) - [转义规则](#转义规则) - [消息链的 mirai 码](#消息链的-mirai-码) - [由 `CodableMessage` 取得 mirai 码字符串](#由-codablemessage-取得-mirai-码字符串) - [由 mirai 码字符串取得 `MessageChain` 实例](#由-mirai-码字符串取得-messagechain-实例) - [`serializeToString` 与 `toString` 的区别](#serializeToString-与-toString-的区别) ## 消息系统 在 Contacts 章节提到,要发送消息,使用 `Contact.sendMessage(Message)`。`Message` 架构如下图所示。 [![](https://mermaid.ink/img/eyJjb2RlIjoiY2xhc3NEaWFncmFtXG5cbmNsYXNzIE1lc3NhZ2VDaGFpblxuTWVzc2FnZUNoYWluIDogTGlzdH5TaW5nbGVNZXNzYWdlflxuXG5NZXNzYWdlPHwtLU1lc3NhZ2VDaGFpblxuTWVzc2FnZTx8LS1TaW5nbGVNZXNzYWdlXG5cbk1lc3NhZ2VDaGFpbiBvLS0gU2luZ2xlTWVzc2FnZVxuXG5TaW5nbGVNZXNzYWdlPHwtLU1lc3NhZ2VDb250ZW50XG5TaW5nbGVNZXNzYWdlPHwtLU1lc3NhZ2VNZXRhZGF0YVxuXG4iLCJtZXJtYWlkIjp7InRoZW1lIjoiZGVmYXVsdCJ9LCJ1cGRhdGVFZGl0b3IiOmZhbHNlfQ)](https://mermaid-js.github.io/mermaid-live-editor/#/edit/eyJjb2RlIjoiY2xhc3NEaWFncmFtXG5cbmNsYXNzIE1lc3NhZ2VDaGFpblxuTWVzc2FnZUNoYWluIDogTGlzdH5TaW5nbGVNZXNzYWdlflxuXG5NZXNzYWdlPHwtLU1lc3NhZ2VDaGFpblxuTWVzc2FnZTx8LS1TaW5nbGVNZXNzYWdlXG5cbk1lc3NhZ2VDaGFpbiBvLS0gU2luZ2xlTWVzc2FnZVxuXG5TaW5nbGVNZXNzYWdlPHwtLU1lc3NhZ2VDb250ZW50XG5TaW5nbGVNZXNzYWdlPHwtLU1lc3NhZ2VNZXRhZGF0YVxuXG4iLCJtZXJtYWlkIjp7InRoZW1lIjoiZGVmYXVsdCJ9LCJ1cGRhdGVFZGl0b3IiOmZhbHNlfQ) `SingleMessage` 表示单个消息元素。 `MessageChain`(消息链) 是 `List`。主动发送的消息和从服务器接收消息都是 `MessageChain`。 > 回到 [目录](#目录) ## 消息类型 Mirai 支持富文本消息。 *单个消息元素(`SingleMessage`)* 分为 *内容(`MessageContent`)* 和 *元数据(`MessageMetadata`)*。 实践中,消息内容和消息元数据会混合存在于消息链中。 ### 内容 *内容(`MessageContent`)* 即为 *纯文本*、*提及某人*、*图片*、*语音* 和 *音乐分享* 等**有内容**的数据,一条消息中必须包含内容才能发送。 ### 元数据 *元数据(`MessageMetadata`)* 包含 *来源*、*引用回复* 和 *秀图标识* 等。 - *消息来源*(`MessageSource`)存在于每条消息中,包含唯一识别信息,用于撤回和引用回复的定位。 - *引用回复*(`QuoteReply`)若存在,则会在客户端中解析为本条消息引用了另一条消息。 - *秀图标识*(`ShowImageFlag`)若存在,则表明这条消息中的图片是以秀图发送(QQ 的一个功能)。 元数据与内容的区分就在于,一条消息没有元数据也能显示,但一条消息不能没有内容。**元数据是消息的属性**。 > 回到 [目录](#目录) ## 消息元素 Mirai 支持多种消息类型。 消息拥有三种转换到字符串的表示方式。 | 方法 | 解释 | |:-------------------------|:---------------------------------------------------------------------------------------------| | `serializeToMiraiCode()` | 对应的 Mirai 码. 消息的一种序列化方式,格式为 `[mirai:TYPE:PROP]`,其中 `TYPE` 为消息类型, `PROP` 为属性 | | `contentToSting()` | QQ 对话框中以纯文本方式会显示的消息内容。无法用纯文字表示的消息会丢失信息,如任何图片都是 `[图片]` | | `toString()` | Java 对象的 `toString()`,会尽可能包含多的信息用于调试作用,**行为可能不确定** | 各类型消息元素及其 `contentToString()` 如下表格所示。 [`MessageContent`]: ../mirai-core-api/src/commonMain/kotlin/message/data/SingleMessage.kt [`MessageMetadata`]: ../mirai-core-api/src/commonMain/kotlin/message/data/SingleMessage.kt [`PlainText`]: ../mirai-core-api/src/commonMain/kotlin/message/data/PlainText.kt [`At`]: ../mirai-core-api/src/commonMain/kotlin/message/data/At.kt [`AtAll`]: ../mirai-core-api/src/commonMain/kotlin/message/data/AtAll.kt [`Face`]: ../mirai-core-api/src/commonMain/kotlin/message/data/Face.kt [`PokeMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/PokeMessage.kt [`VipFace`]: ../mirai-core-api/src/commonMain/kotlin/message/data/VipFace.kt [`Image`]: ../mirai-core-api/src/commonMain/kotlin/message/data/Image.kt [`FlashImage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/FlashImage.kt [`MarketFace`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MarketFace.kt [`MusicShare`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MusicShare.kt [`Dice`]: ../mirai-core-api/src/commonMain/kotlin/message/data/Dice.kt [`FileMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/FileMessage.kt [`MessageSource`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MessageSource.kt [`QuoteReply`]: ../mirai-core-api/src/commonMain/kotlin/message/data/QuoteReply.kt [`LightApp`]: ../mirai-core-api/src/commonMain/kotlin/message/data/RichMessage.kt [`SimpleServiceMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/RichMessage.kt [`Voice`]: ../mirai-core-api/src/commonMain/kotlin/message/data/Voice.kt [`ForwardMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/ForwardMessage.kt [`ShowImageFlag`]: ../mirai-core-api/src/commonMain/kotlin/message/data/ShowImageFlag.kt [`RichMessageOrigin`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MessageOrigin.kt [`MessageOrigin`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MessageOrigin.kt | [`MessageContent`] 类型 | 解释 | `contentToString()` | 最低支持的版本 | |:------------------------:|:--------------------|:------------------------|:---------------------:| | [`PlainText`] | 纯文本 | `$content` | 2.0 | | [`Image`] | 自定义图片 | `[图片]` | 2.0 | | [`At`] | 提及某人 | `@$target` | 2.0 | | [`AtAll`] | 提及全体成员 | `@全体成员` | 2.0 | | [`Face`] | 原生表情 | `[表情对应的中文名]` | 2.0 | | [`FlashImage`] | 闪照 | `[闪照]` | 2.0 | | [`PokeMessage`] | 戳一戳消息(消息非动作) | `[戳一戳]` | 2.0 | | [`VipFace`] | VIP 表情 | `[${kind.name}]x$count` | 2.0 | | [`LightApp`] | 小程序 | `$content` | 2.0 | | [`Voice`] | 语音 | `[语音消息]` | 2.0 | | [`MarketFace`] | 商城表情 | `[表情对应的中文名]` | 2.0 | | [`ForwardMessage`] | 合并转发 | `[转发消息]` | 2.0 *(1)* | | [`SimpleServiceMessage`] | (不稳定)服务消息 | `$content` | 2.0 | | [`MusicShare`] | 音乐分享 | `[分享]曲名` | 2.1 | | [`Dice`] | 骰子 | `[骰子:$value]` | 2.5 | | [`FileMessage`] | 文件消息 | `[文件]文件名称` | 2.5 | | [`MessageMetadata`] 类型 | 解释 | 最低支持的版本 | |:-----------------------:|:------------|:-------------------:| | [`MessageSource`] | 消息来源元数据 | 2.0 | | [`QuoteReply`] | 引用回复 | 2.0 | | [`ShowImageFlag`] | 秀图标识 | 2.2 | | [`RichMessageOrigin`] | 富文本消息源 | 2.3 *(2)* | | [`MessageOrigin`] | 富文本消息源 | 2.6 | > *(1):* [`ForwardMessage`] 在 2.0 支持发送, 在 2.3 支持接收 > *(2):* [`RichMessageOrigin`] 在 2.3 增加, 在 2.6 弃用并以 [`MessageOrigin`] 替换 ### 用法 只需要得到各种类型 `Message` 的实例就可以使用,可以直接发送(`Contact.sendMessage`)也可以连接到消息链中(`Message.plus`)。 可在上文表格中找到需要的类型并在源码内文档获取更多实践上的帮助。 > 回到 [目录](#目录) ## 消息链 [`MessageChain`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MessageChain.kt [`SingleMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/SingleMessage.kt [`CodableMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/code/CodableMessage.kt 前文已经介绍消息链,这里简略介绍消息链的使用。详细的使用请查看源码内注释。 ### 发送消息 在 [Contacts 章节](Contacts.md) 提到,要发送消息使用 `Contact.sendMessage`。`Contact.sendMessage` 的定义是: ```kotlin suspend fun sendMessage(message: Message): MessageReceipt ``` 要发送字符串消息,使用:(第一部分是 Kotlin,随后是 Java,下同) ```kotlin contact.sendMessage("Hello!") ``` ```java contact.sendMessage("Hello!"); ``` 发送字符串实际上是在发送纯文本消息。上面的代码相当于: ```kotlin contact.sendMessage(PlainText("Hello!")) ``` ```java contact.sendMessage(new PlainText("Hello!")); ``` 要发送多元素消息,可将消息使用 `plus` 操作连接: ```kotlin contact.sendMessage(PlainText("你要的图片是") + Image("{f8f1ab55-bf8e-4236-b55e-955848d7069f}.png")) // 一个纯文本加一个图片 ``` ```java contact.sendMessage(new PlainText("你要的图片是:").plus(Image.fromId("{f8f1ab55-bf8e-4236-b55e-955848d7069f}.png"))); // 一个纯文本加一个图片 ``` 注: 当需要拼接的消息较多的时候, 建议使用 [构造消息链](#构造消息链) 而不是 `plus` - `plus` 在需要拼接的元素较多的时候运行效率较慢 ### 构造消息链 更复杂的消息则需要构造为消息链。 #### 在 Kotlin 构造消息链 | 定义 | |:--------------------------------------------------------| | `fun Iterable.toMessageChain(): MessageChain` | | `fun Sequence.toMessageChain(): MessageChain` | | `fun Array.toMessageChain(): MessageChain` | | `fun Message.toMessageChain(): MessageChain` | | `fun messageChainOf(vararg Message): MessageChain` | | `fun Message.plus(tail: Message): MessageChain` | 可以使用如上表格所示的方法构造,或使用 DSL builder。 ```kotlin class MessageChainBuilder : MutableList, Appendable { operator fun Message.unaryPlus() operator fun String.unaryPlus() fun add(vararg messages: Message) } ``` ```kotlin val chain = buildMessageChain { +PlainText("a") +AtAll +Image("/f8f1ab55-bf8e-4236-b55e-955848d7069f") add(At(123456)) } // chain 结果是包含 PlainText, AtAll, Image, At 的 MessageChain ``` 该示例中 `+` 是位于 `MessageChainBuilder` 的 `Message.unaryPlus` 扩展。使用 `+` 和使用 `add` 是相等的。 #### 在 Java 构造消息链 | 定义 | |:---------------------------------------------------------------------| | `public static MessageChain newChain(Iterable iterable)` | | `public static MessageChain newChain(Message iterable...)` | | `public static MessageChain newChain(Iterator iterable...)` | 方法都位于 `net.mamoe.mirai.message.data.MessageUtils`。 使用 `newChain`: ```java MessageChain chain = MessageUtils.newChain(new PlainText("Hello"), Image.fromId("{f8f1ab55-bf8e-4236-b55e-955848d7069f}.png")); ``` 使用 `MessageChainBuilder`: ```java MessageChain chain = new MessageChainBuilder() .append(new PlainText("string")) .append("string") // 会被构造成 PlainText 再添加, 相当于上一行 .append(AtAll.INSTANCE) .append(Image.fromId("{f8f1ab55-bf8e-4236-b55e-955848d7069f}.png")) .build(); ``` ### 作为字符串处理消息 通常要把消息作为字符串处理,在 Kotlin 使用 `message.content` 或在 Java 使用 `message.contentToString()`。 获取到的字符串表示只包含各 [`MessageContent`] 以官方风格显示的消息内容。如 `"你本次测试的成绩是[图片]"`、`[语音]`、`[微笑]` ### 元素唯一性 [`MessageKey`]: ../mirai-core-api/src/commonMain/kotlin/message/data/MessageKey.kt [`ConstrainSingle`]: ../mirai-core-api/src/commonMain/kotlin/message/data/ConstrainSingle.kt [`HummerMessage`]: ../mirai-core-api/src/commonMain/kotlin/message/data/HummerMessage.kt 部分元素只能单一存在于消息链中。这样的元素实现接口 [`ConstrainSingle`]。 唯一的元素例如 *消息元数据* [`MessageSource`],在连接时,新的(右侧)元素会替换旧的(左侧)元素。如: ```kotlin val source1: MessageSource val source2: MessageSource val chain: MessageChain = source1 + source2 // 结果 chain 只包含一个元素,即右侧的 source2。 ``` 元素唯一性的识别基于 [`MessageKey`]。[`MessageKey`] 拥有多态机制。元素替换时会替换。如 [`HummerMessage`] 的继承关系 ``` MessageContent ↑ HummerMessage ↑ +------------+-------------+------------+ | | | | PokeMessage VipFace FlashImage ... ``` 当连接一个 [`VipFace`] 到一个 [`MessageChain`] 时,由于 [`VipFace`] 最上层为 `MessageContent`,消息链中第一个 `MessageContent` 会被(保留顺序地)替换为 [`VipFace`],其他所有 `MessageContent` 都会被删除。 ```kotlin val chain = messageChainOf(quoteReply, plainText, at, atAll) // quoteReply 是 MessageMetadata, 其他三个都是 MessageContent val result = chain + VipFace(VipFace.AiXin, 1) // VipFace 是 ConstrainSingle,最上层键为 MessageContent,因此替换所有的 MessageContent // 结果为 [quoteReply, VipFace] ``` ### 获取消息链中的消息元素 #### A. 筛选 List [`MessageChain`] 继承接口 `List`。 ```kotlin val image: Image? = chain.filterIsInstance().firstOrNull() ``` ```java Image image = (Image) chain.stream().filter(Image.class::isInstance).findFirst().orElse(null); ``` 在 Kotlin 要获取第一个指定类型实例还可以使用快捷扩展。 ```kotlin val image: Image? = chain.findIsInstance() val image: Image = chain.firstIsInstance() // 不存在时 NoSuchElementException ``` #### B. 获取唯一消息 如果要获取 `ConstrainSingle` 的消息元素,可以快速通过键获得。 ```kotlin val quote: QuoteReply? = chain[QuoteReply] // 类似 Map.get val quote: QuoteReply = chain.getOrFail(QuoteReply) // 不存在时 NoSuchElementException ``` ```java QuoteReply quote = chain.get(QuoteReply.Key); ``` > 这是因为 `MessageKey` 一般都以消息元素的 `companion object` 实现 #### C. 使用属性委托 可在 Kotlin 使用属性委托。这样的方法与上述方法在性能上等价。 ```kotlin val image: Image by chain // 不存在时 NoSuchElementException val image: Image? by chain.orNull() val image: Image? by chain.orElse { /* 返回一个 Image */ } ``` ### 序列化 消息可以序列化为 JSON 字符串,使用 `MessageChain.serializeToJsonString` 和 `MessageChain.deserializeFromJsonString`。 ## Mirai 码 实现了接口 `CodableMessage` 的消息类型支持 mirai 码表示。 ### 转义规则 mirai 码内的属性字符串会被转义。 | 原字符 | 转义结果字符 | |:----------:|:---------:| | `[` | `\[` | | `]` | `\]` | | `:` | `\:` | | `,` | `\,` | | `\` | `\\` | | *换行符 \n* | `\n` | | *换行符 \r* | `\r` | ### 组成约定 一个有效的 mirai 码 (如 `[mirai:atall]` (无参数), `[mirai:at:123]` (有参数)) 可分为以下几个组成部分 - `[mirai:` 固定开头 - 消息类型, 如 `at` - 消息参数 - `:` 固定分隔符 - 参数内容 **(需要进行转义)** - `]` 固定结尾 #### 为何需要进行转义 为了 mirai 码的正确解析, 不转义无法正确解析原本意义 假如有以下参数 ``` {"msg": [1, 2, 3]} ``` 如果不进行转义直接进行 mirai 码拼接 (如: `[mirai:msg:{"msg": [1, 2, 3]}]`), 那么 mirai 码会被错误解析 > 解析结果如下: > > - mirai 码 `[mirai:msg:{"msg": [1, 2, 3]` > - 纯文本 `}]` ### 消息链的 mirai 码 消息链 [`MessageChain`] 是多个 [`SingleMessage`] 的集合。[`MessageChain`] 也实现 [`CodableMessage`]。在转换为 mirai 码时所有 [`CodableMessage`] 直接相连: ``` val chain = messageChainOf(PlainText("plain"), At(123), AtAll) chain.serializeToMiraiCode() // "plain[mirai:at:123][mirai:atall]" ``` ### 由 `CodableMessage` 取得 mirai 码字符串 通过 `CodableMessage.serializeToMiraiCode()`。 ``` val at = At(123) at.serializeToMiraiCode() // 结果为 `[mirai:at:123]` ``` | 消息类型 | `serializeToMiraiCode()` | |:------------------------:|:-------------------------------------------------| | [`PlainText`] | `$content` | | [`Image`] | `[mirai:image:$imageId]` | | [`At`] | `[mirai:at:$target]` | | [`AtAll`] | `[mirai:atall]` | | [`Face`] | `[mirai:face:id]` | | [`FlashImage`] | `[mirai:flash:${image.imageId}]` | | [`PokeMessage`] | `[mirai:poke:$name,$pokeType,$id]` | | [`VipFace`] | `[mirai:vipface:${kind.id},${kind.name},$count]` | | [`LightApp`] | `[mirai:app:$content]` | | [`SimpleServiceMessage`] | `[mirai:service:$serviceId,$content]` | | [`Dice`] | `[mirai:dice:$value]` | | [`MusicShare`] | `[mirai:musicshare:$args]` | | [`FileMessage`] | `[mirai:file:$id,$internalId,$name,$size]` | ### 由 mirai 码字符串取得 `MessageChain` 实例 ```kotlin val chain = "[mirai:atall]".deserializeMiraiCode() ``` ```java MessageChain chain = MiraiCode.deserializeFromMiraiCode("[mirai:atall]"); ``` ### 转义字符串 ```kotlin PlainText("[mirai:atall]").serializeToMiraiCode() // \[mirai\:atall\] ``` ```java new PlainText("[mirai:atall]").serializeToMiraiCode() // \[mirai\:atall\] ``` ### `serializeToString` 与 `toString` 的区别 - 如 [消息元素](#消息元素) 所示, `toString()` 会尽可能包含多的信息用于调试作用,**行为可能不确定** - `toString()` 偏人类可读, `serializeToString()` 偏机器可读 - `toString()` **没有转义**, `serializeToString()` 有正确转义 - `serializeToString()` 会跳过 `不支持 mirai 码处理` 的元素 - `toString()` 基本不可用于 `deserializeMiraiCode`/`MiraiCode.deserializeFromMiraiCode` --------- 到这里,你已经完成了 Mirai 所有文档的阅读。现在你已经熟悉了 Mirai,并可以开始使用了。 你可以首先构造 Bot,登录,然后从监听事件起开始创建你的机器人,或从 Bot 获取到指定群主动发送消息。在使用中遇到问题可以参考 Mirai 源码内注释,该注释会包含更多实践上的帮助。 如果你仍然对 Mirai 架构有不明确的地方,欢迎在 [#848](https://github.com/mamoe/mirai/discussions/848) 提出建议,或者直接在 PR 提交你的修改。 > 回到 [目录](#目录) > > [回到 Mirai 文档索引](CoreAPI.md)