434 lines
16 KiB
PHP
Raw Normal View History

<?php
2020-03-02 16:14:20 +08:00
declare(strict_types=1);
2020-03-02 16:14:20 +08:00
namespace ZM\API;
2022-05-04 17:53:49 +08:00
use Stringable;
use ZM\Entity\CQObject;
2018-10-29 11:47:58 +08:00
class CQ
{
/**
* at一下QQ用户仅在QQ群支持at全体
2022-03-31 02:24:38 +08:00
* @param int|string $qq 用户QQ号/ID号
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function at($qq): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('at', ['qq' => $qq]);
2018-10-29 11:47:58 +08:00
}
/**
* 发送QQ原生表情
2022-03-31 02:24:38 +08:00
* @param int|string $id 表情ID
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function face($id): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('face', ['id' => $id]);
2018-10-29 11:47:58 +08:00
}
/**
2020-12-31 13:56:59 +08:00
* 发送图片
2022-03-31 02:24:38 +08:00
* @param string $file 文件的路径、URL或者base64编码的图片数据
* @param bool $cache 是否缓存默认为true
* @param bool $flash 是否闪照默认为false
* @param bool $proxy 是否使用代理默认为true
* @param int $timeout 超时时间(默认不超时)
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function image(string $file, bool $cache = true, bool $flash = false, bool $proxy = true, int $timeout = -1): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'cache' => !$cache ? 'cache=0' : '',
'flash' => $flash ? 'type=flash' : '',
'proxy' => !$proxy ? 'proxy=false' : '',
'timeout' => $timeout != -1 ? 'timeout=' . $timeout : '',
];
return self::buildCQ('image', ['file' => $file], $optional_values);
2018-10-29 11:47:58 +08:00
}
/**
2020-12-31 13:56:59 +08:00
* 发送语音
2022-03-31 02:24:38 +08:00
* @param string $file 文件的路径、URL或者base64编码的语音数据
* @param bool $magic 是否加特技默认为false
* @param bool $cache 是否缓存默认为true
* @param bool $proxy 是否使用代理默认为true
* @param int $timeout 超时时间(默认不超时)
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function record(string $file, bool $magic = false, bool $cache = true, bool $proxy = true, int $timeout = -1): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'magic' => $magic ? 'magic=true' : '',
'cache' => !$cache ? 'cache=0' : '',
'proxy' => !$proxy ? 'proxy=false' : '',
'timeout' => $timeout != -1 ? 'timeout=' . $timeout : '',
];
return self::buildCQ('record', ['file' => $file], $optional_values);
2018-10-29 11:47:58 +08:00
}
/**
2020-12-31 13:56:59 +08:00
* 发送短视频
2022-03-31 02:24:38 +08:00
* @param string $file 文件的路径、URL或者base64编码的短视频数据
* @param bool $cache 是否缓存默认为true
* @param bool $proxy 是否使用代理默认为true
* @param int $timeout 超时时间(默认不超时)
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function video(string $file, bool $cache = true, bool $proxy = true, int $timeout = -1): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'cache' => !$cache ? 'cache=0' : '',
'proxy' => !$proxy ? 'proxy=false' : '',
'timeout' => $timeout != -1 ? 'timeout=' . $timeout : '',
];
return self::buildCQ('video', ['file' => $file], $optional_values);
2018-10-29 11:47:58 +08:00
}
/**
* 发送投掷骰子(只能在单条回复中单独使用)
2022-03-31 02:24:38 +08:00
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function rps(): string
{
return '[CQ:rps]';
2018-10-29 11:47:58 +08:00
}
/**
* 发送掷骰子表情(只能在单条回复中单独使用)
2022-03-31 02:24:38 +08:00
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function dice(): string
{
return '[CQ:dice]';
2018-10-29 11:47:58 +08:00
}
/**
* 戳一戳(原窗口抖动,仅支持好友消息使用)
2022-03-31 02:24:38 +08:00
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function shake(): string
{
return '[CQ:shake]';
2018-10-29 11:47:58 +08:00
}
2020-12-31 13:56:59 +08:00
/**
* 发送新的戳一戳
2022-03-31 02:24:38 +08:00
* @param int|string $type 焯一戳类型
* @param int|string $id 戳一戳ID号
* @param string $name 戳一戳名称(可选)
* @return string CQ码
2020-12-31 13:56:59 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function poke($type, $id, string $name = ''): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'name' => $name ? 'name=' . $name : '',
];
return self::buildCQ('poke', ['type' => $type, 'id' => $id], $optional_values);
2020-12-31 13:56:59 +08:00
}
/**
* 发送匿名消息
2022-03-31 02:24:38 +08:00
* @param int $ignore 是否忽略错误默认为10表示不忽略错误
* @return string CQ码
2020-12-31 13:56:59 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function anonymous(int $ignore = 1): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('anonymous', [], ['ignore' => $ignore != 1 ? 'ignore=0' : '']);
2020-12-31 13:56:59 +08:00
}
/**
* 发送链接分享(只能在单条回复中单独使用)
2022-03-31 02:24:38 +08:00
* @param string $url 分享地址
* @param string $title 标题
* @param null|string $content 卡片内容(可选)
* @param null|string $image 卡片图片(可选)
* @return string CQ码
2020-12-31 13:56:59 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function share(string $url, string $title, ?string $content = null, ?string $image = null): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'content' => $content ? 'content=' . self::encode($content, true) : '',
'image' => $image ? 'image=' . self::encode($image, true) : '',
];
return self::buildCQ('share', ['url' => $url, 'title' => $title], $optional_values);
2020-12-31 13:56:59 +08:00
}
/**
* 发送好友或群推荐名片
2022-05-04 17:53:49 +08:00
* @param int|string $type 名片类型
2022-03-31 02:24:38 +08:00
* @param int|string $id 好友或群ID
* @return string CQ码
2020-12-31 13:56:59 +08:00
*/
2022-05-04 17:53:49 +08:00
public static function contact($type, $id): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('contact', ['type' => $type, 'id' => $id]);
2020-12-31 13:56:59 +08:00
}
/**
* 发送位置
2022-03-31 02:24:38 +08:00
* @param float|string $lat 纬度
* @param float|string $lon 经度
* @param string $title 标题(可选)
* @param string $content 卡片内容(可选)
* @return string CQ码
*/
2022-03-31 02:24:38 +08:00
public static function location($lat, $lon, string $title = '', string $content = ''): string
{
2022-05-04 23:24:55 +08:00
$optional_values = [
'title' => $title ? 'title=' . self::encode($title, true) : '',
'content' => $content ? 'content=' . self::encode($content, true) : '',
];
return self::buildCQ('location', ['lat' => $lat, 'lon' => $lon], $optional_values);
2020-12-31 13:56:59 +08:00
}
2018-10-29 11:47:58 +08:00
/**
* 发送音乐分享(只能在单条回复中单独使用)
2022-03-31 02:24:38 +08:00
*
2018-10-29 11:47:58 +08:00
* qq、163、xiami为内置分享需要先通过搜索功能获取id后使用
2022-03-31 02:24:38 +08:00
*
* @param string $type 分享类型(仅限 `qq``163``xiami` `custom`
* @param int|string $id_or_url 当分享类型不是 `custom` 表示的是分享音乐的ID需要先通过搜索功能获取id后使用反之表示的是音乐卡片点入的链接
* @param null|string $audio 当分享类型是 `custom` 表示为音乐如mp3文件的HTTP链接地址不可为空
* @param null|string $title 当分享类型是 `custom` 表示为音乐卡片的标题建议12字以内不可为空
* @param null|string $content 当分享类型是 `custom` 时,表示为音乐卡片的简介(可忽略)
* @param null|string $image 当分享类型是 `custom` 时,表示为音乐卡片的图片链接地址(可忽略)
* @return string CQ码
2018-10-29 11:47:58 +08:00
*/
2022-03-31 02:24:38 +08:00
public static function music(string $type, $id_or_url, ?string $audio = null, ?string $title = null, ?string $content = null, ?string $image = null): string
{
2018-10-29 11:47:58 +08:00
switch ($type) {
case 'qq':
case '163':
case 'xiami':
2022-05-04 23:24:55 +08:00
return self::buildCQ('music', ['type' => $type, 'id' => $id_or_url]);
case 'custom':
2018-10-29 11:47:58 +08:00
if ($title === null || $audio === null) {
2022-06-08 23:11:17 +08:00
logger()->warning(zm_internal_errcode('E00035') . '传入CQ码实例的标题和音频链接不能为空');
return ' ';
}
2022-05-04 23:24:55 +08:00
$optional_values = [
'content' => $content ? 'content=' . self::encode($content, true) : '',
'image' => $image ? 'image=' . self::encode($image, true) : '',
];
return self::buildCQ('music', ['type' => 'custom', 'url' => $id_or_url, 'audio' => $audio, 'title' => $title], $optional_values);
2018-10-29 11:47:58 +08:00
default:
2022-06-08 23:11:17 +08:00
logger()->warning(zm_internal_errcode('E00035') . "传入的music type({$type})错误!");
return ' ';
2018-10-29 11:47:58 +08:00
}
}
2022-03-31 02:24:38 +08:00
/**
* 合并转发消息
* @param int|string $id 合并转发ID, 需要通过 `/get_forward_msg` API获取转发的具体内容
* @return string CQ码
*/
public static function forward($id): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('forward', ['id' => $id]);
2020-12-31 13:56:59 +08:00
}
2022-03-31 02:24:38 +08:00
/**
* 合并转发消息节点
* 特殊说明: 需要使用单独的API /send_group_forward_msg 发送, 并且由于消息段较为复杂, 仅支持Array形式入参。
* 如果引用消息和自定义消息同时出现, 实际查看顺序将取消息段顺序。
* 另外按 CQHTTP 文档说明, data 应全为字符串, 但由于需要接收message 类型的消息, 所以 仅限此Type的content字段 支持Array套娃
* @param int|string $user_id 转发消息id
* @param string $nickname 发送者显示名字
* @param string $content 具体消息
* @return string CQ码
2022-05-04 23:24:55 +08:00
* @deprecated 这个不推荐使用,因为 go-cqhttp 官方没有对其提供CQ码模式相关支持仅支持Array模式发送
2022-03-31 02:24:38 +08:00
*/
public static function node($user_id, string $nickname, string $content): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('node', ['user_id' => $user_id, 'nickname' => $nickname, 'content' => $content]);
}
2022-03-31 02:24:38 +08:00
/**
* XML消息
* @param string $data xml内容, xml中的value部分
* @return string CQ码
*/
public static function xml(string $data): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('xml', ['data' => $data]);
}
2022-03-31 02:24:38 +08:00
/**
* JSON消息
* @param string $data json内容
* @param int $resid 0为走小程序通道其他值为富文本通道默认为0
* @return string CQ码
*/
public static function json(string $data, int $resid = 0): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ('json', ['data' => $data, 'resid' => $resid]);
}
2022-03-31 02:24:38 +08:00
/**
* 返回一个自定义扩展的CQ码支持自定义类型和参数
* @param string $type_name CQ码类型名称
* @param array $params 参数
* @return string CQ码
*/
public static function _custom(string $type_name, array $params): string
{
2022-05-04 23:24:55 +08:00
return self::buildCQ($type_name, $params);
2018-10-29 11:47:58 +08:00
}
/**
* 反转义字符串中的CQ码敏感符号
2022-05-04 17:53:49 +08:00
* @param int|string|Stringable $msg 字符串
* @param bool $is_content 如果是解码CQ码本体内容则为false默认如果是参数内的字符串则为true
* @return string 转义后的CQ码
2018-10-29 11:47:58 +08:00
*/
2022-05-04 17:53:49 +08:00
public static function decode($msg, bool $is_content = false): string
{
2022-05-04 23:24:55 +08:00
$msg = str_replace(['&amp;', '&#91;', '&#93;'], ['&', '[', ']'], (string) $msg);
if ($is_content) {
$msg = str_replace('&#44;', ',', $msg);
}
return $msg;
2018-10-29 11:47:58 +08:00
}
2022-03-31 02:24:38 +08:00
/**
* 简单反转义替换CQ码的方括号
2022-05-04 17:53:49 +08:00
* @param int|string|Stringable $str 字符串
* @return string 字符串
2022-03-31 02:24:38 +08:00
*/
2022-05-04 17:53:49 +08:00
public static function replace($str): string
{
2022-05-04 23:24:55 +08:00
$str = str_replace('{{', '[', (string) $str);
return str_replace('}}', ']', $str);
2018-10-29 11:47:58 +08:00
}
2020-03-02 16:14:20 +08:00
/**
* 转义CQ码的特殊字符同encode
2022-05-04 17:53:49 +08:00
* @param int|string|Stringable $msg 字符串
* @param bool $is_content 如果是转义CQ码本体内容则为false默认如果是参数内的字符串则为true
* @return string 转义后的CQ码
2020-03-02 16:14:20 +08:00
*/
2022-05-04 17:53:49 +08:00
public static function escape($msg, bool $is_content = false): string
{
2022-05-04 23:24:55 +08:00
$msg = str_replace(['&', '[', ']'], ['&amp;', '&#91;', '&#93;'], (string) $msg);
if ($is_content) {
$msg = str_replace(',', '&#44;', $msg);
}
2020-03-02 16:14:20 +08:00
return $msg;
}
/**
* 转义CQ码的特殊字符
2022-05-04 17:53:49 +08:00
* @param int|string|Stringable $msg 字符串
* @param bool $is_content 如果是转义CQ码本体内容则为false默认如果是参数内的字符串则为true
* @return string 转义后的CQ码
*/
2022-05-04 17:53:49 +08:00
public static function encode($msg, bool $is_content = false): string
{
2022-05-04 23:24:55 +08:00
$msg = str_replace(['&', '[', ']'], ['&amp;', '&#91;', '&#93;'], (string) $msg);
if ($is_content) {
$msg = str_replace(',', '&#44;', $msg);
}
return $msg;
2020-05-06 15:01:32 +08:00
}
/**
* 移除消息中所有的CQ码并返回移除CQ码后的消息
2022-03-31 02:24:38 +08:00
* @param string $msg 消息
* @return string 消息内容
*/
2022-03-31 02:24:38 +08:00
public static function removeCQ(string $msg): string
{
$final = '';
$last_end = 0;
2021-06-16 00:17:30 +08:00
foreach (self::getAllCQ($msg) as $v) {
$final .= mb_substr($msg, $last_end, $v['start'] - $last_end);
$last_end = $v['end'] + 1;
}
$final .= mb_substr($msg, $last_end);
return $final;
}
2020-05-08 16:37:38 +08:00
/**
* 获取消息中第一个CQ码
2022-04-03 01:47:38 +08:00
* @param string $msg 消息内容
* @param bool $is_object 是否以对象形式返回如果为False的话返回数组形式默认为false
* @return null|array|CQObject 返回的CQ码数组或对象
*/
2022-03-31 02:24:38 +08:00
public static function getCQ(string $msg, bool $is_object = false)
{
if (($head = mb_strpos($msg, '[CQ:')) !== false) {
$key_offset = mb_substr($msg, $head);
$close = mb_strpos($key_offset, ']');
if ($close === false) {
return null;
}
$content = mb_substr($msg, $head + 4, $close + $head - mb_strlen($msg));
$exp = explode(',', $content);
$cq['type'] = array_shift($exp);
2021-06-16 00:17:30 +08:00
foreach ($exp as $v) {
$ss = explode('=', $v);
$sk = array_shift($ss);
$cq['params'][$sk] = self::decode(implode('=', $ss), true);
}
$cq['start'] = $head;
$cq['end'] = $close + $head;
return !$is_object ? $cq : CQObject::fromArray($cq);
}
return null;
}
/**
* 获取消息中所有的CQ码
2022-03-31 02:24:38 +08:00
* @param string $msg 消息内容
* @param bool $is_object 是否以对象形式返回如果为False的话返回数组形式默认为false
* @return array|CQObject[] 返回的CQ码们数组或对象
*/
2022-03-31 02:24:38 +08:00
public static function getAllCQ(string $msg, bool $is_object = false): array
{
$cqs = [];
$offset = 0;
while (($head = mb_strpos(($submsg = mb_substr($msg, $offset)), '[CQ:')) !== false) {
$key_offset = mb_substr($submsg, $head);
$tmpmsg = mb_strpos($key_offset, ']');
if ($tmpmsg === false) {
break;
} // 没闭合不算CQ码
$content = mb_substr($submsg, $head + 4, $tmpmsg + $head - mb_strlen($submsg));
$exp = explode(',', $content);
$cq = [];
$cq['type'] = array_shift($exp);
2021-06-16 00:17:30 +08:00
foreach ($exp as $v) {
$ss = explode('=', $v);
$sk = array_shift($ss);
$cq['params'][$sk] = self::decode(implode('=', $ss), true);
}
$cq['start'] = $offset + $head;
$cq['end'] = $offset + $tmpmsg + $head;
2021-11-08 22:05:41 +08:00
$offset += $head + $tmpmsg + 1;
$cqs[] = (!$is_object ? $cq : CQObject::fromArray($cq));
2020-05-08 16:37:38 +08:00
}
return $cqs;
2020-05-08 16:37:38 +08:00
}
2022-05-04 23:24:55 +08:00
private static function buildCQ(string $cq, array $array, array $optional_values = []): string
{
$str = '[CQ:' . $cq;
foreach ($array as $k => $v) {
if ($v === null) {
2022-06-08 23:11:17 +08:00
logger()->warning('param ' . $k . ' cannot be set with null, empty CQ will returned!');
2022-05-04 23:24:55 +08:00
return ' ';
}
$str .= ',' . $k . '=' . self::encode($v);
}
foreach ($optional_values as $v) {
if ($v !== '') {
$str .= ',' . $v;
}
}
return $str . ']';
}
}