koishi-plugin-node-async-bot-all 2.19.0 → 2.20.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/commands.d.ts CHANGED
@@ -33,3 +33,4 @@ export declare function serverTest(ctx: Context, session: Session): Promise<Obje
33
33
  export declare function getMeme(ctx: Context, session: Session, count: number): Promise<Number>;
34
34
  export declare function getCat(ctx: Context, session: Session): Promise<Number>;
35
35
  export declare function getQQInfo(ctx: Context, session: Session, qq: string): Promise<number>;
36
+ export declare function getMsg(ctx: Context, session: Session): Promise<number>;
package/lib/fun.d.ts CHANGED
@@ -16,3 +16,4 @@ export declare function hostPing(host: string): Promise<{
16
16
  export declare function random(type: number | undefined, data: any, data2?: any): number;
17
17
  export declare function request<T = any>(url: string, options?: RequestInit, timeout?: number, log?: any): Promise<HttpResponse<T>>;
18
18
  export declare function readUserCardFile(userInfo: APIUserInfo): Promise<string>;
19
+ export declare function readUserMsgFile(userName: string, userAvatar: string, msg: string): Promise<string>;
package/lib/index.js CHANGED
@@ -34,7 +34,7 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
34
34
  // src/locales/zh-CN.yml
35
35
  var require_zh_CN = __commonJS({
36
36
  "src/locales/zh-CN.yml"(exports2, module2) {
37
- module2.exports = { commands: { cxgame: { description: "查询服务器当前人数。", messages: { msg: "{time}{list}\n进服指南请在群公告中查看。", list: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 玩家列表:{list}\n➣ 备注:{note}", listNoPlayer: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 备注:{note}", listFailed: "【MC 服务器 {count}】\n➣ 查询失败:{data}\n➣ 请稍后重试。", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\n查询失败:{data}", timeout: "请求超时。", timeout2: "响应超时。", fewData: "服务端返回的数据过少。", close: "服务器已关闭。", error: "执行错误。", host: "没有到主机的路由。" } }, status: { description: "查询机器人状态。", messages: { msg: "{time}\n--- 系统状态 ---\n系统名称:{name}\nCPU使用率:{cpu}\n内存使用率:{memory}\n--- 机器人状态 ---\n昨日收/发消息数量:{msgCount}\n机器人版本:{version}\n运行时间:{online}", failed: "{time}\n状态获取失败。" } }, random: { description: "随机数生成器。", usage: "缺少参数时默认生成 0-10000 的随机数。\n使用示例:", examples: "random 1 128 生成1到128范围的随机数", messages: { msg: "{time}\n生成的随机数:{data}" } }, info: { description: "查询机器人信息。", messages: { msg: "{data}", failed: "{time}\n读取信息失败。" } }, rw: { description: "随机名言名句。", messages: { msg: "{time}\n{data}", error: "执行错误。", failed: "{time}\n获取失败({data})。" } }, randomba: { description: "随机BA图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。" } }, servertest: { description: "Ping服务器。", messages: { msg: "{time}\n== Ping {host} ==\n状态:{alive}\n丢包率:{packetLoss}\n返回IP:{ip}", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\nPing 失败:{data}" } }, meme: { description: "群友的怪话!", messages: { msg: "{quote}{time}\n{image}\n{title}", failed: "{quote}{time}\n获取失败:{data}", forbidden: "{quote}{time}\n此指令不允许在本群使用。", error: "执行错误。" } }, randomcat: { description: "随机猫猫图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。", failed: "{quote}{time}\n获取失败:{data}" } }, getqqinfo: { description: "获取 QQ 号的信息。", messages: { msg: "{quote}{image}", failed: "{quote}{time}\n获取失败:{data}", command: "QQ 号不正确。" } } } };
37
+ module2.exports = { commands: { cxgame: { description: "查询服务器当前人数。", messages: { msg: "{time}{list}\n进服指南请在群公告中查看。", list: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 玩家列表:{list}\n➣ 备注:{note}", listNoPlayer: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 备注:{note}", listFailed: "【MC 服务器 {count}】\n➣ 查询失败:{data}\n➣ 请稍后重试。", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\n查询失败:{data}", timeout: "请求超时。", timeout2: "响应超时。", fewData: "服务端返回的数据过少。", close: "服务器已关闭。", error: "执行错误。", host: "没有到主机的路由。" } }, status: { description: "查询机器人状态。", messages: { msg: "{time}\n--- 系统状态 ---\n系统名称:{name}\nCPU使用率:{cpu}\n内存使用率:{memory}\n--- 机器人状态 ---\n昨日收/发消息数量:{msgCount}\n机器人版本:{version}\n运行时间:{online}", failed: "{time}\n状态获取失败。" } }, random: { description: "随机数生成器。", usage: "缺少参数时默认生成 0-10000 的随机数。\n使用示例:", examples: "random 1 128 生成1到128范围的随机数", messages: { msg: "{time}\n生成的随机数:{data}" } }, info: { description: "查询机器人信息。", messages: { msg: "{data}", failed: "{time}\n读取信息失败。" } }, rw: { description: "随机名言名句。", messages: { msg: "{time}\n{data}", error: "执行错误。", failed: "{time}\n获取失败({data})。" } }, randomba: { description: "随机BA图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。" } }, servertest: { description: "Ping服务器。", messages: { msg: "{time}\n== Ping {host} ==\n状态:{alive}\n丢包率:{packetLoss}\n返回IP:{ip}", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\nPing 失败:{data}" } }, meme: { description: "群友的怪话!", messages: { msg: "{quote}{time}\n{image}\n{title}", failed: "{quote}{time}\n获取失败:{data}", forbidden: "{quote}{time}\n此指令不允许在本群使用。", error: "执行错误。" } }, randomcat: { description: "随机猫猫图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。", failed: "{quote}{time}\n获取失败:{data}" } }, getqqinfo: { description: "获取 QQ 号的信息。", messages: { msg: "{quote}{image}", failed: "{quote}{time}\n获取失败:{data}", command: "QQ 号不正确。" } }, msg2img: { description: "引用一个消息并使用此指令,即可将消息转图。", messages: { msg: "{quote}{image}", failed: "{quote}{time}\n获取失败:{data}", null: "未引用任何消息。" } } } };
38
38
  }
39
39
  });
40
40
 
@@ -269,6 +269,18 @@ async function readUserCardFile(userInfo) {
269
269
  return card;
270
270
  }
271
271
  __name(readUserCardFile, "readUserCardFile");
272
+ async function readUserMsgFile(userName, userAvatar, msg) {
273
+ let html;
274
+ try {
275
+ const aPath = import_path.default.resolve(__dirname, "..") + import_path.default.sep + "res" + import_path.default.sep + "userMsg.html";
276
+ html = await import_fs.default.promises.readFile(aPath, "utf8");
277
+ html = html.toString().replace("{userData.avatarUrl}", userAvatar).replace("{userData.username}", userName).replace("{userData.message}", msg);
278
+ } catch (error) {
279
+ html = error.message;
280
+ }
281
+ return html;
282
+ }
283
+ __name(readUserMsgFile, "readUserMsgFile");
272
284
 
273
285
  // src/commands.ts
274
286
  async function getServer(ctx, session) {
@@ -634,15 +646,27 @@ async function getQQInfo(ctx, session, qq) {
634
646
  if (response.success) {
635
647
  const fullHtml = await readUserCardFile(response.data);
636
648
  const page = await ctx.puppeteer.page();
637
- await page.setViewport({
638
- width: 450,
639
- height: 650,
640
- deviceScaleFactor: 2
641
- });
642
- await page.setContent(fullHtml, { waitUntil: "networkidle0" });
643
- const image = await page.screenshot({ type: "png", omitBackground: true });
644
- const status = await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
645
- if (!status) await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
649
+ try {
650
+ await page.setViewport({
651
+ width: 450,
652
+ height: 650,
653
+ deviceScaleFactor: 2
654
+ });
655
+ await page.setContent(fullHtml, { waitUntil: "networkidle0" });
656
+ const { width, height } = await page.evaluate(() => ({
657
+ width: document.body.scrollWidth,
658
+ height: document.body.scrollHeight
659
+ }));
660
+ await page.setViewport({ width, height, deviceScaleFactor: 2 });
661
+ const image = await page.screenshot({ type: "png", omitBackground: true });
662
+ await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
663
+ } catch (err) {
664
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "time": time, "data": "图片渲染失败" }));
665
+ log.error("图片渲染失败:", err);
666
+ return 1;
667
+ } finally {
668
+ if (page && !page.isClosed()) await page.close();
669
+ }
646
670
  log.debug("Sent: Image");
647
671
  } else {
648
672
  if (response.code) {
@@ -658,9 +682,58 @@ async function getQQInfo(ctx, session, qq) {
658
682
  return 0;
659
683
  }
660
684
  __name(getQQInfo, "getQQInfo");
685
+ async function getMsg(ctx, session) {
686
+ const log = ctx.logger("getQQInfo");
687
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
688
+ const time = getHongKongTime();
689
+ if (!session.quote || !session.quote.user) {
690
+ await session.send(session.text(".null"));
691
+ log.warn("未引用任何信息");
692
+ return 1;
693
+ }
694
+ const user = await session.bot.getUser(session.quote.user.id, session.channelId);
695
+ const msg = session.quote.content;
696
+ if (!user.name || !user.avatar) {
697
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "time": time, "data": "获取用户信息失败。" }));
698
+ log.error("获取用户信息失败");
699
+ return 1;
700
+ }
701
+ const page = await ctx.puppeteer.page();
702
+ const html = await readUserMsgFile(user.name, user.avatar, msg);
703
+ try {
704
+ await page.setViewport({
705
+ width: 400,
706
+ height: 800,
707
+ deviceScaleFactor: 2
708
+ });
709
+ await page.setContent(html, { waitUntil: "networkidle0" });
710
+ const element = await page.$("#target-element");
711
+ if (element) {
712
+ const image = await element.screenshot({
713
+ type: "png",
714
+ omitBackground: true
715
+ // 使得 CSS 中未定义的背景部分透明
716
+ });
717
+ await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
718
+ log.debug("Sent: Image");
719
+ } else {
720
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "time": time, "data": "未找到目标元素。" }));
721
+ log.error("未找到目标元素");
722
+ return 1;
723
+ }
724
+ } catch (err) {
725
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "time": time, "data": "图片渲染失败" }));
726
+ log.error("图片渲染失败:", err);
727
+ return 1;
728
+ } finally {
729
+ if (page && !page.isClosed()) await page.close();
730
+ }
731
+ return 0;
732
+ }
733
+ __name(getMsg, "getMsg");
661
734
 
662
735
  // package.json
663
- var version = "2.19.0";
736
+ var version = "2.20.0";
664
737
 
665
738
  // src/index.ts
666
739
  var inject = ["database", "installer", "puppeteer"];
@@ -776,6 +849,9 @@ function apply(ctx) {
776
849
  if (qq == void 0 || isNaN(Number(qq))) return session?.text(".command");
777
850
  await getQQInfo(ctx, session, qq);
778
851
  });
852
+ ctx.command("msg2img").alias("消息转图").alias("m").action(async ({ session }) => {
853
+ await getMsg(ctx, session);
854
+ });
779
855
  }
780
856
  __name(apply, "apply");
781
857
  // Annotate the CommonJS export names for ESM import in node:
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "koishi-plugin-node-async-bot-all",
3
3
  "description": "NodeAsync Bot插件(自用)",
4
- "version": "2.19.0",
4
+ "version": "2.20.0",
5
5
  "main": "lib/index.js",
6
6
  "typings": "lib/index.d.ts",
7
7
  "contributors": [
package/res/userCard.html CHANGED
@@ -3,7 +3,7 @@
3
3
  <head>
4
4
  <meta charset="UTF-8">
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
- <title>User Profile Card</title>
6
+ <title>UserCard</title>
7
7
  <style>
8
8
  /* 全局重置 */
9
9
  * {
@@ -0,0 +1,81 @@
1
+ <!DOCTYPE html>
2
+ <html lang="zh-CN">
3
+ <head>
4
+ <meta charset="UTF-8">
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
+ <title>UserMsg</title>
7
+ <style>
8
+ * { box-sizing: border-box; margin: 0; padding: 0; }
9
+ body {
10
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
11
+ background-color: transparent; /* 背景透明 */
12
+ padding: 20px;
13
+ display: flex;
14
+ justify-content: flex-start;
15
+ }
16
+
17
+ /* 消息容器 */
18
+ .msg-container {
19
+ display: flex;
20
+ flex-direction: row;
21
+ align-items: flex-start;
22
+ max-width: 100%;
23
+ background: #f5f5f5; /* 卡片背景色,如果是生成透明图可以去掉 */
24
+ padding: 20px;
25
+ border-radius: 12px;
26
+ /* 如果你想让图片就是气泡本身,可以去掉上面的 background 和 padding,保留下面的布局 */
27
+ }
28
+
29
+ /* 头像 */
30
+ .avatar {
31
+ width: 50px;
32
+ height: 50px;
33
+ border-radius: 50%;
34
+ background-color: #ddd;
35
+ object-fit: cover;
36
+ flex-shrink: 0; /* 防止头像被压缩 */
37
+ margin-right: 12px;
38
+ box-shadow: 0 2px 4px rgba(0,0,0,0.1);
39
+ }
40
+
41
+ /* 右侧内容区域 */
42
+ .content-wrapper {
43
+ display: flex;
44
+ flex-direction: column;
45
+ }
46
+
47
+ /* 用户名 */
48
+ .username {
49
+ font-size: 14px;
50
+ color: #666;
51
+ margin-bottom: 4px;
52
+ font-weight: 500;
53
+ }
54
+
55
+ /* 消息气泡 */
56
+ .bubble {
57
+ background-color: #fff;
58
+ color: #333;
59
+ padding: 10px 14px;
60
+ border-radius: 0 12px 12px 12px; /* 左上角直角,其余圆角 */
61
+ font-size: 16px;
62
+ line-height: 1.5;
63
+ box-shadow: 0 1px 2px rgba(0,0,0,0.05);
64
+ position: relative;
65
+ word-wrap: break-word; /* 强制换行 */
66
+ }
67
+ </style>
68
+ </head>
69
+ <body>
70
+ <!-- 目标截图元素 -->
71
+ <div id="target-element" class="msg-container">
72
+ <img class="avatar" src="{userData.avatarUrl}" />
73
+ <div class="content-wrapper">
74
+ <div class="username">{userData.username}</div>
75
+ <div class="bubble">
76
+ {userData.message}
77
+ </div>
78
+ </div>
79
+ </div>
80
+ </body>
81
+ </html>