mirror of
https://github.com/Gloridust/WechatOnCloud.git
synced 2026-06-16 19:53:53 +08:00
Compare commits
9 Commits
@@ -1,9 +1,54 @@
|
||||
# WechatOnCloud
|
||||
<div align="center">
|
||||
|
||||
在飞牛 NAS(x86_64 / arm64)上运行服务端微信:可管理**多个**微信实例,每个实例是一个独立的微信会话;多个 web 用户通过浏览器访问被授权的实例,实现跨设备消息同步、多端共享。
|
||||
<img src="doc/img/icon-192.png" width="88" height="88" alt="云微 logo" />
|
||||
|
||||
> 设计与选型详见 [技术方案.md](技术方案.md)。
|
||||
> 部署形态:拉取 GHCR 预构建多架构镜像(或本地自构建),面板按需动态创建微信实例容器。不熟悉 Docker?直接看 [Docker 运行模式详解](#docker-运行模式详解新手向)。
|
||||
<h1>云微 · WechatOnCloud</h1>
|
||||
|
||||
<p><b>在自己的 NAS / 服务器上运行「服务端微信」,多端浏览器共享同一个微信会话</b></p>
|
||||
|
||||
<p>
|
||||
<a href="https://github.com/Gloridust/WechatOnCloud/stargazers"><img src="https://img.shields.io/github/stars/Gloridust/WechatOnCloud?style=flat-square&logo=github" alt="stars" /></a>
|
||||
<a href="https://github.com/Gloridust/WechatOnCloud/releases"><img src="https://img.shields.io/github/v/release/Gloridust/WechatOnCloud?style=flat-square" alt="release" /></a>
|
||||
<a href="https://github.com/Gloridust/WechatOnCloud/issues"><img src="https://img.shields.io/github/issues/Gloridust/WechatOnCloud?style=flat-square" alt="issues" /></a>
|
||||
<img src="https://img.shields.io/badge/arch-amd64%20%7C%20arm64-2496ED?style=flat-square&logo=docker&logoColor=white" alt="arch" />
|
||||
<img src="https://img.shields.io/badge/PWA-ready-5A0FC8?style=flat-square" alt="pwa" />
|
||||
</p>
|
||||
|
||||
<p>
|
||||
<a href="#快速开始">快速开始</a> ·
|
||||
<a href="#核心特性">核心特性</a> ·
|
||||
<a href="#docker-运行模式详解新手向">运行原理</a> ·
|
||||
<a href="#安全须知必读">安全须知</a> ·
|
||||
<a href="doc/技术方案.md">技术方案</a>
|
||||
</p>
|
||||
|
||||
<table>
|
||||
<tr>
|
||||
<td width="50%"><img src="doc/img/Screenshot-1.png" alt="云微 · 面板主界面" /></td>
|
||||
<td width="50%"><img src="doc/img/Screenshot-2.png" alt="云微 · 实例桌面" /></td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
</div>
|
||||
|
||||
在飞牛 NAS(x86_64 / arm64)或任意 Docker 主机上运行服务端微信:可管理**多个**微信实例,每个实例是一个独立的微信会话;多个 web 用户通过浏览器访问被授权的实例,实现跨设备消息同步、多端共享。**不修改微信客户端。**
|
||||
|
||||
> 设计与选型详见 [技术方案.md](doc/技术方案.md)。不熟悉 Docker?直接看 [Docker 运行模式详解](#docker-运行模式详解新手向)。
|
||||
|
||||
---
|
||||
|
||||
## 核心特性
|
||||
|
||||
- 🗂️ **多实例** — 一个面板管理多个独立微信会话,每个实例独立容器 + 独立数据卷,互不干扰。
|
||||
- 👥 **多端共享 + 权限** — 多浏览器 / 设备共享同一会话;子账号体系,按账号分配可访问的实例(RBAC)。
|
||||
- 🖥️ **微信 PC 式界面** — 左侧实例栏 + 右侧内嵌桌面,侧栏可折叠,移动端自动转抽屉。
|
||||
- 📦 **微信本体运行时下载** — 镜像不打包微信,面板一键「下载安装 / 更新」带进度条;按 CPU 架构自动取包。
|
||||
- 🔁 **实例生命周期** — 启动 / 停止 / 重启 / 升级(拉新镜像重建、保留聊天记录),均在面板内一键完成。
|
||||
- 📎 **文件传输** — 原生拖拽上传 + 下载 + 删除,直达微信桌面 `~/Desktop`。
|
||||
- 🧩 **多端协作软锁** — 同一实例多人操作时自动只读 + 申请接管,避免键鼠打架。
|
||||
- 🔒 **安全优先** — 面板为唯一入口,KasmVNC 凭据服务端注入、永不下发前端;docker.sock 仅管理员可触达。
|
||||
- 📱 **PWA** — iOS「添加到主屏幕」、桌面 Chrome「安装」当原生 App。
|
||||
- 🏗️ **多架构** — amd64 / arm64 预构建镜像(GHCR + GitHub Actions 自动发布)。
|
||||
|
||||
---
|
||||
|
||||
@@ -248,7 +293,9 @@ docker buildx build --platform linux/amd64,linux/arm64 \
|
||||
|
||||
---
|
||||
|
||||
## ⚠️ 安全须知(必读)
|
||||
## 安全须知(必读)
|
||||
|
||||
> ⚠️ **这套系统暴露的是已登录的微信,请务必认真阅读本节。**
|
||||
|
||||
这套系统暴露的是**已登录的微信**——能登录面板的人就能看聊天记录、以你身份发消息。**面板还挂载了宿主的 `docker.sock`**(创建/销毁实例所需),它等同宿主 root 权限。因此:
|
||||
|
||||
@@ -257,7 +304,7 @@ docker buildx build --platform linux/amd64,linux/arm64 \
|
||||
- 实例的增删、微信安装/更新等触碰 docker 引擎的操作**仅限管理员**;docker API 绝不暴露给前端;
|
||||
- KasmVNC 凭据由面板服务端注入,**浏览器永远拿不到**;实例容器名由内部随机 ID 派生,避免注入;
|
||||
- 面板与外网之间再套一层 HTTPS 反代(飞牛自带反代 / Caddy / Nginx)获得正经 TLS;
|
||||
- 进一步加固(陌生设备验证码、审计日志、并发控制)见 [技术方案.md](技术方案.md) 第 5 节。
|
||||
- 进一步加固(陌生设备验证码、审计日志、并发控制)见 [技术方案.md](doc/技术方案.md) 第 5 节。
|
||||
|
||||
---
|
||||
|
||||
@@ -278,7 +325,7 @@ docker buildx build --platform linux/amd64,linux/arm64 \
|
||||
| 界面/消息显示成方块 | 中文字体没装好,确认实例镜像含 `fonts-noto-cjk` |
|
||||
| 微信起不来 / 黑屏 | 看实例日志 `docker logs woc-wx-<id>`;确认 `seccomp=unconfined` 与 `shm_size` 生效。微信 deb 漏声明的运行时依赖已在 Dockerfile 内置 |
|
||||
| 排查缺哪个库 | `docker exec woc-wx-<id> ldd /config/wechat/opt/wechat/wechat`,看 `not found` 项补进 Dockerfile 依赖层 |
|
||||
| 多人同时操作很乱 | 单会话多端共享、键鼠会打架。未做并发控制,建议同一时刻一人操作(见技术方案 6.1) |
|
||||
| 多人同时操作很乱 | 已内置「操作控制权」软锁:当前操作者每数秒心跳续约,其余端自动转为**只读遮罩**(仍可看画面),空闲超时(约 10s)自动释放,他人可点「申请控制」接管。仍建议同一时刻一人操作 |
|
||||
| 过段时间掉登录 | 微信桌面会话会定期失效,需手机重新扫码(见技术方案 6.2) |
|
||||
| 下载 / 更新微信失败 | 腾讯 CDN 偶发波动,重新点「下载并安装 / 更新」即可;脚本已内置主/备 CDN 自动回退 |
|
||||
| 架构不支持报错 | 微信仅提供 x86_64 / arm64;其他架构下载时会在面板状态里报错 |
|
||||
@@ -305,9 +352,11 @@ docker compose stop panel # 1) 先停面板,避免覆盖你的
|
||||
```
|
||||
|
||||
```bash
|
||||
docker compose up -d # 3) 重启,面板启动时会重置该账号
|
||||
docker compose up -d # 3) 启动,面板初始化时会重置该账号
|
||||
```
|
||||
|
||||
> ⚠️ 重置逻辑只在面板**进程启动**时执行。若你没先 `stop` 而面板仍在运行,直接 `docker compose up -d` 会因「容器无变化」而空操作(输出 `Running` 而非 `Started`),重置不会发生。此时执行 **`docker compose restart panel`**(或 `docker restart woc-panel`)强制重启即可生效。
|
||||
|
||||
重启后该账号密码被重置为 `PANEL_ADMIN_PASSWORD`(即 `.env` 的 `WOC_PASSWORD`,默认 `wechat`),并自动**解禁**、清除该标记;用此密码登录后请立即在「修改密码」改掉。日志会打印 `[store] 已重置用户 '<用户名>' 的密码`。
|
||||
|
||||
---
|
||||
@@ -319,19 +368,22 @@ WechatOnCloud/
|
||||
├── .github/workflows/
|
||||
│ └── release.yml # 打 tag / 发 Release 时构建多架构镜像并推送 GHCR
|
||||
├── docker/ # 微信实例镜像(ghcr.io/<owner>/wechat-on-cloud)
|
||||
│ ├── Dockerfile # KasmVNC base + 中文字体 + 微信运行时依赖 + 默认开 IME(不打包微信本体)
|
||||
│ ├── Dockerfile # KasmVNC base + 中文字体 + 微信运行时依赖 + xdotool + 默认开 IME(不打包微信本体)
|
||||
│ ├── wechat-ctl.sh # 运行时下载/解压/更新微信(面板经 docker exec 触发,状态写 /config/.woc-state)
|
||||
│ └── autostart # openbox 会话启动:等待微信就绪 + 常驻拉起(含崩溃自重启)
|
||||
│ ├── autostart # openbox 会话启动:常驻拉起微信(崩溃自重启)+ 最小化窗口自动复原看守
|
||||
│ └── woc-update-autostart # 启动钩子:每次启动用镜像内最新 autostart 覆盖数据卷旧副本
|
||||
├── panel/ # 自研面板(ghcr.io/<owner>/woc-panel,唯一对外入口)
|
||||
│ ├── Dockerfile # 前端 Vite 打包 + 后端 Fastify 网关(多架构)
|
||||
│ ├── server/ # Fastify:cookie 鉴权 + 账号/实例/权限 API + dockerode 管理实例 + 反代
|
||||
│ └── web/ # React + TS + PWA(牛奶布艺 + 微信绿主题)
|
||||
│ ├── server/ # Fastify:cookie 鉴权 + 账号/实例/权限/生命周期 API + dockerode + 反代
|
||||
│ └── web/ # React + TS + PWA(微信 PC 式布局,牛奶布艺 + 微信绿主题)
|
||||
├── fnos/ # 飞牛 fnOS 应用打包(.fpk 工程 + 构建说明)
|
||||
├── scripts/
|
||||
│ └── build-local.sh # 本地构建面板+微信镜像(发布前自测 / 自托管自构建)
|
||||
├── doc/ # 文档与素材
|
||||
│ ├── 技术方案.md # 完整设计文档
|
||||
│ └── img/ # logo 与界面截图
|
||||
├── docker-compose.yml # 单服务:panel(挂 docker.sock,按需创建实例)
|
||||
├── .env.example # 可选配置(账号密码、镜像版本、PUID/PGID、端口、时区)
|
||||
├── 技术方案.md # 完整设计文档
|
||||
└── README.md
|
||||
```
|
||||
|
||||
@@ -347,6 +399,30 @@ WechatOnCloud/
|
||||
- [x] 多实例管理 + 按账号的实例访问权限(RBAC)
|
||||
- [x] 预构建多架构镜像发布到 GHCR + GitHub Actions 自动化
|
||||
- [ ] 面板外层 TLS / 陌生设备验证码 / 审计日志
|
||||
- [ ] 多端并发控制(控制权令牌)
|
||||
- [x] 多端并发控制(操作控制权心跳软锁 + 只读遮罩 + 申请接管)
|
||||
- [ ] 掉登录时 web 端二维码重扫入口
|
||||
- [~] 打包成飞牛原生 fpk 分发(工程已就绪见 [fnos/](fnos/),待真实设备验证 docker.sock 权限)
|
||||
|
||||
## 致谢
|
||||
|
||||
创意启发自懒猫微服(原 Deepin 团队做的硬件产品),推荐有经济实力、追求稳定运营的朋友尝试!
|
||||
|
||||
也感谢每一位 Star / Issue / PR 的朋友——**两天突破 500 ⭐**,是继续打磨的最大动力 🙌
|
||||
|
||||
## Star History
|
||||
|
||||
<a href="https://www.star-history.com/?repos=Gloridust%2FWechatOnCloud&type=date&legend=top-left">
|
||||
<picture>
|
||||
<source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/chart?repos=Gloridust/WechatOnCloud&type=date&theme=dark&legend=top-left" />
|
||||
<source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/chart?repos=Gloridust/WechatOnCloud&type=date&legend=top-left" />
|
||||
<img alt="Star History Chart" src="https://api.star-history.com/chart?repos=Gloridust/WechatOnCloud&type=date&legend=top-left" />
|
||||
</picture>
|
||||
</a>
|
||||
|
||||
---
|
||||
|
||||
<div align="center">
|
||||
<sub>如果这个项目帮到了你,欢迎点个 ⭐ Star 支持一下 ·
|
||||
<a href="https://github.com/Gloridust/WechatOnCloud/issues">反馈问题</a> ·
|
||||
<a href="https://github.com/Gloridust/WechatOnCloud/pulls">参与贡献</a></sub>
|
||||
</div>
|
||||
Binary file not shown.
|
After Width: | Height: | Size: 905 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 916 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 2.8 KiB |
@@ -1,7 +1,6 @@
|
||||
# WechatOnCloud 技术方案
|
||||
|
||||
> 目标:在飞牛 NAS(x86_64)上运行一个服务端微信,多个 web 用户通过浏览器访问同一个微信会话,实现跨设备消息同步、多端共享,解决原生微信"一台电脑一个登录"的痛点。
|
||||
> 参考形态:懒猫微服「云微信」。
|
||||
> 目标:在飞牛 NAS(x86_64)或任何服务器上运行服务端微信,多个 web 用户通过浏览器访问同一个微信会话,实现跨设备消息同步、多端共享,解决原生微信"一台电脑一个登录"的痛点。
|
||||
|
||||
---
|
||||
|
||||
@@ -147,7 +146,6 @@
|
||||
|
||||
```
|
||||
WechatOnCloud/
|
||||
├── 技术方案.md # 本文档
|
||||
├── docker/
|
||||
│ ├── Dockerfile # Debian + Xvfb + 微信deb + KasmVNC
|
||||
│ ├── entrypoint.sh # 启动 Xvfb → WM → 微信 → KasmVNC
|
||||
@@ -183,8 +181,7 @@ WechatOnCloud/
|
||||
|
||||
- [微信 Linux 官网](https://linux.weixin.qq.com/en)
|
||||
- [腾讯上线 Linux 微信官网(x86/Arm/LoongArch)](https://finance.sina.com.cn/tech/roll/2024-11-06/doc-incvceuc0172323.shtml)
|
||||
- [懒猫微服 云微信](https://lazycat.cloud/lcmd/)
|
||||
- [开源云微信部署方案(低配版懒猫远程微信)](https://linux.do/t/topic/1306818)
|
||||
- [懒猫微服 云微信](https://lazycat.cloud/)
|
||||
- [Xpra 官方文档](https://xpra.org/manual)
|
||||
- [Xpra HTML5 客户端](https://github.com/Xpra-org/xpra-html5)
|
||||
- [飞牛 fpk 框架文档](https://developer.fnnas.com/docs/core-concepts/framework)
|
||||
+15
-15
@@ -14,7 +14,8 @@ RUN set -eux; \
|
||||
apt-get install -y --no-install-recommends \
|
||||
curl ca-certificates locales dpkg \
|
||||
fonts-wqy-zenhei fonts-wqy-microhei fonts-noto-cjk fonts-noto-color-emoji \
|
||||
libnss3 libgbm1 libasound2 libxss1; \
|
||||
libnss3 libgbm1 libasound2 libxss1 \
|
||||
xdotool; \
|
||||
sed -i 's/# zh_CN.UTF-8 UTF-8/zh_CN.UTF-8 UTF-8/' /etc/locale.gen; \
|
||||
locale-gen; \
|
||||
apt-get clean; \
|
||||
@@ -53,28 +54,27 @@ ENV LANG=zh_CN.UTF-8 \
|
||||
LC_ALL=zh_CN.UTF-8 \
|
||||
LIBGL_ALWAYS_SOFTWARE=1
|
||||
|
||||
# 让 KasmVNC web 客户端默认开启 IME 输入模式:
|
||||
# 用户用本地(客户端)输入法打中文,拼音联想在本地完成、只把成品汉字发进容器,无需容器内装 IME。
|
||||
# 默认值仅在浏览器未存过该设置时生效,不会覆盖用户手动改过的偏好。
|
||||
# 改 KasmVNC web 客户端的 webpack 产物 dist/*.bundle.js:
|
||||
# (1) 默认开启 IME 输入模式:本地(客户端)输入法打中文,拼音联想在本地完成、只把成品汉字
|
||||
# 发进容器,无需容器内装 IME。默认值仅在浏览器未存过该设置时生效,不覆盖用户手动改过的偏好。
|
||||
# (2) 修复 noVNC 中文 IME 输入:原实现靠隐藏 textarea 差分逐字符重发 keysym,会泄漏中间拼音、
|
||||
# 累积不 reset、退格风暴,导致大量丢字 / ~21 字卡住 / 跨浏览器不稳。改为只在 compositionend
|
||||
# 用 e.data 直发成品字符串(详见 woc-www-patch.sh / woc-ime.pl)。
|
||||
# 注意:实际加载的是 webpack 产物 dist/main.bundle.js(app/ui.js 是未打包源码、运行时不加载),故必须改 bundle。
|
||||
# 末尾的 grep 作为断言:若 base 镜像换了打包结构、改不到任何文件,则构建直接失败而非静默放过。
|
||||
RUN set -eux; \
|
||||
patched=0; \
|
||||
for f in /usr/share/kasmvnc/www/dist/*.bundle.js /usr/local/share/kasmvnc/www/dist/*.bundle.js; do \
|
||||
if [ -f "$f" ] && grep -q "initSetting('enable_ime', false)" "$f"; then \
|
||||
sed -i "s/initSetting('enable_ime', false)/initSetting('enable_ime', true)/g" "$f"; \
|
||||
patched=1; \
|
||||
fi; \
|
||||
done; \
|
||||
[ "$patched" = "1" ]
|
||||
COPY woc-www-patch.sh woc-ime.pl /woc/
|
||||
RUN chmod +x /woc/woc-www-patch.sh && /woc/woc-www-patch.sh
|
||||
|
||||
# 微信下载/解压控制脚本(运行时由面板经 docker exec 触发,状态写入数据卷 /config/.woc-state)
|
||||
COPY wechat-ctl.sh /woc/wechat-ctl.sh
|
||||
RUN chmod +x /woc/wechat-ctl.sh
|
||||
|
||||
# openbox 会话启动时执行此脚本:等待微信就绪 + 常驻拉起微信
|
||||
# openbox 会话启动时执行此脚本:等待微信就绪 + 常驻拉起微信 + 最小化自动复原看守
|
||||
COPY autostart /defaults/autostart
|
||||
RUN chmod +x /defaults/autostart
|
||||
|
||||
# 启动钩子:每次启动用镜像内最新 autostart 覆盖数据卷旧副本(否则旧实例升级后用不上新逻辑)
|
||||
COPY woc-update-autostart /custom-cont-init.d/01-woc-autostart
|
||||
RUN chmod +x /custom-cont-init.d/01-woc-autostart
|
||||
|
||||
# 3000 = HTTP web 客户端, 3001 = HTTPS
|
||||
EXPOSE 3000 3001
|
||||
|
||||
@@ -10,6 +10,19 @@ WECHAT_BIN=/config/wechat/opt/wechat/wechat
|
||||
# 容器内无 GPU,强制软件渲染
|
||||
export LIBGL_ALWAYS_SOFTWARE=1
|
||||
|
||||
# 防“最小化后丢失”:本桌面(openbox)无任务栏,微信被最小化就无处恢复 → 黑屏。
|
||||
# 看守进程:每 2s 把“被最小化(不可见)”的顶层窗口重新激活,相当于禁用最小化。
|
||||
(
|
||||
export DISPLAY="${DISPLAY:-:1}"
|
||||
while sleep 2; do
|
||||
all=$(xdotool search --name '.+' 2>/dev/null) || continue
|
||||
vis=$(xdotool search --onlyvisible --name '.+' 2>/dev/null)
|
||||
for w in ${all}; do
|
||||
printf '%s\n' "${vis}" | grep -qx "${w}" || xdotool windowactivate "${w}" 2>/dev/null || true
|
||||
done
|
||||
done
|
||||
) &
|
||||
|
||||
# 1) 等待微信安装就绪(首次需在面板点「下载并安装」)
|
||||
notified=0
|
||||
while [ ! -x "${WECHAT_BIN}" ]; do
|
||||
|
||||
@@ -0,0 +1,45 @@
|
||||
# perl -0777 -pe 补丁脚本(被 woc-www-patch.sh 引用)。
|
||||
# 对 dist/*.bundle.js 里 noVNC 键盘 IME 逻辑做两处替换,全程字面匹配(\Q..\E)。
|
||||
#
|
||||
# 背景:noVNC 原实现靠"隐藏 textarea 差分→逐字符重发 keysym"还原 IME 输入,会在合成过程中
|
||||
# 把中间拼音也发给远端、且永不 reset 导致累积+退格风暴 → 大量丢字 / 卡住 / 跨浏览器不稳。
|
||||
#
|
||||
# 改法:彻底不靠 textarea 差分还原中文。
|
||||
# - 合成进行中(input 事件):只同步 _lastKeyboardInput、不发送(避免中间拼音泄漏 / 丢字)。
|
||||
# - 提交时(compositionend):用 e.data(最终上屏字符串)逐字发 keysym,并把 _lastKeyboardInput
|
||||
# 同步成当前 textarea 值。不 reset、不吞键——避免吞掉下一个词的首键、避免打断下一次合成。
|
||||
# - 若个别浏览器在 compositionend 后还补发一次"提交 input":此时 isComposing/_imeHold 均为假,
|
||||
# 落到非 IME 差分分支,但 newValue 与刚同步的 _lastKeyboardInput 相等 → 差分为空 → 不重复发送。
|
||||
|
||||
# (A) _handleCompositionEnd:提交时 e.data 直发 + 同步 _lastKeyboardInput(不 reset、不吞键)
|
||||
s~\Q if (this._enableIME) {
|
||||
this._imeInProgress = false;
|
||||
}
|
||||
|
||||
if (isChromiumBased()) {
|
||||
this._imeHold = false;
|
||||
}\E~ if (this._enableIME) { // WOC-IME
|
||||
this._imeInProgress = false;
|
||||
var _wocStr = (e && typeof e.data === "string") ? e.data : "";
|
||||
for (var _wocI = 0; _wocI < _wocStr.length; _wocI++) {
|
||||
this._sendKeyEvent(keysymdef.lookup(_wocStr.charCodeAt(_wocI)), 'Unidentified', true);
|
||||
this._sendKeyEvent(keysymdef.lookup(_wocStr.charCodeAt(_wocI)), 'Unidentified', false);
|
||||
}
|
||||
this._imeHold = false;
|
||||
this._lastKeyboardInput = e.target.value;
|
||||
return;
|
||||
}
|
||||
|
||||
if (isChromiumBased()) {
|
||||
this._imeHold = false;
|
||||
}~;
|
||||
|
||||
# (B) _handleInput 顶部守卫:合成进行中只同步值、不发送;提交已在 compositionend 完成。
|
||||
s~\Q if (this._enableIME && this._imeHold) {
|
||||
Debug("IME input change, sending differential");\E~ if (this._enableIME && (e.isComposing || this._imeHold || this._imeInProgress)) { // WOC-IME
|
||||
this._lastKeyboardInput = e.target.value;
|
||||
return;
|
||||
}
|
||||
|
||||
if (this._enableIME && this._imeHold) {
|
||||
Debug("IME input change, sending differential");~;
|
||||
@@ -0,0 +1,9 @@
|
||||
#!/usr/bin/with-contenv bash
|
||||
# linuxserver 启动钩子(/custom-cont-init.d,root 身份,每次启动执行)。
|
||||
# 作用:始终用镜像内最新的 /defaults/autostart 覆盖数据卷里的副本。
|
||||
# 原因:baseimage 的 init-kasmvnc-config 只在 /config/.config/openbox/autostart "缺失时"才拷贝,
|
||||
# 导致旧实例(卷里已有旧 autostart)升级镜像后仍跑旧逻辑(如缺少"最小化自动复原"看守)。
|
||||
mkdir -p /config/.config/openbox
|
||||
cp /defaults/autostart /config/.config/openbox/autostart
|
||||
chmod +x /config/.config/openbox/autostart
|
||||
chown "${PUID:-1000}:${PGID:-1000}" /config/.config/openbox/autostart 2>/dev/null || true
|
||||
@@ -0,0 +1,43 @@
|
||||
#!/usr/bin/env bash
|
||||
# 构建期补丁:改 KasmVNC web 客户端的 webpack 产物 dist/*.bundle.js
|
||||
# (1) 默认开启 IME 输入模式(本地输入法打中文,成品汉字发进容器,容器内不装 IME)
|
||||
# (2) 修复 noVNC 的中文 IME 输入:原实现靠"隐藏 textarea 差分→逐字符重发 keysym",
|
||||
# 会在合成过程中把中间拼音也发给远端、且永不 reset 导致累积+退格风暴,
|
||||
# 在 VNC 上表现为大量丢字、~21 字后卡住、跨浏览器不稳定。
|
||||
# 改为:合成期间(input)一律不发;只在 compositionend 用 e.data(最终上屏串)逐字发 keysym,
|
||||
# 提交后 reset textarea,并吞掉紧随其后的提交 input 事件,避免重复发送/跨分支竞争。
|
||||
# 末尾断言:若 base 镜像换了打包结构、一个文件都没改到,则构建失败而非静默放过。
|
||||
set -euo pipefail
|
||||
|
||||
PATCH_PL="$(dirname "$0")/woc-ime.pl"
|
||||
patched=0
|
||||
|
||||
for f in /usr/share/kasmvnc/www/dist/*.bundle.js /usr/local/share/kasmvnc/www/dist/*.bundle.js; do
|
||||
[ -f "$f" ] || continue
|
||||
changed=0
|
||||
|
||||
# (1) enable_ime 默认开启
|
||||
if grep -q "initSetting('enable_ime', false)" "$f"; then
|
||||
sed -i "s/initSetting('enable_ime', false)/initSetting('enable_ime', true)/g" "$f"
|
||||
changed=1
|
||||
fi
|
||||
|
||||
# (2) IME 差分逻辑修复(仅含 noVNC 键盘逻辑的 bundle)
|
||||
# 幂等:/usr/share/kasmvnc 是 /usr/local/share/kasmvnc 的软链,两个 glob 会命中同一 inode,
|
||||
# 故已含 _imeJustCommitted 的文件直接跳过,避免重复注入守卫块。
|
||||
if grep -q "IME input change, sending differential" "$f" && ! grep -q "WOC-IME" "$f"; then
|
||||
perl -0777 -i -pe "$(cat "$PATCH_PL")" "$f"
|
||||
after="$(grep -c "WOC-IME" "$f" || true)"
|
||||
# 断言两处替换都命中(compositionend 标记 1 + _handleInput 守卫标记 1 = 2 行)
|
||||
if [ "$after" -ne 2 ]; then
|
||||
echo "FATAL: IME patch mismatch on $f (markers=$after, expect 2)" >&2
|
||||
exit 1
|
||||
fi
|
||||
changed=1
|
||||
fi
|
||||
|
||||
[ "$changed" = "1" ] && { echo "woc-www-patch: patched $f"; patched=1; }
|
||||
done
|
||||
|
||||
[ "$patched" = "1" ] || { echo "FATAL: no bundle patched" >&2; exit 1; }
|
||||
echo "woc-www-patch: done"
|
||||
@@ -125,6 +125,26 @@ export async function ensureRunning(inst: Instance): Promise<void> {
|
||||
}
|
||||
}
|
||||
|
||||
// 升级实例:拉取最新微信镜像后重建容器(保留数据卷 → 登录态不丢)。
|
||||
// 拉取失败(本地自构建 / 离线 / 仓库不可达)则用本地现有镜像重建,不阻断。
|
||||
export async function upgradeInstance(inst: Instance): Promise<void> {
|
||||
try {
|
||||
await pullImage();
|
||||
} catch (e: any) {
|
||||
console.warn('[docker] 升级时拉取镜像失败,改用本地镜像重建:', e?.message || e);
|
||||
}
|
||||
await runInstance(inst);
|
||||
}
|
||||
|
||||
// 停止实例容器(保留容器与数据卷,可再启动)。
|
||||
export async function stopInstance(inst: Instance): Promise<void> {
|
||||
try {
|
||||
await docker.getContainer(inst.containerName).stop({ t: 5 } as any);
|
||||
} catch {
|
||||
/* 已停止或不存在 */
|
||||
}
|
||||
}
|
||||
|
||||
export async function removeInstance(inst: Instance, purgeVolume: boolean): Promise<void> {
|
||||
try {
|
||||
const c = docker.getContainer(inst.containerName);
|
||||
@@ -269,6 +289,12 @@ export async function listInstanceFiles(inst: Instance): Promise<TransferFile[]>
|
||||
});
|
||||
}
|
||||
|
||||
export async function deleteInstanceFile(inst: Instance, name: string): Promise<void> {
|
||||
if (!safeName(name)) throw new Error('文件名不合法');
|
||||
// argv 数组直传,不经 shell;safeName 已排除路径穿越
|
||||
await execCapture(inst, ['rm', '-f', `${TRANSFER_DIR}/${name}`]);
|
||||
}
|
||||
|
||||
export async function downloadFromInstance(inst: Instance, name: string): Promise<Buffer> {
|
||||
if (!safeName(name)) throw new Error('文件名不合法');
|
||||
const c = docker.getContainer(inst.containerName);
|
||||
@@ -286,6 +312,22 @@ export async function downloadFromInstance(inst: Instance, name: string): Promis
|
||||
return tar.subarray(512, 512 + size);
|
||||
}
|
||||
|
||||
// 拉取实例容器日志(末尾 N 行),供前端"查看/导出日志"排错。
|
||||
export async function instanceLogs(inst: Instance, tail = 600): Promise<string> {
|
||||
const c = docker.getContainer(inst.containerName);
|
||||
const buf = (await c.logs({ stdout: true, stderr: true, tail, timestamps: true })) as unknown as Buffer;
|
||||
// docker 非 TTY 日志为多路复用流:每帧 8 字节头([stream,0,0,0,size BE])+ 负载;解出纯文本。
|
||||
let out = '';
|
||||
let i = 0;
|
||||
while (i + 8 <= buf.length) {
|
||||
const size = buf.readUInt32BE(i + 4);
|
||||
if (size < 0 || i + 8 + size > buf.length) break;
|
||||
out += buf.subarray(i + 8, i + 8 + size).toString('utf8');
|
||||
i += 8 + size;
|
||||
}
|
||||
return out || buf.toString('utf8'); // 兜底:TTY 模式非多路复用
|
||||
}
|
||||
|
||||
// 实例容器名(供反代构造 target)。
|
||||
export function instanceTarget(inst: Instance): string {
|
||||
return `http://${inst.containerName}:3000`;
|
||||
|
||||
+118
-1
@@ -34,6 +34,8 @@ import {
|
||||
ensureNetwork,
|
||||
ensureRunning,
|
||||
runInstance,
|
||||
stopInstance,
|
||||
upgradeInstance,
|
||||
removeInstance as removeInstanceContainer,
|
||||
instanceRuntime,
|
||||
triggerWechat,
|
||||
@@ -42,6 +44,8 @@ import {
|
||||
uploadToInstance,
|
||||
listInstanceFiles,
|
||||
downloadFromInstance,
|
||||
deleteInstanceFile,
|
||||
instanceLogs,
|
||||
} from './docker.js';
|
||||
import { createSession, getSession, destroySession, destroyUserSessions } from './sessions.js';
|
||||
|
||||
@@ -248,6 +252,7 @@ app.delete('/api/admin/instances/:id', async (req, reply) => {
|
||||
if (!inst) return reply.code(404).send({ error: '实例不存在' });
|
||||
await removeInstanceContainer(inst, purge);
|
||||
removeInstanceRecord(id);
|
||||
controlHolders.delete(id);
|
||||
return { ok: true };
|
||||
});
|
||||
|
||||
@@ -262,7 +267,7 @@ app.post('/api/admin/instances/:id/rename', async (req, reply) => {
|
||||
}
|
||||
});
|
||||
|
||||
// 启动/重启实例容器(仅管理员):容器停止或被删后,一键拉起(不重建数据卷)。
|
||||
// 启动实例容器(仅管理员):容器停止或被删后,一键拉起(不重建数据卷)。
|
||||
app.post('/api/admin/instances/:id/start', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
const inst = findInstance((req.params as any).id);
|
||||
@@ -275,6 +280,46 @@ app.post('/api/admin/instances/:id/start', async (req, reply) => {
|
||||
}
|
||||
});
|
||||
|
||||
// 停止实例容器(仅管理员):保留容器与数据卷。
|
||||
app.post('/api/admin/instances/:id/stop', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
const inst = findInstance((req.params as any).id);
|
||||
if (!inst) return reply.code(404).send({ error: '实例不存在' });
|
||||
try {
|
||||
await stopInstance(inst);
|
||||
return { ok: true };
|
||||
} catch (e: any) {
|
||||
return reply.code(500).send({ error: '停止失败:' + (e?.message || e) });
|
||||
}
|
||||
});
|
||||
|
||||
// 重启实例容器(仅管理员):按当前本地镜像重建(保留数据卷 → 登录态不丢;快速,不联网拉取)。
|
||||
app.post('/api/admin/instances/:id/restart', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
const inst = findInstance((req.params as any).id);
|
||||
if (!inst) return reply.code(404).send({ error: '实例不存在' });
|
||||
try {
|
||||
await runInstance(inst);
|
||||
return { ok: true };
|
||||
} catch (e: any) {
|
||||
return reply.code(500).send({ error: '重启失败:' + (e?.message || e) });
|
||||
}
|
||||
});
|
||||
|
||||
// 升级实例(仅管理员):拉取最新微信镜像后重建(保留数据卷)。用于把旧实例更新到新版镜像
|
||||
// (如修复"最小化丢失"等),类似「更新微信」但更新的是实例容器镜像本身。
|
||||
app.post('/api/admin/instances/:id/upgrade', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
const inst = findInstance((req.params as any).id);
|
||||
if (!inst) return reply.code(404).send({ error: '实例不存在' });
|
||||
try {
|
||||
await upgradeInstance(inst);
|
||||
return { ok: true };
|
||||
} catch (e: any) {
|
||||
return reply.code(500).send({ error: '升级失败:' + (e?.message || e) });
|
||||
}
|
||||
});
|
||||
|
||||
// 实例侧:设置该实例可被哪些账户访问
|
||||
app.post('/api/admin/instances/:id/users', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
@@ -319,6 +364,21 @@ app.get('/api/instances/:id/files', async (req, reply) => {
|
||||
}
|
||||
});
|
||||
|
||||
// 删除某个中转文件(有访问权限即可)
|
||||
app.delete('/api/instances/:id/files', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
if (!u) return;
|
||||
const id = (req.params as any).id;
|
||||
if (!userCanAccess(u, id)) return reply.code(403).send({ error: '无权访问该实例' });
|
||||
const name = String((req.query as any)?.name || '').trim();
|
||||
try {
|
||||
await deleteInstanceFile(findInstance(id)!, name);
|
||||
return { ok: true };
|
||||
} catch (e: any) {
|
||||
return reply.code(400).send({ error: e?.message || '删除失败' });
|
||||
}
|
||||
});
|
||||
|
||||
// 下载某个中转文件
|
||||
app.get('/api/instances/:id/download', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
@@ -336,6 +396,63 @@ app.get('/api/instances/:id/download', async (req, reply) => {
|
||||
}
|
||||
});
|
||||
|
||||
// ---------- 多端协作:操作控制权(心跳软锁,避免多人同时操作打架) ----------
|
||||
// 同一实例被多个浏览器连的是同一会话,键鼠会互相打架。这里用"心跳持锁":
|
||||
// 当前操作者每隔几秒 beat 续约;TTL 内他人只读(前端盖只读遮罩)。空闲超 TTL 自动释放。
|
||||
const CONTROL_TTL = 10_000; // ms:超过则视为已空闲,可被接管
|
||||
const controlHolders = new Map<string, { userId: string; username: string; at: number }>();
|
||||
|
||||
// 续约/认领:无人持有、已超时、或本来就是我 → 我成为操作者;否则返回当前操作者。
|
||||
app.post('/api/instances/:id/control/beat', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
if (!u) return;
|
||||
const id = (req.params as any).id;
|
||||
if (!userCanAccess(u, id)) return reply.code(403).send({ error: '无权访问该实例' });
|
||||
const now = Date.now();
|
||||
const h = controlHolders.get(id);
|
||||
if (!h || now - h.at > CONTROL_TTL || h.userId === u.id) {
|
||||
controlHolders.set(id, { userId: u.id, username: u.username, at: now });
|
||||
return { mine: true, holder: u.username };
|
||||
}
|
||||
return { mine: false, holder: h.username };
|
||||
});
|
||||
|
||||
// 只读查询当前操作者(前端轮询;不认领)。超 TTL 视为空闲。
|
||||
app.get('/api/instances/:id/control', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
if (!u) return;
|
||||
const id = (req.params as any).id;
|
||||
if (!userCanAccess(u, id)) return reply.code(403).send({ error: '无权访问该实例' });
|
||||
const h = controlHolders.get(id);
|
||||
if (!h || Date.now() - h.at > CONTROL_TTL) return { free: true, mine: false, holder: null };
|
||||
return { free: false, mine: h.userId === u.id, holder: h.username };
|
||||
});
|
||||
|
||||
// 主动接管("申请控制"):强制把操作权抢过来。
|
||||
app.post('/api/instances/:id/control/take', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
if (!u) return;
|
||||
const id = (req.params as any).id;
|
||||
if (!userCanAccess(u, id)) return reply.code(403).send({ error: '无权访问该实例' });
|
||||
controlHolders.set(id, { userId: u.id, username: u.username, at: Date.now() });
|
||||
return { mine: true, holder: u.username };
|
||||
});
|
||||
|
||||
// 查看实例容器日志(仅管理员):排查"无法进入/未安装/卡死"等。inline 文本,浏览器可直接看/另存。
|
||||
app.get('/api/admin/instances/:id/logs', async (req, reply) => {
|
||||
if (!requireAdmin(req, reply)) return;
|
||||
const inst = findInstance((req.params as any).id);
|
||||
if (!inst) return reply.code(404).send({ error: '实例不存在' });
|
||||
try {
|
||||
const text = await instanceLogs(inst);
|
||||
reply.header('content-type', 'text/plain; charset=utf-8');
|
||||
return reply.send(text || '(暂无日志)');
|
||||
} catch (e: any) {
|
||||
reply.header('content-type', 'text/plain; charset=utf-8');
|
||||
return reply.send('获取日志失败:' + (e?.message || e));
|
||||
}
|
||||
});
|
||||
|
||||
// 该实例的微信安装状态(有访问权限即可看)
|
||||
app.get('/api/instances/:id/wechat/status', async (req, reply) => {
|
||||
const u = requireAuth(req, reply);
|
||||
|
||||
+4
-24
@@ -2,9 +2,7 @@ import { Navigate, Route, Routes } from 'react-router-dom';
|
||||
import { AuthProvider, useAuth } from './auth';
|
||||
import { UIProvider } from './ui';
|
||||
import Login from './pages/Login';
|
||||
import Dashboard from './pages/Dashboard';
|
||||
import Admin from './pages/Admin';
|
||||
import Desktop from './pages/Desktop';
|
||||
import AppShell from './AppShell';
|
||||
import type { ReactNode } from 'react';
|
||||
|
||||
function Splash() {
|
||||
@@ -15,11 +13,10 @@ function Splash() {
|
||||
);
|
||||
}
|
||||
|
||||
function RequireAuth({ children, admin }: { children: ReactNode; admin?: boolean }) {
|
||||
function RequireAuth({ children }: { children: ReactNode }) {
|
||||
const { user, loading } = useAuth();
|
||||
if (loading) return <Splash />;
|
||||
if (!user) return <Navigate to="/login" replace />;
|
||||
if (admin && user.role !== 'admin') return <Navigate to="/" replace />;
|
||||
return <>{children}</>;
|
||||
}
|
||||
|
||||
@@ -28,30 +25,13 @@ function Shell() {
|
||||
<Routes>
|
||||
<Route path="/login" element={<Login />} />
|
||||
<Route
|
||||
path="/"
|
||||
path="/*"
|
||||
element={
|
||||
<RequireAuth>
|
||||
<Dashboard />
|
||||
<AppShell />
|
||||
</RequireAuth>
|
||||
}
|
||||
/>
|
||||
<Route
|
||||
path="/admin"
|
||||
element={
|
||||
<RequireAuth admin>
|
||||
<Admin />
|
||||
</RequireAuth>
|
||||
}
|
||||
/>
|
||||
<Route
|
||||
path="/desktop/:id"
|
||||
element={
|
||||
<RequireAuth>
|
||||
<Desktop />
|
||||
</RequireAuth>
|
||||
}
|
||||
/>
|
||||
<Route path="*" element={<Navigate to="/" replace />} />
|
||||
</Routes>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -0,0 +1,337 @@
|
||||
import { createContext, useContext, useEffect, useRef, useState, type ReactNode } from 'react';
|
||||
import { Navigate, Route, Routes, useLocation, useNavigate } from 'react-router-dom';
|
||||
import { useAuth } from './auth';
|
||||
import { useUI, PasswordInput } from './ui';
|
||||
import { api, type InstanceWithStatus } from './api';
|
||||
import InstanceView from './pages/Desktop';
|
||||
import Admin from './pages/Admin';
|
||||
|
||||
const BUSY = ['downloading', 'extracting', 'installing'];
|
||||
|
||||
// ---- 实例数据:侧栏 / 主页 / 实例视图共享,安装中轮询 ----
|
||||
interface InstancesState {
|
||||
instances: InstanceWithStatus[];
|
||||
loaded: boolean;
|
||||
reload: () => Promise<void>;
|
||||
}
|
||||
const InstancesCtx = createContext<InstancesState>({ instances: [], loaded: false, reload: async () => {} });
|
||||
export const useInstances = () => useContext(InstancesCtx);
|
||||
|
||||
function useInstancesLoader(): InstancesState {
|
||||
const [instances, setInstances] = useState<InstanceWithStatus[]>([]);
|
||||
const [loaded, setLoaded] = useState(false);
|
||||
const timer = useRef<number | undefined>(undefined);
|
||||
const reload = async () => {
|
||||
try {
|
||||
const { instances } = await api.listInstances();
|
||||
setInstances(instances);
|
||||
} catch {
|
||||
/* 401 会被 api 层重定向到登录 */
|
||||
} finally {
|
||||
setLoaded(true);
|
||||
}
|
||||
};
|
||||
useEffect(() => {
|
||||
reload();
|
||||
return () => window.clearTimeout(timer.current);
|
||||
}, []);
|
||||
useEffect(() => {
|
||||
window.clearTimeout(timer.current);
|
||||
if (instances.some((i) => BUSY.includes(i.wechat.phase))) timer.current = window.setTimeout(reload, 1500);
|
||||
return () => window.clearTimeout(timer.current);
|
||||
}, [instances]);
|
||||
return { instances, loaded, reload };
|
||||
}
|
||||
|
||||
// 实例状态点(颜色 + 文案)
|
||||
export function statusOf(inst: InstanceWithStatus): { cls: string; text: string } {
|
||||
const offline = inst.runtime !== 'running';
|
||||
if (offline) return { cls: 'st-off', text: inst.runtime === 'missing' ? '未创建' : '已停止' };
|
||||
if (BUSY.includes(inst.wechat.phase)) return { cls: 'st-busy', text: '处理中' };
|
||||
if (inst.wechat.installed) return { cls: 'st-on', text: '在线' };
|
||||
return { cls: 'st-warn', text: '待安装' };
|
||||
}
|
||||
|
||||
// ---- 图标 ----
|
||||
const Icon = {
|
||||
home: (
|
||||
<svg viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||
<path d="M3 10.5 12 3l9 7.5" /><path d="M5 9.5V20h14V9.5" /><path d="M9.5 20v-6h5v6" />
|
||||
</svg>
|
||||
),
|
||||
gear: (
|
||||
<svg viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||
<circle cx="12" cy="12" r="3.2" /><path d="M19.4 13a1.6 1.6 0 0 0 .3 1.8l.1.1a2 2 0 1 1-2.8 2.8l-.1-.1a1.6 1.6 0 0 0-2.7 1.1V21a2 2 0 1 1-4 0v-.1a1.6 1.6 0 0 0-2.7-1.1l-.1.1a2 2 0 1 1-2.8-2.8l.1-.1a1.6 1.6 0 0 0-1.1-2.7H3a2 2 0 1 1 0-4h.1a1.6 1.6 0 0 0 1.1-2.7l-.1-.1A2 2 0 1 1 6.9 4.5l.1.1a1.6 1.6 0 0 0 1.8.3H9a1.6 1.6 0 0 0 1-1.5V3a2 2 0 1 1 4 0v.1a1.6 1.6 0 0 0 2.7 1.1l.1-.1a2 2 0 1 1 2.8 2.8l-.1.1a1.6 1.6 0 0 0-.3 1.8V9a1.6 1.6 0 0 0 1.5 1H21a2 2 0 1 1 0 4h-.1a1.6 1.6 0 0 0-1.5 1z" />
|
||||
</svg>
|
||||
),
|
||||
logout: (
|
||||
<svg viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||
<path d="M9 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h4" /><path d="M16 17l5-5-5-5" /><path d="M21 12H9" />
|
||||
</svg>
|
||||
),
|
||||
collapse: (
|
||||
<svg viewBox="0 0 24 24" width="20" height="20" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||
<rect x="3" y="4" width="18" height="16" rx="2.5" /><path d="M9 4v16" />
|
||||
</svg>
|
||||
),
|
||||
menu: (
|
||||
<svg viewBox="0 0 24 24" width="22" height="22" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round">
|
||||
<path d="M4 6h16M4 12h16M4 18h16" />
|
||||
</svg>
|
||||
),
|
||||
};
|
||||
|
||||
export default function AppShell() {
|
||||
const state = useInstancesLoader();
|
||||
const { refresh } = useAuth();
|
||||
const [collapsed, setCollapsed] = useState(() => localStorage.getItem('woc_sb_collapsed') === '1');
|
||||
const [drawer, setDrawer] = useState(false);
|
||||
const [showPw, setShowPw] = useState(false);
|
||||
const [isDesktop, setIsDesktop] = useState(() => window.matchMedia('(min-width: 768px)').matches);
|
||||
const loc = useLocation();
|
||||
|
||||
useEffect(() => {
|
||||
const m = window.matchMedia('(min-width: 768px)');
|
||||
const h = () => setIsDesktop(m.matches);
|
||||
m.addEventListener('change', h);
|
||||
return () => m.removeEventListener('change', h);
|
||||
}, []);
|
||||
|
||||
useEffect(() => setDrawer(false), [loc.pathname]); // 路由变化关抽屉
|
||||
|
||||
// 移动端不收成窄栏(改用抽屉);折叠仅桌面生效
|
||||
const railed = collapsed && isDesktop;
|
||||
|
||||
const toggleCollapsed = () =>
|
||||
setCollapsed((c) => {
|
||||
localStorage.setItem('woc_sb_collapsed', c ? '0' : '1');
|
||||
return !c;
|
||||
});
|
||||
|
||||
useEffect(() => {
|
||||
const onKey = (e: KeyboardEvent) => {
|
||||
if ((e.metaKey || e.ctrlKey) && e.key.toLowerCase() === 'b') {
|
||||
e.preventDefault();
|
||||
toggleCollapsed();
|
||||
}
|
||||
};
|
||||
window.addEventListener('keydown', onKey);
|
||||
return () => window.removeEventListener('keydown', onKey);
|
||||
}, []);
|
||||
|
||||
const openMenu = () => setDrawer(true);
|
||||
const openChangePassword = () => setShowPw(true);
|
||||
|
||||
return (
|
||||
<InstancesCtx.Provider value={state}>
|
||||
<div className={'shell' + (railed ? ' collapsed' : '') + (drawer ? ' drawer-open' : '')}>
|
||||
<Sidebar collapsed={railed} onToggleCollapsed={toggleCollapsed} />
|
||||
<div className="shell-backdrop" onClick={() => setDrawer(false)} />
|
||||
<main className="workspace">
|
||||
<Routes>
|
||||
<Route path="/" element={<HomeView onOpenMenu={openMenu} onChangePassword={openChangePassword} />} />
|
||||
<Route path="/admin" element={<Admin onOpenMenu={openMenu} onChangePassword={openChangePassword} />} />
|
||||
<Route path="/i/:id" element={<InstanceView onOpenMenu={openMenu} />} />
|
||||
<Route path="*" element={<Navigate to="/" replace />} />
|
||||
</Routes>
|
||||
</main>
|
||||
</div>
|
||||
{showPw && <ChangePassword onClose={() => setShowPw(false)} onSaved={() => refresh()} />}
|
||||
</InstancesCtx.Provider>
|
||||
);
|
||||
}
|
||||
|
||||
function Sidebar({ collapsed, onToggleCollapsed }: { collapsed: boolean; onToggleCollapsed: () => void }) {
|
||||
const { user, logout } = useAuth();
|
||||
const { confirm } = useUI();
|
||||
const { instances } = useInstances();
|
||||
const nav = useNavigate();
|
||||
const loc = useLocation();
|
||||
const isAdmin = user?.role === 'admin';
|
||||
const go = (p: string) => nav(p);
|
||||
|
||||
return (
|
||||
<aside className="sidebar">
|
||||
<div className="sb-top">
|
||||
<div className="sb-brand">
|
||||
<img src="/favicon.svg" className="sb-logo" alt="" />
|
||||
{!collapsed && <span className="sb-name">云微</span>}
|
||||
</div>
|
||||
<button className="sb-collapse" title="折叠侧栏 (⌘B)" onClick={onToggleCollapsed}>
|
||||
{Icon.collapse}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<nav className="sb-nav">
|
||||
<button className={'sb-item' + (loc.pathname === '/' ? ' on' : '')} onClick={() => go('/')} title="主页">
|
||||
<span className="sb-ic">{Icon.home}</span>
|
||||
{!collapsed && <span className="sb-label">主页</span>}
|
||||
</button>
|
||||
</nav>
|
||||
|
||||
{!collapsed && <div className="sb-section">微信实例</div>}
|
||||
<div className="sb-list">
|
||||
{instances.length === 0 && !collapsed && <div className="sb-empty">暂无可用实例</div>}
|
||||
{instances.map((inst) => {
|
||||
const on = loc.pathname === `/i/${inst.id}`;
|
||||
const st = statusOf(inst);
|
||||
return (
|
||||
<button key={inst.id} className={'sb-item sb-inst' + (on ? ' on' : '')} onClick={() => go(`/i/${inst.id}`)} title={inst.name}>
|
||||
<span className="sb-avatar">
|
||||
{inst.name.slice(0, 1)}
|
||||
<span className={'sb-dot ' + st.cls} />
|
||||
</span>
|
||||
{!collapsed && <span className="sb-label">{inst.name}</span>}
|
||||
{!collapsed && <span className="sb-stxt">{st.text}</span>}
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
|
||||
<div className="sb-footer">
|
||||
<button className={'sb-item' + (loc.pathname === '/admin' ? ' on' : '')} onClick={() => go('/admin')} title={isAdmin ? '管理' : '设置'}>
|
||||
<span className="sb-ic">{Icon.gear}</span>
|
||||
{!collapsed && <span className="sb-label">{isAdmin ? '管理' : '设置'}</span>}
|
||||
</button>
|
||||
<button
|
||||
className="sb-item"
|
||||
title="退出"
|
||||
onClick={async () => {
|
||||
if (await confirm({ title: '退出登录?', confirmText: '退出' })) logout();
|
||||
}}
|
||||
>
|
||||
<span className="sb-ic">{Icon.logout}</span>
|
||||
{!collapsed && <span className="sb-label">退出</span>}
|
||||
</button>
|
||||
{!collapsed && (
|
||||
<div className="sb-user">
|
||||
{user?.username}
|
||||
{isAdmin && ' · 管理员'}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</aside>
|
||||
);
|
||||
}
|
||||
|
||||
function HomeView({ onOpenMenu, onChangePassword }: { onOpenMenu: () => void; onChangePassword: () => void }) {
|
||||
const { user } = useAuth();
|
||||
const { instances, loaded } = useInstances();
|
||||
const nav = useNavigate();
|
||||
const isAdmin = user?.role === 'admin';
|
||||
|
||||
return (
|
||||
<div className="ws-page">
|
||||
<header className="ws-head">
|
||||
<button className="ws-menu" onClick={onOpenMenu} aria-label="菜单">
|
||||
{Icon.menu}
|
||||
</button>
|
||||
<span className="ws-title">主页</span>
|
||||
</header>
|
||||
|
||||
<div className="content">
|
||||
<div className="hello">
|
||||
你好,<b>{user?.username}</b>
|
||||
{isAdmin && <span className="tag">管理员</span>}
|
||||
</div>
|
||||
|
||||
{user?.mustChangePassword && (
|
||||
<button className="warn-banner" onClick={onChangePassword}>
|
||||
<span className="warn-icon">!</span>
|
||||
<span className="warn-text">
|
||||
<b>你还在使用默认密码</b>
|
||||
<span>该系统登录着你的微信,请立即修改密码 ›</span>
|
||||
</span>
|
||||
</button>
|
||||
)}
|
||||
|
||||
<div className="section-row">
|
||||
<span className="section-title">我的微信实例</span>
|
||||
{isAdmin && (
|
||||
<button className="btn-text" onClick={() => nav('/admin')}>
|
||||
管理 ›
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{loaded && instances.length === 0 ? (
|
||||
<div className="empty-state">
|
||||
<div className="empty-blob">
|
||||
<img src="/favicon.svg" alt="" />
|
||||
</div>
|
||||
<div className="empty-title">还没有微信实例</div>
|
||||
<div className="empty-sub">{isAdmin ? '去「管理」新建一个微信实例' : '请联系管理员为你分配实例'}</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="inst-grid">
|
||||
{instances.map((inst) => {
|
||||
const st = statusOf(inst);
|
||||
return (
|
||||
<button key={inst.id} className="home-card" onClick={() => nav(`/i/${inst.id}`)}>
|
||||
<span className="home-card-av">{inst.name.slice(0, 1)}</span>
|
||||
<span className="home-card-main">
|
||||
<span className="home-card-name">{inst.name}</span>
|
||||
<span className={'home-card-st ' + st.cls}>● {st.text}</span>
|
||||
</span>
|
||||
<span className="enter-arrow">›</span>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
)}
|
||||
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export function ChangePassword({ onClose, onSaved }: { onClose: () => void; onSaved?: () => void }) {
|
||||
const [oldPassword, setOld] = useState('');
|
||||
const [newPassword, setNew] = useState('');
|
||||
const [confirm, setConfirm] = useState('');
|
||||
const [msg, setMsg] = useState('');
|
||||
const [busy, setBusy] = useState(false);
|
||||
const mismatch = confirm.length > 0 && newPassword !== confirm;
|
||||
const canSubmit = !busy && !!oldPassword && newPassword.length >= 6 && newPassword === confirm;
|
||||
|
||||
const submit = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
setMsg('');
|
||||
if (newPassword !== confirm) {
|
||||
setMsg('两次输入的新密码不一致');
|
||||
return;
|
||||
}
|
||||
setBusy(true);
|
||||
try {
|
||||
await api.changePassword(oldPassword, newPassword);
|
||||
setMsg('修改成功');
|
||||
onSaved?.();
|
||||
setTimeout(onClose, 800);
|
||||
} catch (e: any) {
|
||||
setMsg(e.message || '修改失败');
|
||||
} finally {
|
||||
setBusy(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="modal-mask" onClick={onClose}>
|
||||
<form className="card modal" onClick={(e) => e.stopPropagation()} onSubmit={submit}>
|
||||
<h2>修改密码</h2>
|
||||
<PasswordInput placeholder="原密码" autoComplete="current-password" value={oldPassword} onChange={setOld} />
|
||||
<PasswordInput placeholder="新密码(至少 6 位)" autoComplete="new-password" value={newPassword} onChange={setNew} />
|
||||
<PasswordInput placeholder="再次输入新密码" autoComplete="new-password" value={confirm} onChange={setConfirm} />
|
||||
{mismatch && <div className="error">两次输入的新密码不一致</div>}
|
||||
{msg && <div className={msg === '修改成功' ? 'ok' : 'error'}>{msg}</div>}
|
||||
<div className="modal-actions">
|
||||
<button type="button" className="btn" onClick={onClose}>
|
||||
取消
|
||||
</button>
|
||||
<button className="btn btn-primary" disabled={!canSubmit}>
|
||||
确定
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -90,6 +90,10 @@ export const api = {
|
||||
instanceWechatInstall: (id: string) => req(`/api/admin/instances/${id}/wechat/install`, { method: 'POST' }),
|
||||
instanceWechatUpdate: (id: string) => req(`/api/admin/instances/${id}/wechat/update`, { method: 'POST' }),
|
||||
instanceStart: (id: string) => req(`/api/admin/instances/${id}/start`, { method: 'POST' }),
|
||||
instanceStop: (id: string) => req(`/api/admin/instances/${id}/stop`, { method: 'POST' }),
|
||||
instanceRestart: (id: string) => req(`/api/admin/instances/${id}/restart`, { method: 'POST' }),
|
||||
instanceUpgrade: (id: string) => req(`/api/admin/instances/${id}/upgrade`, { method: 'POST' }),
|
||||
instanceLogsUrl: (id: string) => `/api/admin/instances/${id}/logs`,
|
||||
|
||||
// 文件中转
|
||||
listFiles: (id: string) => req<{ files: { name: string; size: number }[] }>(`/api/instances/${id}/files`),
|
||||
@@ -104,4 +108,10 @@ export const api = {
|
||||
return res.json();
|
||||
},
|
||||
downloadFileUrl: (id: string, name: string) => `/api/instances/${id}/download?name=${encodeURIComponent(name)}`,
|
||||
deleteFile: (id: string, name: string) => req(`/api/instances/${id}/files?name=${encodeURIComponent(name)}`, { method: 'DELETE' }),
|
||||
|
||||
// 多端协作:操作控制权
|
||||
controlStatus: (id: string) => req<{ free: boolean; mine: boolean; holder: string | null }>(`/api/instances/${id}/control`),
|
||||
controlBeat: (id: string) => req<{ mine: boolean; holder: string }>(`/api/instances/${id}/control/beat`, { method: 'POST' }),
|
||||
controlTake: (id: string) => req<{ mine: boolean; holder: string }>(`/api/instances/${id}/control/take`, { method: 'POST' }),
|
||||
};
|
||||
|
||||
+205
-121
@@ -2,11 +2,20 @@ import { useEffect, useRef, useState } from 'react';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
import { api, type PanelUser, type InstanceWithStatus } from '../api';
|
||||
import { useUI, PasswordInput } from '../ui';
|
||||
import { useAuth } from '../auth';
|
||||
|
||||
const BUSY_PHASES = ['downloading', 'extracting', 'installing'];
|
||||
|
||||
export default function Admin() {
|
||||
const MenuIcon = (
|
||||
<svg viewBox="0 0 24 24" width="22" height="22" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round">
|
||||
<path d="M4 6h16M4 12h16M4 18h16" />
|
||||
</svg>
|
||||
);
|
||||
|
||||
export default function Admin({ onOpenMenu, onChangePassword }: { onOpenMenu: () => void; onChangePassword: () => void }) {
|
||||
const nav = useNavigate();
|
||||
const { user } = useAuth();
|
||||
const isAdmin = user?.role === 'admin';
|
||||
const { toast, confirm } = useUI();
|
||||
const [users, setUsers] = useState<PanelUser[]>([]);
|
||||
const [instances, setInstances] = useState<InstanceWithStatus[]>([]);
|
||||
@@ -18,12 +27,20 @@ export default function Admin() {
|
||||
const [resetTarget, setResetTarget] = useState<PanelUser | null>(null); // 重置密码弹窗
|
||||
const [deleteInst, setDeleteInst] = useState<InstanceWithStatus | null>(null); // 删除实例弹窗
|
||||
const [renameInst, setRenameInst] = useState<InstanceWithStatus | null>(null); // 重命名实例弹窗
|
||||
const [starting, setStarting] = useState<Set<string>>(new Set());
|
||||
const [acting, setActing] = useState<Record<string, string>>({}); // 实例 id → 进行中的动作文案(启动中/升级中…)
|
||||
const setAct = (id: string, label: string | null) =>
|
||||
setActing((a) => {
|
||||
const n = { ...a };
|
||||
if (label) n[id] = label;
|
||||
else delete n[id];
|
||||
return n;
|
||||
});
|
||||
|
||||
const subs = users.filter((u) => u.role !== 'admin');
|
||||
const timer = useRef<number | undefined>(undefined);
|
||||
|
||||
const load = async () => {
|
||||
if (!isAdmin) return; // 子账号无管理数据权限,管理页只给改密
|
||||
try {
|
||||
const [{ users }, { instances }] = await Promise.all([api.listUsers(), api.listInstances()]);
|
||||
setUsers(users);
|
||||
@@ -62,7 +79,7 @@ export default function Admin() {
|
||||
};
|
||||
|
||||
const start = async (inst: InstanceWithStatus) => {
|
||||
setStarting((s) => new Set(s).add(inst.id));
|
||||
setAct(inst.id, '启动中…');
|
||||
try {
|
||||
await api.instanceStart(inst.id);
|
||||
toast('实例已启动', 'ok');
|
||||
@@ -70,11 +87,22 @@ export default function Admin() {
|
||||
} catch (e: any) {
|
||||
toast(e.message || '启动失败', 'error');
|
||||
} finally {
|
||||
setStarting((s) => {
|
||||
const n = new Set(s);
|
||||
n.delete(inst.id);
|
||||
return n;
|
||||
});
|
||||
setAct(inst.id, null);
|
||||
}
|
||||
};
|
||||
|
||||
const lifecycle = async (inst: InstanceWithStatus, kind: 'stop' | 'restart' | 'upgrade') => {
|
||||
const label = kind === 'stop' ? '停止中…' : kind === 'upgrade' ? '升级中…' : '重启中…';
|
||||
setAct(inst.id, label);
|
||||
if (kind === 'upgrade') toast('正在升级实例:拉取最新镜像并重建,可能需要几分钟,请勿离开…', 'info');
|
||||
try {
|
||||
await (kind === 'stop' ? api.instanceStop(inst.id) : kind === 'upgrade' ? api.instanceUpgrade(inst.id) : api.instanceRestart(inst.id));
|
||||
toast(kind === 'stop' ? '已停止' : kind === 'upgrade' ? '已升级到最新镜像并重启' : '已重启', 'ok');
|
||||
await load();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '操作失败', 'error');
|
||||
} finally {
|
||||
setAct(inst.id, null);
|
||||
}
|
||||
};
|
||||
|
||||
@@ -103,93 +131,117 @@ export default function Admin() {
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="page">
|
||||
<header className="topbar">
|
||||
<button className="btn-text" onClick={() => nav('/')}>
|
||||
‹ 返回
|
||||
<div className="ws-page">
|
||||
<header className="ws-head">
|
||||
<button className="ws-menu" onClick={onOpenMenu} aria-label="菜单">
|
||||
{MenuIcon}
|
||||
</button>
|
||||
<span className="topbar-title">管理</span>
|
||||
<span style={{ width: 48 }} />
|
||||
<span className="ws-title">{isAdmin ? '管理' : '设置'}</span>
|
||||
</header>
|
||||
|
||||
<main className="content">
|
||||
{err && <div className="error">{err}</div>}
|
||||
|
||||
<div className="section-row">
|
||||
<span className="section-title">微信实例</span>
|
||||
<button className="btn-text" onClick={() => setCreatingInst(true)}>
|
||||
+ 新建实例
|
||||
</button>
|
||||
</div>
|
||||
{instances.length === 0 ? (
|
||||
<div className="list">
|
||||
<div className="muted small" style={{ padding: '14px 16px' }}>暂无实例</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="inst-grid">
|
||||
{instances.map((inst) => (
|
||||
<InstanceAdminCard
|
||||
key={inst.id}
|
||||
inst={inst}
|
||||
userCount={usersForInstance(inst.id).length}
|
||||
starting={starting.has(inst.id)}
|
||||
onTrigger={trigger}
|
||||
onStart={() => start(inst)}
|
||||
onRename={() => setRenameInst(inst)}
|
||||
onAssign={() => setAssignInst(inst)}
|
||||
onDelete={() => setDeleteInst(inst)}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
{isAdmin && (
|
||||
<>
|
||||
<div className="section-row">
|
||||
<span className="section-title">微信实例</span>
|
||||
<button className="btn-text" onClick={() => setCreatingInst(true)}>
|
||||
+ 新建实例
|
||||
</button>
|
||||
</div>
|
||||
{instances.length === 0 ? (
|
||||
<div className="list">
|
||||
<div className="muted small" style={{ padding: '14px 16px' }}>暂无实例</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="inst-grid">
|
||||
{instances.map((inst) => (
|
||||
<InstanceAdminCard
|
||||
key={inst.id}
|
||||
inst={inst}
|
||||
userCount={usersForInstance(inst.id).length}
|
||||
acting={acting[inst.id]}
|
||||
onEnter={() => nav(`/i/${inst.id}`)}
|
||||
onTrigger={trigger}
|
||||
onStart={() => start(inst)}
|
||||
onStop={() => lifecycle(inst, 'stop')}
|
||||
onRestart={() => lifecycle(inst, 'restart')}
|
||||
onUpgrade={() => lifecycle(inst, 'upgrade')}
|
||||
onRename={() => setRenameInst(inst)}
|
||||
onAssign={() => setAssignInst(inst)}
|
||||
onDelete={() => setDeleteInst(inst)}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="section-row" style={{ marginTop: 22 }}>
|
||||
<span className="section-title">子账号</span>
|
||||
<button className="btn-text" onClick={() => setCreatingUser(true)}>
|
||||
+ 新建子账号
|
||||
</button>
|
||||
</div>
|
||||
{subs.length === 0 ? (
|
||||
<div className="list">
|
||||
<div className="muted small" style={{ padding: '14px 16px' }}>暂无子账号</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="inst-grid">
|
||||
{subs.map((u) => (
|
||||
<div key={u.id} className="inst-card">
|
||||
<div className="inst-head">
|
||||
<span className="inst-name">{u.username}</span>
|
||||
{u.disabled ? <span className="tag tag-off">已禁用</span> : <span className="tag tag-on">正常</span>}
|
||||
</div>
|
||||
<div className="inst-sub">{u.allowedInstances.length > 0 ? `可访问 ${u.allowedInstances.length} 个实例` : '未分配实例'}</div>
|
||||
{u.allowedInstances.length > 0 && (
|
||||
<div className="chip-row" style={{ marginTop: 8 }}>
|
||||
{u.allowedInstances.map((id) => (
|
||||
<span key={id} className="chip chip-static">
|
||||
{instName(id)}
|
||||
</span>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
<div className="inst-admin-links">
|
||||
<button className="btn-text" onClick={() => setAssignUser(u)}>
|
||||
可访问实例
|
||||
</button>
|
||||
<button className="btn-text" onClick={() => toggle(u)}>
|
||||
{u.disabled ? '启用' : '禁用'}
|
||||
</button>
|
||||
<button className="btn-text" onClick={() => setResetTarget(u)}>
|
||||
重置密码
|
||||
</button>
|
||||
<button className="btn-text danger" onClick={() => removeUser(u)}>
|
||||
删除
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
</>
|
||||
)}
|
||||
|
||||
<div className="section-row" style={{ marginTop: 22 }}>
|
||||
<span className="section-title">子账号</span>
|
||||
<button className="btn-text" onClick={() => setCreatingUser(true)}>
|
||||
+ 新建子账号
|
||||
</button>
|
||||
{/* 账号:所有人(含子账号)都能在此改密 */}
|
||||
<div className="section-row" style={{ marginTop: isAdmin ? 22 : 0 }}>
|
||||
<span className="section-title">账号</span>
|
||||
</div>
|
||||
<div className="list">
|
||||
{users.map((u) => (
|
||||
<div key={u.id} className="user-row">
|
||||
<div className="user-main">
|
||||
<span className="user-name">
|
||||
{u.username}
|
||||
{u.role === 'admin' && <span className="tag">管理员</span>}
|
||||
{u.disabled && <span className="tag tag-off">已禁用</span>}
|
||||
</span>
|
||||
{u.role === 'admin' ? (
|
||||
<span className="muted small">可访问全部实例</span>
|
||||
) : u.allowedInstances.length > 0 ? (
|
||||
<span className="chip-row">
|
||||
{u.allowedInstances.map((id) => (
|
||||
<span key={id} className="chip chip-static">
|
||||
{instName(id)}
|
||||
</span>
|
||||
))}
|
||||
</span>
|
||||
) : (
|
||||
<span className="muted small">未分配实例</span>
|
||||
)}
|
||||
</div>
|
||||
{u.role !== 'admin' && (
|
||||
<div className="user-actions">
|
||||
<button className="btn-text" onClick={() => setAssignUser(u)}>
|
||||
可访问实例
|
||||
</button>
|
||||
<button className="btn-text" onClick={() => toggle(u)}>
|
||||
{u.disabled ? '启用' : '禁用'}
|
||||
</button>
|
||||
<button className="btn-text" onClick={() => setResetTarget(u)}>
|
||||
重置密码
|
||||
</button>
|
||||
<button className="btn-text danger" onClick={() => removeUser(u)}>
|
||||
删除
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
<div className="inst-grid">
|
||||
<div className="inst-card">
|
||||
<div className="inst-head">
|
||||
<span className="inst-name">{user?.username}</span>
|
||||
{isAdmin ? <span className="tag">管理员</span> : <span className="tag tag-on">子账号</span>}
|
||||
</div>
|
||||
))}
|
||||
<div className="inst-sub">{isAdmin ? '可访问全部实例' : `可访问 ${user?.allowedInstances.length ?? 0} 个实例`}</div>
|
||||
<div className="inst-actions">
|
||||
<button className="btn btn-primary inst-act-wide" onClick={onChangePassword}>
|
||||
修改密码
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
|
||||
@@ -397,18 +449,26 @@ function DeleteInstance({ inst, onClose, onDone }: { inst: InstanceWithStatus; o
|
||||
function InstanceAdminCard({
|
||||
inst,
|
||||
userCount,
|
||||
starting,
|
||||
acting,
|
||||
onEnter,
|
||||
onTrigger,
|
||||
onStart,
|
||||
onStop,
|
||||
onRestart,
|
||||
onUpgrade,
|
||||
onRename,
|
||||
onAssign,
|
||||
onDelete,
|
||||
}: {
|
||||
inst: InstanceWithStatus;
|
||||
userCount: number;
|
||||
starting?: boolean;
|
||||
acting?: string;
|
||||
onEnter: () => void;
|
||||
onTrigger: (inst: InstanceWithStatus, kind: 'install' | 'update') => void;
|
||||
onStart: () => void;
|
||||
onStop: () => void;
|
||||
onRestart: () => void;
|
||||
onUpgrade: () => void;
|
||||
onRename: () => void;
|
||||
onAssign: () => void;
|
||||
onDelete: () => void;
|
||||
@@ -417,15 +477,18 @@ function InstanceAdminCard({
|
||||
const busy = BUSY_PHASES.includes(wx.phase);
|
||||
const installed = wx.installed && wx.phase !== 'downloading';
|
||||
const offline = inst.runtime !== 'running';
|
||||
const working = !!acting || busy; // 生命周期操作中 或 微信下载/更新中 → 锁住卡片
|
||||
|
||||
let badge: { text: string; cls: string };
|
||||
if (offline) badge = { text: inst.runtime === 'missing' ? '未创建' : '已停止', cls: 'tag-off' };
|
||||
if (acting) badge = { text: '处理中', cls: 'tag-busy' };
|
||||
else if (offline) badge = { text: inst.runtime === 'missing' ? '未创建' : '已停止', cls: 'tag-off' };
|
||||
else if (busy) badge = { text: '处理中', cls: 'tag-busy' };
|
||||
else if (installed) badge = { text: '在线', cls: 'tag-on' };
|
||||
else badge = { text: '待安装', cls: 'tag-warn' };
|
||||
|
||||
let sub: string;
|
||||
if (busy) sub = wx.percent >= 0 ? `${wx.message || '处理中'} ${wx.percent}%` : wx.message || '请稍候…';
|
||||
if (acting) sub = acting;
|
||||
else if (busy) sub = wx.percent >= 0 ? `${wx.message || '处理中'} ${wx.percent}%` : wx.message || '请稍候…';
|
||||
else if (wx.phase === 'error') sub = wx.message || '操作失败,可重试';
|
||||
else if (offline) sub = inst.runtime === 'missing' ? '容器尚未创建' : '容器已停止';
|
||||
else if (installed) sub = wx.version ? `微信 ${wx.version}` : '微信已安装';
|
||||
@@ -438,47 +501,68 @@ function InstanceAdminCard({
|
||||
<span className={'tag ' + badge.cls}>{badge.text}</span>
|
||||
</div>
|
||||
<div className="inst-sub">
|
||||
{sub} · 可访问 {userCount} 人
|
||||
{sub}
|
||||
{!acting && ` · 可访问 ${userCount} 人`}
|
||||
</div>
|
||||
|
||||
{busy && (
|
||||
{working && (
|
||||
<div className="wx-progress">
|
||||
<div
|
||||
className={'wx-progress-bar' + (wx.percent < 0 ? ' indeterminate' : '')}
|
||||
style={wx.percent >= 0 ? { width: `${wx.percent}%` } : undefined}
|
||||
className={'wx-progress-bar' + (acting || wx.percent < 0 ? ' indeterminate' : '')}
|
||||
style={!acting && wx.percent >= 0 ? { width: `${wx.percent}%` } : undefined}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{!busy && (
|
||||
<div className="inst-actions">
|
||||
{offline ? (
|
||||
<button className="btn btn-primary inst-act-wide" disabled={starting} onClick={onStart}>
|
||||
{starting ? '启动中…' : inst.runtime === 'missing' ? '创建并启动' : '启动实例'}
|
||||
</button>
|
||||
) : installed ? (
|
||||
<button className="btn btn-primary inst-act-wide" onClick={() => onTrigger(inst, 'update')}>
|
||||
更新微信
|
||||
</button>
|
||||
) : (
|
||||
<button className="btn btn-primary inst-act-wide" onClick={() => onTrigger(inst, 'install')}>
|
||||
下载安装微信
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
{/* 进行中(升级/重启/停止/下载)时隐藏所有操作,避免重复点击 */}
|
||||
{!working && (
|
||||
<>
|
||||
<div className="inst-actions">
|
||||
{offline ? (
|
||||
<button className="btn btn-primary inst-act-wide" onClick={onStart}>
|
||||
{inst.runtime === 'missing' ? '创建并启动' : '启动实例'}
|
||||
</button>
|
||||
) : (
|
||||
<button className="btn btn-primary inst-act-wide" disabled={!installed} onClick={onEnter} title={installed ? '' : '需先下载安装微信'}>
|
||||
进入实例
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="inst-admin-links">
|
||||
<button className="btn-text" onClick={onRename}>
|
||||
重命名
|
||||
</button>
|
||||
<button className="btn-text" onClick={onAssign}>
|
||||
分配账户
|
||||
</button>
|
||||
<button className="btn-text danger" onClick={onDelete}>
|
||||
删除
|
||||
</button>
|
||||
</div>
|
||||
<div className="inst-admin-links">
|
||||
{!offline && (
|
||||
<button className="btn-text" onClick={() => onTrigger(inst, installed ? 'update' : 'install')}>
|
||||
{installed ? '更新微信' : '下载安装'}
|
||||
</button>
|
||||
)}
|
||||
<button className="btn-text" onClick={onUpgrade} title="拉取最新镜像并重建(保留聊天记录),把实例更新到新版">
|
||||
升级实例
|
||||
</button>
|
||||
{!offline && (
|
||||
<button className="btn-text" onClick={onRestart}>
|
||||
重启
|
||||
</button>
|
||||
)}
|
||||
{!offline && (
|
||||
<button className="btn-text" onClick={onStop}>
|
||||
停止
|
||||
</button>
|
||||
)}
|
||||
<button className="btn-text" onClick={onRename}>
|
||||
重命名
|
||||
</button>
|
||||
<button className="btn-text" onClick={onAssign}>
|
||||
分配账户
|
||||
</button>
|
||||
<button className="btn-text" onClick={() => window.open(api.instanceLogsUrl(inst.id), '_blank')} title="查看实例容器日志(排错)">
|
||||
日志
|
||||
</button>
|
||||
<button className="btn-text danger" onClick={onDelete}>
|
||||
删除
|
||||
</button>
|
||||
</div>
|
||||
</>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -1,256 +1,2 @@
|
||||
import { useEffect, useRef, useState } from 'react';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
import { useAuth } from '../auth';
|
||||
import { useUI, PasswordInput } from '../ui';
|
||||
import { api, type InstanceWithStatus } from '../api';
|
||||
|
||||
const BUSY_PHASES = ['downloading', 'extracting', 'installing'];
|
||||
|
||||
export default function Dashboard() {
|
||||
const { user, logout, refresh } = useAuth();
|
||||
const { toast, confirm } = useUI();
|
||||
const nav = useNavigate();
|
||||
const [showPw, setShowPw] = useState(false);
|
||||
const [instances, setInstances] = useState<InstanceWithStatus[] | null>(null);
|
||||
const [err, setErr] = useState('');
|
||||
const [starting, setStarting] = useState<Set<string>>(new Set());
|
||||
const timer = useRef<number | undefined>(undefined);
|
||||
const isAdmin = user?.role === 'admin';
|
||||
|
||||
const load = async () => {
|
||||
try {
|
||||
const { instances } = await api.listInstances();
|
||||
setInstances(instances);
|
||||
} catch (e: any) {
|
||||
setErr(e.message || '加载失败');
|
||||
}
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
load();
|
||||
return () => window.clearTimeout(timer.current);
|
||||
}, []);
|
||||
|
||||
// 任一实例安装/更新进行中时轮询
|
||||
useEffect(() => {
|
||||
window.clearTimeout(timer.current);
|
||||
const busy = instances?.some((i) => BUSY_PHASES.includes(i.wechat.phase));
|
||||
if (busy) timer.current = window.setTimeout(load, 1500);
|
||||
return () => window.clearTimeout(timer.current);
|
||||
}, [instances]);
|
||||
|
||||
const start = async (inst: InstanceWithStatus) => {
|
||||
setErr('');
|
||||
setStarting((s) => new Set(s).add(inst.id));
|
||||
try {
|
||||
await api.instanceStart(inst.id);
|
||||
toast('实例已启动', 'ok');
|
||||
await load();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '启动失败', 'error');
|
||||
} finally {
|
||||
setStarting((s) => {
|
||||
const n = new Set(s);
|
||||
n.delete(inst.id);
|
||||
return n;
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="page">
|
||||
<header className="topbar">
|
||||
<span className="topbar-title">云微</span>
|
||||
<button
|
||||
className="btn-text"
|
||||
onClick={async () => {
|
||||
if (await confirm({ title: '退出登录?', confirmText: '退出' })) logout();
|
||||
}}
|
||||
>
|
||||
退出
|
||||
</button>
|
||||
</header>
|
||||
|
||||
<main className="content">
|
||||
<div className="hello">
|
||||
你好,<b>{user?.username}</b>
|
||||
{isAdmin && <span className="tag">管理员</span>}
|
||||
</div>
|
||||
|
||||
{user?.mustChangePassword && (
|
||||
<button className="warn-banner" onClick={() => setShowPw(true)}>
|
||||
<span className="warn-icon">!</span>
|
||||
<span className="warn-text">
|
||||
<b>你还在使用默认密码</b>
|
||||
<span>该系统登录着你的微信,请立即修改密码 ›</span>
|
||||
</span>
|
||||
</button>
|
||||
)}
|
||||
|
||||
{err && <div className="error">{err}</div>}
|
||||
|
||||
<div className="section-row">
|
||||
<span className="section-title">微信实例</span>
|
||||
{isAdmin && (
|
||||
<button className="btn-text" onClick={() => nav('/admin')}>
|
||||
管理 ›
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{instances && instances.length === 0 && (
|
||||
<div className="empty-state">
|
||||
<div className="empty-blob"><img src="/favicon.svg" alt="" /></div>
|
||||
<div className="empty-title">还没有微信实例</div>
|
||||
<div className="empty-sub">{isAdmin ? '去「管理」新建一个微信实例' : '请联系管理员为你分配实例'}</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="inst-grid">
|
||||
{instances?.map((inst) => (
|
||||
<InstanceCard
|
||||
key={inst.id}
|
||||
inst={inst}
|
||||
isAdmin={isAdmin}
|
||||
starting={starting.has(inst.id)}
|
||||
onEnter={() => nav(`/desktop/${inst.id}`)}
|
||||
onStart={() => start(inst)}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
|
||||
<div className="list">
|
||||
<button className="list-item" onClick={() => setShowPw(true)}>
|
||||
<span>修改密码</span>
|
||||
<span className="enter-arrow">›</span>
|
||||
</button>
|
||||
{isAdmin && (
|
||||
<button className="list-item" onClick={() => nav('/admin')}>
|
||||
<span>实例与子账号管理</span>
|
||||
<span className="enter-arrow">›</span>
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</main>
|
||||
|
||||
{showPw && <ChangePassword onClose={() => setShowPw(false)} onSaved={() => refresh()} />}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function InstanceCard({
|
||||
inst,
|
||||
isAdmin,
|
||||
starting,
|
||||
onEnter,
|
||||
onStart,
|
||||
}: {
|
||||
inst: InstanceWithStatus;
|
||||
isAdmin?: boolean;
|
||||
starting?: boolean;
|
||||
onEnter: () => void;
|
||||
onStart: () => void;
|
||||
}) {
|
||||
const wx = inst.wechat;
|
||||
const busy = BUSY_PHASES.includes(wx.phase);
|
||||
const installed = wx.installed && wx.phase !== 'downloading';
|
||||
const offline = inst.runtime !== 'running';
|
||||
|
||||
let badge: { text: string; cls: string };
|
||||
if (offline) badge = { text: inst.runtime === 'missing' ? '未创建' : '已停止', cls: 'tag-off' };
|
||||
else if (busy) badge = { text: '处理中', cls: 'tag-busy' };
|
||||
else if (installed) badge = { text: '在线', cls: 'tag-on' };
|
||||
else badge = { text: '待安装', cls: 'tag-warn' };
|
||||
|
||||
let sub: string;
|
||||
if (offline) sub = inst.runtime === 'missing' ? '容器尚未创建' : '容器已停止,需先启动';
|
||||
else if (busy) sub = wx.percent >= 0 ? `${wx.message || '处理中'} ${wx.percent}%` : wx.message || '请稍候…';
|
||||
else if (wx.phase === 'error') sub = wx.message || '操作失败,可重试';
|
||||
else if (installed) sub = wx.version ? `微信 ${wx.version}` : '微信已安装';
|
||||
else sub = '微信尚未安装';
|
||||
|
||||
const canEnter = !offline && installed && !busy;
|
||||
|
||||
return (
|
||||
<div className="inst-card">
|
||||
<div className="inst-head">
|
||||
<span className="inst-name">{inst.name}</span>
|
||||
<span className={'tag ' + badge.cls}>{badge.text}</span>
|
||||
</div>
|
||||
<div className="inst-sub">{sub}</div>
|
||||
|
||||
{busy && (
|
||||
<div className="wx-progress">
|
||||
<div
|
||||
className={'wx-progress-bar' + (wx.percent < 0 ? ' indeterminate' : '')}
|
||||
style={wx.percent >= 0 ? { width: `${wx.percent}%` } : undefined}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="inst-actions">
|
||||
{offline && isAdmin ? (
|
||||
<button className="btn btn-primary inst-enter" disabled={starting} onClick={onStart}>
|
||||
{starting ? '启动中…' : inst.runtime === 'missing' ? '创建并启动' : '启动实例'}
|
||||
</button>
|
||||
) : (
|
||||
<button className="btn btn-primary inst-enter" disabled={!canEnter} onClick={onEnter}>
|
||||
进入微信
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function ChangePassword({ onClose, onSaved }: { onClose: () => void; onSaved?: () => void }) {
|
||||
const [oldPassword, setOld] = useState('');
|
||||
const [newPassword, setNew] = useState('');
|
||||
const [confirm, setConfirm] = useState('');
|
||||
const [msg, setMsg] = useState('');
|
||||
const [busy, setBusy] = useState(false);
|
||||
|
||||
const mismatch = confirm.length > 0 && newPassword !== confirm;
|
||||
const canSubmit = !busy && !!oldPassword && newPassword.length >= 6 && newPassword === confirm;
|
||||
|
||||
const submit = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
setMsg('');
|
||||
if (newPassword !== confirm) {
|
||||
setMsg('两次输入的新密码不一致');
|
||||
return;
|
||||
}
|
||||
setBusy(true);
|
||||
try {
|
||||
await api.changePassword(oldPassword, newPassword);
|
||||
setMsg('修改成功');
|
||||
onSaved?.(); // 刷新当前用户,清除「默认密码」提示
|
||||
setTimeout(onClose, 800);
|
||||
} catch (e: any) {
|
||||
setMsg(e.message || '修改失败');
|
||||
} finally {
|
||||
setBusy(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="modal-mask" onClick={onClose}>
|
||||
<form className="card modal" onClick={(e) => e.stopPropagation()} onSubmit={submit}>
|
||||
<h2>修改密码</h2>
|
||||
<PasswordInput placeholder="原密码" autoComplete="current-password" value={oldPassword} onChange={setOld} />
|
||||
<PasswordInput placeholder="新密码(至少 6 位)" autoComplete="new-password" value={newPassword} onChange={setNew} />
|
||||
<PasswordInput placeholder="再次输入新密码" autoComplete="new-password" value={confirm} onChange={setConfirm} />
|
||||
{mismatch && <div className="error">两次输入的新密码不一致</div>}
|
||||
{msg && <div className={msg === '修改成功' ? 'ok' : 'error'}>{msg}</div>}
|
||||
<div className="modal-actions">
|
||||
<button type="button" className="btn" onClick={onClose}>
|
||||
取消
|
||||
</button>
|
||||
<button className="btn btn-primary" disabled={!canSubmit}>
|
||||
确定
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
// 已废弃:主页与改密已迁移到 ../AppShell.tsx(HomeView / ChangePassword)。
|
||||
export {};
|
||||
|
||||
+456
-84
@@ -2,10 +2,10 @@ import { useEffect, useRef, useState } from 'react';
|
||||
import { useNavigate, useParams } from 'react-router-dom';
|
||||
import { api } from '../api';
|
||||
import { useUI } from '../ui';
|
||||
import { useAuth } from '../auth';
|
||||
import { useInstances } from '../AppShell';
|
||||
|
||||
// 直接加载 KasmVNC 的 noVNC 页面(由 kclient 静态托管)。
|
||||
// 反代按实例隔离:所有桌面流量走 /desktop/<id>/*,网关据 <id> 选目标容器并注入该实例凭据。
|
||||
// path=desktop/<id>/websockify:让 noVNC 把 ws 连到该实例路径,网关剥前缀反代回 KasmVNC 根 /websockify。
|
||||
// KasmVNC noVNC 页面;反代按实例隔离:/desktop/<id>/* → 对应容器,注入凭据。
|
||||
function desktopUrl(id: string) {
|
||||
return (
|
||||
`/desktop/${id}/vnc/index.html?autoconnect=1&path=desktop/${id}/websockify&resize=remote` +
|
||||
@@ -24,20 +24,63 @@ function humanSize(n: number) {
|
||||
return (n / 1024 / 1024 / 1024).toFixed(2) + ' GB';
|
||||
}
|
||||
|
||||
export default function Desktop() {
|
||||
const nav = useNavigate();
|
||||
const { toast } = useUI();
|
||||
const MenuIcon = (
|
||||
<svg viewBox="0 0 24 24" width="22" height="22" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round">
|
||||
<path d="M4 6h16M4 12h16M4 18h16" />
|
||||
</svg>
|
||||
);
|
||||
|
||||
export default function InstanceView({ onOpenMenu }: { onOpenMenu: () => void }) {
|
||||
const { id } = useParams<{ id: string }>();
|
||||
const [loaded, setLoaded] = useState(false);
|
||||
const nav = useNavigate();
|
||||
const { user } = useAuth();
|
||||
const { toast, confirm } = useUI();
|
||||
const { instances, loaded, reload } = useInstances();
|
||||
const isAdmin = user?.role === 'admin';
|
||||
|
||||
const [frameLoaded, setFrameLoaded] = useState(false);
|
||||
const [dragging, setDragging] = useState(false);
|
||||
const [showFiles, setShowFiles] = useState(false);
|
||||
const [files, setFiles] = useState<TFile[]>([]);
|
||||
const [showClip, setShowClip] = useState(false);
|
||||
const [clipText, setClipText] = useState('');
|
||||
const [uploading, setUploading] = useState(false);
|
||||
const [starting, setStarting] = useState(false);
|
||||
const [control, setControl] = useState<{ free: boolean; mine: boolean; holder: string | null } | null>(null);
|
||||
const [vncNonce, setVncNonce] = useState(0);
|
||||
const fileInput = useRef<HTMLInputElement>(null);
|
||||
const frameRef = useRef<HTMLIFrameElement>(null);
|
||||
const dragDepth = useRef(0);
|
||||
const lastBeat = useRef(0);
|
||||
|
||||
// 文件拖到窗口任意位置时,弹出落区(覆盖 iframe,否则 drop 会被 iframe 吞掉)
|
||||
const inst = instances.find((i) => i.id === id);
|
||||
const offline = inst ? inst.runtime !== 'running' : false;
|
||||
const installed = !!inst && inst.wechat.installed && inst.wechat.phase !== 'downloading';
|
||||
const showVnc = !!inst && !offline && installed;
|
||||
|
||||
// 切换实例时重置内嵌态
|
||||
useEffect(() => {
|
||||
setFrameLoaded(false);
|
||||
setShowFiles(false);
|
||||
setFiles([]);
|
||||
setShowClip(false);
|
||||
setClipText('');
|
||||
}, [id]);
|
||||
|
||||
// 实例未就绪(启动中 / 安装中 / 上下文状态未刷新)时,每 3s 拉取最新状态:
|
||||
// 就绪后自动进入桌面,无需手动刷新(修复"安装完进度 100% 仍提示无实例")。
|
||||
useEffect(() => {
|
||||
if (showVnc || !id) return;
|
||||
const t = window.setInterval(() => {
|
||||
if (!document.hidden) reload();
|
||||
}, 3000);
|
||||
return () => window.clearInterval(t);
|
||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||
}, [showVnc, id]);
|
||||
|
||||
// 文件拖到窗口 → 弹出落区(覆盖 iframe 接住 drop)
|
||||
useEffect(() => {
|
||||
if (!showVnc) return;
|
||||
const hasFiles = (e: DragEvent) => Array.from(e.dataTransfer?.types || []).includes('Files');
|
||||
const onEnter = (e: DragEvent) => {
|
||||
if (!hasFiles(e)) return;
|
||||
@@ -45,9 +88,7 @@ export default function Desktop() {
|
||||
dragDepth.current++;
|
||||
setDragging(true);
|
||||
};
|
||||
const onOver = (e: DragEvent) => {
|
||||
if (hasFiles(e)) e.preventDefault();
|
||||
};
|
||||
const onOver = (e: DragEvent) => hasFiles(e) && e.preventDefault();
|
||||
const onLeave = (e: DragEvent) => {
|
||||
if (!hasFiles(e)) return;
|
||||
dragDepth.current = Math.max(0, dragDepth.current - 1);
|
||||
@@ -68,7 +109,65 @@ export default function Desktop() {
|
||||
window.removeEventListener('dragleave', onLeave);
|
||||
window.removeEventListener('drop', onDropWin);
|
||||
};
|
||||
}, []);
|
||||
}, [showVnc]);
|
||||
|
||||
// 控制权(交互驱动的心跳软锁):每 3s 只读轮询当前操作者;超 TTL 自动释放。
|
||||
useEffect(() => {
|
||||
if (!showVnc || !id) {
|
||||
setControl(null);
|
||||
return;
|
||||
}
|
||||
let alive = true;
|
||||
const poll = async () => {
|
||||
if (document.hidden) return;
|
||||
try {
|
||||
const r = await api.controlStatus(id);
|
||||
if (!alive) return;
|
||||
setControl(r);
|
||||
if (!r.free && !r.mine) frameRef.current?.blur(); // 只读:移开键盘焦点
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
};
|
||||
poll();
|
||||
const t = window.setInterval(poll, 3000);
|
||||
return () => {
|
||||
alive = false;
|
||||
window.clearInterval(t);
|
||||
};
|
||||
}, [showVnc, id]);
|
||||
|
||||
// 用户在 VNC 内真实操作(鼠标/键盘/滚轮)时续约控制权(同源 iframe 可监听)。节流 2.5s。
|
||||
// 只读用户的操作已被遮罩拦截/失焦,不会误续约;空闲不操作则超时自动释放。
|
||||
useEffect(() => {
|
||||
if (!showVnc || !id || !frameLoaded) return;
|
||||
const win = frameRef.current?.contentWindow;
|
||||
if (!win) return;
|
||||
const onInteract = async () => {
|
||||
const now = Date.now();
|
||||
if (now - lastBeat.current < 2500) return;
|
||||
lastBeat.current = now;
|
||||
try {
|
||||
const r = await api.controlBeat(id);
|
||||
setControl({ free: false, mine: r.mine, holder: r.holder });
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
};
|
||||
const evs = ['mousedown', 'keydown', 'wheel'] as const;
|
||||
try {
|
||||
evs.forEach((e) => win.addEventListener(e, onInteract, { capture: true, passive: true }));
|
||||
} catch {
|
||||
return;
|
||||
}
|
||||
return () => {
|
||||
try {
|
||||
evs.forEach((e) => win.removeEventListener(e, onInteract, { capture: true } as any));
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
};
|
||||
}, [showVnc, id, frameLoaded]);
|
||||
|
||||
if (!id) {
|
||||
nav('/', { replace: true });
|
||||
@@ -111,86 +210,359 @@ export default function Desktop() {
|
||||
if (e.dataTransfer.files?.length) uploadFiles(e.dataTransfer.files);
|
||||
};
|
||||
|
||||
const delFile = async (name: string) => {
|
||||
if (!(await confirm({ title: `删除「${name}」?`, body: '将从微信桌面(~/Desktop)移除该文件。', danger: true, confirmText: '删除' }))) return;
|
||||
try {
|
||||
await api.deleteFile(id, name);
|
||||
toast('已删除', 'ok');
|
||||
refreshFiles();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '删除失败', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
// 同源 iframe:把键盘焦点交给 VNC,帮助宿主机输入法把合成的字送进去
|
||||
const focusFrame = () => {
|
||||
try {
|
||||
frameRef.current?.focus();
|
||||
frameRef.current?.contentWindow?.focus();
|
||||
const ki = frameRef.current?.contentDocument?.getElementById('noVNC_keyboardinput') as HTMLElement | null;
|
||||
ki?.focus();
|
||||
} catch {
|
||||
/* 跨域兜底(正常同源不会到这) */
|
||||
}
|
||||
};
|
||||
|
||||
// 桌面加载后给 noVNC 原生控制条注入"实心可见"样式:原生背景近纯黑半透明,叠在深色/黑屏上看不见。
|
||||
// 注入后,用 KasmVNC 自带的左侧边缘手柄拉出控制条(音频/剪贴板/键盘/全屏等)时即可见。iframe 同源可直接访问。
|
||||
const injectVncStyle = () => {
|
||||
try {
|
||||
const doc = frameRef.current?.contentDocument;
|
||||
if (!doc || doc.getElementById('woc-vnc-style')) return;
|
||||
const st = doc.createElement('style');
|
||||
st.id = 'woc-vnc-style';
|
||||
st.textContent =
|
||||
'#noVNC_control_bar_anchor{z-index:2147483647!important;}' +
|
||||
'#noVNC_control_bar{background:rgba(18,22,30,.96)!important;border:1px solid rgba(255,255,255,.55)!important;box-shadow:0 0 24px rgba(0,0,0,.55)!important;}' +
|
||||
'#noVNC_control_bar_handle{opacity:1!important;background:rgba(18,22,30,.96)!important;border:1px solid rgba(255,255,255,.5)!important;}';
|
||||
(doc.head || doc.documentElement).appendChild(st);
|
||||
} catch {
|
||||
/* 同源正常不会到这 */
|
||||
}
|
||||
};
|
||||
|
||||
// 跨设备剪贴板(文本):通过同源 iframe 直接喂给 KasmVNC 自带的剪贴板 textarea 并触发其发送逻辑
|
||||
// (内部走 RFB.clipboardPasteFrom → clientCutText)。不依赖浏览器异步剪贴板 API,故 http/局域网 IP 下也可用,
|
||||
// 规避了"非安全上下文禁用 navigator.clipboard 导致粘贴失败"的问题。文本会进入容器系统剪贴板,
|
||||
// 在微信输入框按 Ctrl+V 即可粘贴。
|
||||
const pushClipboardToRemote = (text: string): boolean => {
|
||||
try {
|
||||
const doc = frameRef.current?.contentDocument;
|
||||
const ta = doc?.getElementById('noVNC_clipboard_text') as HTMLTextAreaElement | null;
|
||||
if (!doc || !ta) return false;
|
||||
ta.value = text;
|
||||
ta.dispatchEvent(new (frameRef.current!.contentWindow as any).Event('change', { bubbles: true }));
|
||||
return true;
|
||||
} catch {
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
const sendClip = () => {
|
||||
const t = clipText;
|
||||
if (!t) {
|
||||
toast('请先输入要发送的文本', 'error');
|
||||
return;
|
||||
}
|
||||
if (pushClipboardToRemote(t)) {
|
||||
toast('已发送到容器剪贴板,请在微信输入框按 Ctrl+V 粘贴', 'ok');
|
||||
} else {
|
||||
toast('发送失败:桌面尚未连接', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
// 读取容器(微信侧)当前剪贴板内容到本框,便于把容器内复制的文字带回本地
|
||||
const pullClipboardFromRemote = () => {
|
||||
try {
|
||||
const doc = frameRef.current?.contentDocument;
|
||||
const ta = doc?.getElementById('noVNC_clipboard_text') as HTMLTextAreaElement | null;
|
||||
if (ta) {
|
||||
setClipText(ta.value || '');
|
||||
toast('已读取容器剪贴板', 'ok');
|
||||
} else {
|
||||
toast('读取失败:桌面尚未连接', 'error');
|
||||
}
|
||||
} catch {
|
||||
toast('读取失败', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
const restartInstance = async () => {
|
||||
const ok = await confirm({
|
||||
title: '重启该实例?',
|
||||
body: '会重建容器(聊天记录保留),微信重新启动,约十几秒;用于修复卡死/最小化丢失等。',
|
||||
confirmText: '重启',
|
||||
});
|
||||
if (!ok) return;
|
||||
try {
|
||||
await api.instanceRestart(id);
|
||||
toast('已重启,正在重连…', 'ok');
|
||||
setFrameLoaded(false);
|
||||
setVncNonce((n) => n + 1); // 强制 iframe 重挂、重连
|
||||
await reload();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '重启失败', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
const takeControl = async () => {
|
||||
try {
|
||||
const r = await api.controlTake(id);
|
||||
setControl({ free: false, mine: r.mine, holder: r.holder });
|
||||
lastBeat.current = Date.now();
|
||||
focusFrame();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '接管失败', 'error');
|
||||
}
|
||||
};
|
||||
|
||||
const start = async () => {
|
||||
setStarting(true);
|
||||
try {
|
||||
await api.instanceStart(id);
|
||||
toast('实例已启动', 'ok');
|
||||
await reload();
|
||||
} catch (e: any) {
|
||||
toast(e.message || '启动失败', 'error');
|
||||
} finally {
|
||||
setStarting(false);
|
||||
}
|
||||
};
|
||||
|
||||
const title = inst?.name || '微信实例';
|
||||
|
||||
return (
|
||||
<div className="desktop-wrap">
|
||||
<iframe
|
||||
className="desktop-frame"
|
||||
src={desktopUrl(id)}
|
||||
title="电脑版微信"
|
||||
allow="clipboard-read; clipboard-write; microphone; camera; autoplay"
|
||||
onLoad={() => setLoaded(true)}
|
||||
/>
|
||||
<div className="ws-page">
|
||||
<header className="ws-head">
|
||||
<button className="ws-menu" onClick={onOpenMenu} aria-label="菜单">
|
||||
{MenuIcon}
|
||||
</button>
|
||||
<span className="ws-title">{title}</span>
|
||||
{showVnc && (
|
||||
<>
|
||||
<button
|
||||
className="ws-action"
|
||||
title="文件传输"
|
||||
onClick={() => {
|
||||
setShowFiles((v) => !v);
|
||||
if (!showFiles) refreshFiles();
|
||||
}}
|
||||
>
|
||||
文件
|
||||
</button>
|
||||
<button
|
||||
className="ws-action"
|
||||
title="把文本发送到容器剪贴板(局域网 http 下也可用)"
|
||||
onClick={() => setShowClip((v) => !v)}
|
||||
>
|
||||
剪贴板
|
||||
</button>
|
||||
{isAdmin && (
|
||||
<button className="ws-action" title="重启实例(修复卡死/最小化丢失)" onClick={restartInstance}>
|
||||
重启
|
||||
</button>
|
||||
)}
|
||||
</>
|
||||
)}
|
||||
</header>
|
||||
|
||||
{!loaded && (
|
||||
<div className="desktop-loading">
|
||||
{/* —— 各种态 —— */}
|
||||
{!loaded ? (
|
||||
<div className="iv-stage iv-center">
|
||||
<div className="spinner" />
|
||||
<div className="desktop-loading-text">正在连接桌面…</div>
|
||||
<div className="desktop-loading-sub">首次进入请扫码登录微信</div>
|
||||
<div className="desktop-loading-sub">拖文件到窗口即可上传;语音/视频在左侧工具条开启</div>
|
||||
{!window.isSecureContext && (
|
||||
<div className="desktop-loading-warn">当前非 HTTPS 访问,浏览器将禁用麦克风与摄像头(音频播放不受影响)</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* 拖拽落区:仅拖入文件时出现,覆盖 iframe 接住 drop */}
|
||||
{dragging && (
|
||||
<div className="drop-zone" onDrop={onDrop} onDragOver={(e) => e.preventDefault()}>
|
||||
<div className="drop-card">
|
||||
<div className="drop-icon">⬇</div>
|
||||
<div className="drop-title">松开上传到微信桌面</div>
|
||||
<div className="drop-sub">上传后在微信里「+ / 文件」选择即可</div>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<button className="desktop-back" onClick={() => nav('/')} title="返回">
|
||||
‹
|
||||
</button>
|
||||
|
||||
{/* 文件按钮 */}
|
||||
<button
|
||||
className="desktop-files-btn"
|
||||
title="文件传输"
|
||||
onClick={() => {
|
||||
setShowFiles((v) => !v);
|
||||
if (!showFiles) refreshFiles();
|
||||
}}
|
||||
>
|
||||
⇅
|
||||
</button>
|
||||
|
||||
{showFiles && (
|
||||
<div className="files-panel">
|
||||
<div className="files-head">
|
||||
<span>文件传输</span>
|
||||
<button className="btn-text" onClick={() => setShowFiles(false)}>
|
||||
关闭
|
||||
) : !inst ? (
|
||||
<div className="iv-stage iv-center">
|
||||
<div className="iv-notice">
|
||||
<div className="iv-notice-title">无权访问或实例不存在</div>
|
||||
<button className="btn btn-primary iv-notice-btn" onClick={() => nav('/')}>
|
||||
返回主页
|
||||
</button>
|
||||
</div>
|
||||
<input
|
||||
ref={fileInput}
|
||||
type="file"
|
||||
multiple
|
||||
style={{ display: 'none' }}
|
||||
onChange={(e) => {
|
||||
if (e.target.files) uploadFiles(e.target.files);
|
||||
e.target.value = '';
|
||||
</div>
|
||||
) : offline ? (
|
||||
<div className="iv-stage iv-center">
|
||||
<div className="iv-notice">
|
||||
<div className="iv-notice-title">{inst.runtime === 'missing' ? '容器尚未创建' : '实例已停止'}</div>
|
||||
{isAdmin ? (
|
||||
<button className="btn btn-primary iv-notice-btn" disabled={starting} onClick={start}>
|
||||
{starting ? '启动中…' : inst.runtime === 'missing' ? '创建并启动' : '启动实例'}
|
||||
</button>
|
||||
) : (
|
||||
<div className="iv-notice-sub">请联系管理员启动该实例</div>
|
||||
)}
|
||||
{isAdmin && (
|
||||
<button className="btn-text" onClick={() => window.open(api.instanceLogsUrl(id), '_blank')}>
|
||||
查看日志
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
) : ['downloading', 'extracting', 'installing'].includes(inst.wechat.phase) ? (
|
||||
<div className="iv-stage iv-center">
|
||||
<div className="iv-notice">
|
||||
<div className="spinner" />
|
||||
<div className="iv-notice-title">微信安装中…</div>
|
||||
<div className="iv-notice-sub">
|
||||
{inst.wechat.message || '请稍候'}
|
||||
{inst.wechat.percent >= 0 ? ` · ${inst.wechat.percent}%` : ''} ——完成后自动进入,无需刷新
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
) : !installed ? (
|
||||
<div className="iv-stage iv-center">
|
||||
<div className="iv-notice">
|
||||
<div className="iv-notice-title">{inst.wechat.phase === 'error' ? '微信安装出错' : '微信尚未安装'}</div>
|
||||
<div className="iv-notice-sub">
|
||||
{inst.wechat.phase === 'error'
|
||||
? inst.wechat.message || '安装失败,可在「管理」重试'
|
||||
: '该实例容器已就绪,但尚未安装微信'}
|
||||
</div>
|
||||
{isAdmin ? (
|
||||
<button className="btn btn-primary iv-notice-btn" onClick={() => nav('/admin')}>
|
||||
去「管理」{inst.wechat.phase === 'error' ? '重试 / 更新' : '下载安装'}
|
||||
</button>
|
||||
) : (
|
||||
<div className="iv-notice-sub">请联系管理员在「管理」中下载安装微信</div>
|
||||
)}
|
||||
{isAdmin && (
|
||||
<button className="btn-text" onClick={() => window.open(api.instanceLogsUrl(id), '_blank')}>
|
||||
查看日志
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="iv-stage">
|
||||
<iframe
|
||||
key={`${id}:${vncNonce}`}
|
||||
ref={frameRef}
|
||||
className="iv-frame"
|
||||
src={desktopUrl(id)}
|
||||
title="电脑版微信"
|
||||
allow="clipboard-read; clipboard-write; microphone; camera; autoplay"
|
||||
onLoad={() => {
|
||||
setFrameLoaded(true);
|
||||
setTimeout(() => {
|
||||
focusFrame(); // 加载完把键盘焦点交给 VNC(宿主机输入法)
|
||||
injectVncStyle(); // 让原生控制条在深色背景下可见
|
||||
}, 500);
|
||||
}}
|
||||
/>
|
||||
<button className="btn btn-primary files-upload" disabled={uploading} onClick={() => fileInput.current?.click()}>
|
||||
{uploading ? '上传中…' : '+ 选择文件上传'}
|
||||
</button>
|
||||
<div className="files-hint">也可直接把文件拖到窗口。下方为桌面(~/Desktop)里的文件,微信收到的文件另存到桌面即可在此下载。</div>
|
||||
<div className="files-list">
|
||||
{files.length === 0 && <div className="muted small" style={{ padding: '10px 2px' }}>暂无文件</div>}
|
||||
{files.map((f) => (
|
||||
<a key={f.name} className="files-item" href={api.downloadFileUrl(id, f.name)} download={f.name}>
|
||||
<span className="files-name">{f.name}</span>
|
||||
<span className="files-size">{humanSize(f.size)} ↓</span>
|
||||
</a>
|
||||
))}
|
||||
</div>
|
||||
|
||||
{!frameLoaded && (
|
||||
<div className="iv-loading">
|
||||
<div className="spinner" />
|
||||
<div className="iv-loading-text">正在连接桌面…</div>
|
||||
<div className="iv-loading-sub">首次进入请扫码登录微信</div>
|
||||
<div className="iv-loading-sub">拖文件到此处即可上传;音频/剪贴板等在画面左侧边缘的工具条里</div>
|
||||
{!window.isSecureContext && (
|
||||
<div className="iv-loading-warn">当前非 HTTPS 访问,浏览器将禁用麦克风与摄像头(音频播放不受影响)</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{dragging && (
|
||||
<div className="iv-drop" onDrop={onDrop} onDragOver={(e) => e.preventDefault()}>
|
||||
<div className="drop-card">
|
||||
<div className="drop-icon">⬇</div>
|
||||
<div className="drop-title">松开上传到微信桌面</div>
|
||||
<div className="drop-sub">上传后在微信里「+ / 文件」选择即可</div>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{control && !control.free && !control.mine && (
|
||||
<div className="iv-lock">
|
||||
<div className="iv-lock-card">
|
||||
<div className="iv-lock-title">「{control.holder}」正在操作</div>
|
||||
<div className="iv-lock-sub">为避免多端互相干扰,你当前为只读模式。</div>
|
||||
<button className="btn btn-primary iv-notice-btn" onClick={takeControl}>
|
||||
申请控制
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{showFiles && (
|
||||
<div className="iv-files">
|
||||
<div className="files-head">
|
||||
<span>文件传输</span>
|
||||
<button className="btn-text" onClick={() => setShowFiles(false)}>
|
||||
关闭
|
||||
</button>
|
||||
</div>
|
||||
<input
|
||||
ref={fileInput}
|
||||
type="file"
|
||||
multiple
|
||||
style={{ display: 'none' }}
|
||||
onChange={(e) => {
|
||||
if (e.target.files) uploadFiles(e.target.files);
|
||||
e.target.value = '';
|
||||
}}
|
||||
/>
|
||||
<button className="btn btn-primary files-upload" disabled={uploading} onClick={() => fileInput.current?.click()}>
|
||||
{uploading ? '上传中…' : '+ 选择文件上传'}
|
||||
</button>
|
||||
<div className="files-hint">也可直接把文件拖进来。下方为桌面(~/Desktop)里的文件,微信收到的文件另存到桌面即可在此下载。</div>
|
||||
<div className="files-list">
|
||||
{files.length === 0 && (
|
||||
<div className="muted small" style={{ padding: '10px 2px' }}>
|
||||
暂无文件
|
||||
</div>
|
||||
)}
|
||||
{files.map((f) => (
|
||||
<div key={f.name} className="files-item">
|
||||
<a className="files-dl" href={api.downloadFileUrl(id, f.name)} download={f.name} title="下载">
|
||||
<span className="files-name">{f.name}</span>
|
||||
<span className="files-size">{humanSize(f.size)} ↓</span>
|
||||
</a>
|
||||
<button className="files-del" title="删除" onClick={() => delFile(f.name)}>
|
||||
✕
|
||||
</button>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{showClip && (
|
||||
<div className="iv-files">
|
||||
<div className="files-head">
|
||||
<span>文本剪贴板</span>
|
||||
<button className="btn-text" onClick={() => setShowClip(false)}>
|
||||
关闭
|
||||
</button>
|
||||
</div>
|
||||
<textarea
|
||||
className="clip-area"
|
||||
value={clipText}
|
||||
onChange={(e) => setClipText(e.target.value)}
|
||||
placeholder="在此输入或粘贴文本,点「发送到微信」后到微信输入框按 Ctrl+V 粘贴"
|
||||
rows={5}
|
||||
/>
|
||||
<button className="btn btn-primary files-upload" onClick={sendClip}>
|
||||
发送到微信(容器剪贴板)
|
||||
</button>
|
||||
<button className="btn-text" style={{ alignSelf: 'flex-start', marginTop: 6 }} onClick={pullClipboardFromRemote}>
|
||||
↓ 读取容器剪贴板到此框
|
||||
</button>
|
||||
<div className="files-hint">
|
||||
局域网 http 访问时浏览器会禁用系统级剪贴板同步,故用此框中转:文本→容器剪贴板,再在微信里 Ctrl+V。
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
@@ -26,27 +26,33 @@ export default function Login() {
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="center-screen">
|
||||
<form className="card login-card" onSubmit={submit}>
|
||||
<div className="brand">
|
||||
<div className="brand-logo"><img src="/favicon.svg" alt="" /></div>
|
||||
<h1>云微</h1>
|
||||
<p className="muted">登录以访问 NAS 上的微信</p>
|
||||
</div>
|
||||
<input
|
||||
className="input"
|
||||
placeholder="用户名"
|
||||
autoCapitalize="off"
|
||||
autoCorrect="off"
|
||||
value={username}
|
||||
onChange={(e) => setUsername(e.target.value)}
|
||||
/>
|
||||
<PasswordInput placeholder="密码" autoComplete="current-password" value={password} onChange={setPassword} />
|
||||
{err && <div className="error">{err}</div>}
|
||||
<button className="btn btn-primary" disabled={busy || !username || !password}>
|
||||
{busy ? '登录中…' : '登录'}
|
||||
</button>
|
||||
</form>
|
||||
<div className="center-screen login-screen">
|
||||
<div className="login-wrap">
|
||||
<form className="card login-card" onSubmit={submit}>
|
||||
<div className="brand">
|
||||
<div className="brand-logo">
|
||||
<img src="/favicon.svg" alt="" />
|
||||
</div>
|
||||
<h1>云微</h1>
|
||||
<p className="muted">登录以访问 NAS 上的微信</p>
|
||||
</div>
|
||||
<input
|
||||
className="input"
|
||||
placeholder="用户名"
|
||||
autoCapitalize="off"
|
||||
autoCorrect="off"
|
||||
autoComplete="username"
|
||||
value={username}
|
||||
onChange={(e) => setUsername(e.target.value)}
|
||||
/>
|
||||
<PasswordInput placeholder="密码" autoComplete="current-password" value={password} onChange={setPassword} />
|
||||
{err && <div className="error">{err}</div>}
|
||||
<button className="btn btn-primary" disabled={busy || !username || !password}>
|
||||
{busy ? '登录中…' : '登录'}
|
||||
</button>
|
||||
</form>
|
||||
<div className="login-foot">服务端微信 · 多端共享 · 建议仅在内网 / 可信网络访问</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
+567
-4
@@ -107,10 +107,30 @@ button {
|
||||
}
|
||||
|
||||
/* 登录 */
|
||||
.login-screen {
|
||||
background:
|
||||
radial-gradient(60% 50% at 50% 0%, rgba(var(--green-rgb) / 0.1), transparent 70%),
|
||||
var(--base);
|
||||
}
|
||||
.login-wrap {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
width: 100%;
|
||||
max-width: 360px;
|
||||
}
|
||||
.login-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 14px;
|
||||
padding: 28px 24px;
|
||||
}
|
||||
.login-foot {
|
||||
font-size: 12px;
|
||||
color: var(--muted);
|
||||
text-align: center;
|
||||
padding: 0 12px;
|
||||
}
|
||||
.brand {
|
||||
text-align: center;
|
||||
@@ -693,6 +713,25 @@ button {
|
||||
height: 42px;
|
||||
font-size: 15px;
|
||||
}
|
||||
.clip-area {
|
||||
width: 100%;
|
||||
resize: vertical;
|
||||
min-height: 96px;
|
||||
padding: 10px 12px;
|
||||
border-radius: 14px;
|
||||
border: none;
|
||||
background: var(--mf-trough, #edeef1);
|
||||
color: var(--text, #1a1d24);
|
||||
font-size: 14px;
|
||||
line-height: 1.5;
|
||||
font-family: inherit;
|
||||
box-shadow: inset 0 1px 3px rgba(51, 66, 102, 0.16);
|
||||
outline: none;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
.clip-area:focus {
|
||||
box-shadow: inset 0 1px 3px rgba(51, 66, 102, 0.16), 0 0 0 2px rgba(7, 193, 96, 0.35);
|
||||
}
|
||||
.files-hint {
|
||||
font-size: 12px;
|
||||
color: var(--muted);
|
||||
@@ -704,6 +743,17 @@ button {
|
||||
flex-direction: column;
|
||||
}
|
||||
.files-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
box-shadow: inset 0 -1px 0 rgba(var(--shadow) / 0.08);
|
||||
}
|
||||
.files-item:last-child {
|
||||
box-shadow: none;
|
||||
}
|
||||
.files-dl {
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
@@ -712,10 +762,6 @@ button {
|
||||
text-decoration: none;
|
||||
color: var(--text);
|
||||
font-size: 14px;
|
||||
box-shadow: inset 0 -1px 0 rgba(var(--shadow) / 0.08);
|
||||
}
|
||||
.files-item:last-child {
|
||||
box-shadow: none;
|
||||
}
|
||||
.files-name {
|
||||
overflow: hidden;
|
||||
@@ -728,6 +774,21 @@ button {
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.files-del {
|
||||
flex: none;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
border: none;
|
||||
background: none;
|
||||
color: var(--muted);
|
||||
font-size: 14px;
|
||||
cursor: pointer;
|
||||
border-radius: 8px;
|
||||
}
|
||||
.files-del:active {
|
||||
background: rgba(var(--danger-rgb) / 0.12);
|
||||
color: var(--danger);
|
||||
}
|
||||
|
||||
/* ── loading ───────────────────────────────────────────── */
|
||||
.spinner {
|
||||
@@ -1086,3 +1147,505 @@ button {
|
||||
text-align: center;
|
||||
padding: 0 24px;
|
||||
}
|
||||
|
||||
/* ============================================================
|
||||
* 微信 PC 式布局:左侧 tab 栏 + 右侧工作区
|
||||
* ============================================================ */
|
||||
.shell {
|
||||
display: flex;
|
||||
height: 100%;
|
||||
min-height: 0;
|
||||
background: var(--base);
|
||||
}
|
||||
|
||||
/* —— 左侧栏 —— */
|
||||
.sidebar {
|
||||
flex: none;
|
||||
width: 264px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
background: var(--surface);
|
||||
box-shadow: 1px 0 0 rgba(var(--shadow) / 0.08), 4px 0 16px rgba(var(--shadow) / 0.05);
|
||||
z-index: 20;
|
||||
transition: width 0.2s cubic-bezier(0.2, 0.8, 0.2, 1);
|
||||
padding-top: env(safe-area-inset-top);
|
||||
}
|
||||
.shell.collapsed .sidebar {
|
||||
width: 64px;
|
||||
}
|
||||
.sb-top {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 8px;
|
||||
padding: 14px 14px 10px;
|
||||
}
|
||||
.sb-brand {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
min-width: 0;
|
||||
}
|
||||
.sb-logo {
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
flex: none;
|
||||
border-radius: 9px;
|
||||
}
|
||||
.sb-name {
|
||||
font-size: 17px;
|
||||
font-weight: 700;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.sb-collapse {
|
||||
flex: none;
|
||||
width: 32px;
|
||||
height: 32px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border: none;
|
||||
background: none;
|
||||
color: var(--muted);
|
||||
cursor: pointer;
|
||||
border-radius: 9px;
|
||||
}
|
||||
.sb-collapse:active {
|
||||
background: rgba(var(--shadow) / 0.06);
|
||||
}
|
||||
.collapsed .sb-top {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.sb-nav {
|
||||
padding: 4px 10px;
|
||||
}
|
||||
.sb-section {
|
||||
padding: 12px 16px 6px;
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
color: var(--muted);
|
||||
}
|
||||
.sb-list {
|
||||
flex: 1;
|
||||
min-height: 0;
|
||||
overflow-y: auto;
|
||||
padding: 0 10px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
}
|
||||
.sb-empty {
|
||||
padding: 10px 16px;
|
||||
font-size: 13px;
|
||||
color: var(--muted);
|
||||
}
|
||||
|
||||
.sb-item {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
width: 100%;
|
||||
border: none;
|
||||
background: none;
|
||||
cursor: pointer;
|
||||
padding: 9px 12px;
|
||||
border-radius: 12px;
|
||||
font-size: 15px;
|
||||
color: var(--text);
|
||||
text-align: left;
|
||||
transition: background 0.15s;
|
||||
}
|
||||
.sb-item:hover {
|
||||
background: rgba(var(--shadow) / 0.05);
|
||||
}
|
||||
.sb-item.on {
|
||||
background: rgba(var(--green-rgb) / 0.12);
|
||||
color: var(--wx-green-dark);
|
||||
}
|
||||
.sb-item.on .sb-ic {
|
||||
color: var(--wx-green-dark);
|
||||
}
|
||||
.sb-ic {
|
||||
flex: none;
|
||||
width: 24px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
color: var(--muted);
|
||||
}
|
||||
.sb-label {
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
font-weight: 500;
|
||||
}
|
||||
.collapsed .sb-item {
|
||||
justify-content: center;
|
||||
padding: 9px 0;
|
||||
}
|
||||
|
||||
/* 实例头像 + 状态点 */
|
||||
.sb-avatar {
|
||||
position: relative;
|
||||
flex: none;
|
||||
width: 34px;
|
||||
height: 34px;
|
||||
border-radius: 10px;
|
||||
background: linear-gradient(150deg, var(--wx-green), var(--wx-green-dark));
|
||||
color: #fff;
|
||||
font-weight: 700;
|
||||
font-size: 16px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
.sb-dot {
|
||||
position: absolute;
|
||||
right: -2px;
|
||||
bottom: -2px;
|
||||
width: 11px;
|
||||
height: 11px;
|
||||
border-radius: 50%;
|
||||
border: 2px solid var(--surface);
|
||||
}
|
||||
.sb-stxt {
|
||||
flex: none;
|
||||
font-size: 11px;
|
||||
font-weight: 600;
|
||||
color: var(--muted);
|
||||
}
|
||||
.st-on {
|
||||
background: #16c060;
|
||||
color: var(--wx-green-dark);
|
||||
}
|
||||
.st-off {
|
||||
background: #c2c7d0;
|
||||
color: var(--muted);
|
||||
}
|
||||
.st-busy {
|
||||
background: #2f7ad0;
|
||||
color: #2f5fd0;
|
||||
}
|
||||
.st-warn {
|
||||
background: #e6a23c;
|
||||
color: #b9770a;
|
||||
}
|
||||
|
||||
.sb-footer {
|
||||
padding: 8px 10px calc(10px + env(safe-area-inset-bottom));
|
||||
box-shadow: inset 0 1px 0 rgba(var(--shadow) / 0.08);
|
||||
}
|
||||
.sb-user {
|
||||
padding: 8px 12px 2px;
|
||||
font-size: 12px;
|
||||
color: var(--muted);
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
/* —— 右侧工作区 —— */
|
||||
.workspace {
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
height: 100%;
|
||||
min-height: 0;
|
||||
}
|
||||
.ws-page {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
height: 100%;
|
||||
min-height: 0;
|
||||
}
|
||||
.ws-head {
|
||||
flex: none;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
height: calc(52px + env(safe-area-inset-top));
|
||||
padding: env(safe-area-inset-top) 16px 0;
|
||||
background: var(--base);
|
||||
box-shadow: inset 0 -1px 0 rgba(var(--shadow) / 0.07);
|
||||
}
|
||||
.ws-title {
|
||||
font-size: 16px;
|
||||
font-weight: 700;
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.ws-menu {
|
||||
display: none;
|
||||
flex: none;
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border: none;
|
||||
background: none;
|
||||
color: var(--text);
|
||||
cursor: pointer;
|
||||
border-radius: 9px;
|
||||
margin-left: -6px;
|
||||
}
|
||||
.ws-action {
|
||||
flex: none;
|
||||
border: none;
|
||||
background: var(--surface);
|
||||
color: var(--wx-green-dark);
|
||||
font-size: 13px;
|
||||
font-weight: 600;
|
||||
padding: 6px 11px;
|
||||
border-radius: 999px;
|
||||
cursor: pointer;
|
||||
box-shadow: var(--crease);
|
||||
}
|
||||
.ws-action:active {
|
||||
transform: scale(0.96);
|
||||
}
|
||||
.ws-page .content {
|
||||
flex: 1;
|
||||
min-height: 0;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
/* 主页实例快捷卡 */
|
||||
.home-card {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
width: 100%;
|
||||
text-align: left;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
background: var(--surface);
|
||||
border-radius: var(--r-card);
|
||||
padding: 16px;
|
||||
box-shadow: var(--crease);
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
transition: transform 0.16s cubic-bezier(0.2, 0.8, 0.2, 1);
|
||||
}
|
||||
.home-card::before {
|
||||
content: '';
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
border-radius: inherit;
|
||||
background: var(--sheen);
|
||||
pointer-events: none;
|
||||
}
|
||||
.home-card:active {
|
||||
transform: scale(0.985);
|
||||
}
|
||||
.home-card-av {
|
||||
position: relative;
|
||||
flex: none;
|
||||
width: 42px;
|
||||
height: 42px;
|
||||
border-radius: 12px;
|
||||
background: linear-gradient(150deg, var(--wx-green), var(--wx-green-dark));
|
||||
color: #fff;
|
||||
font-weight: 700;
|
||||
font-size: 20px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
.home-card-main {
|
||||
position: relative;
|
||||
flex: 1;
|
||||
min-width: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 3px;
|
||||
}
|
||||
.home-card-name {
|
||||
font-size: 16px;
|
||||
font-weight: 700;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.home-card-st {
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
background: none !important;
|
||||
}
|
||||
|
||||
/* —— 内嵌实例视图 —— */
|
||||
.iv-stage {
|
||||
position: relative;
|
||||
flex: 1;
|
||||
min-height: 0;
|
||||
background: #000;
|
||||
}
|
||||
.iv-center {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
background: var(--base);
|
||||
}
|
||||
.iv-frame {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
border: none;
|
||||
display: block;
|
||||
}
|
||||
.iv-loading,
|
||||
.iv-drop {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 12px;
|
||||
z-index: 5;
|
||||
}
|
||||
.iv-loading {
|
||||
background: var(--base);
|
||||
}
|
||||
.iv-loading-text {
|
||||
font-size: 16px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.iv-loading-sub {
|
||||
font-size: 13px;
|
||||
color: var(--muted);
|
||||
text-align: center;
|
||||
padding: 0 24px;
|
||||
}
|
||||
.iv-loading-warn {
|
||||
margin-top: 4px;
|
||||
font-size: 12px;
|
||||
color: var(--danger);
|
||||
text-align: center;
|
||||
padding: 0 24px;
|
||||
}
|
||||
.iv-drop {
|
||||
z-index: 30;
|
||||
background: rgba(7, 193, 96, 0.16);
|
||||
-webkit-backdrop-filter: blur(3px);
|
||||
backdrop-filter: blur(3px);
|
||||
}
|
||||
.iv-files {
|
||||
position: absolute;
|
||||
top: 14px;
|
||||
right: 14px;
|
||||
width: min(330px, calc(100% - 28px));
|
||||
max-height: calc(100% - 28px);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 10px;
|
||||
background: var(--surface);
|
||||
border-radius: var(--r-card);
|
||||
box-shadow: var(--crease), 0 8px 30px rgba(var(--shadow) / 0.18);
|
||||
z-index: 12;
|
||||
padding: 16px;
|
||||
}
|
||||
/* 只读遮罩:他人正在操作时盖住 VNC,拦截键鼠(半透,仍能看到画面) */
|
||||
.iv-lock {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
z-index: 8;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
background: rgba(26, 29, 36, 0.28);
|
||||
-webkit-backdrop-filter: blur(1.5px);
|
||||
backdrop-filter: blur(1.5px);
|
||||
}
|
||||
.iv-lock-card {
|
||||
background: var(--surface);
|
||||
border-radius: var(--r-card);
|
||||
box-shadow: var(--crease), 0 8px 30px rgba(var(--shadow) / 0.25);
|
||||
padding: 22px 26px;
|
||||
text-align: center;
|
||||
max-width: 320px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
}
|
||||
.iv-lock-title {
|
||||
font-size: 16px;
|
||||
font-weight: 700;
|
||||
}
|
||||
.iv-lock-sub {
|
||||
font-size: 13px;
|
||||
color: var(--muted);
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.iv-notice {
|
||||
text-align: center;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 14px;
|
||||
padding: 24px;
|
||||
}
|
||||
.iv-notice-title {
|
||||
font-size: 17px;
|
||||
font-weight: 700;
|
||||
}
|
||||
.iv-notice-sub {
|
||||
font-size: 13px;
|
||||
color: var(--muted);
|
||||
}
|
||||
.iv-notice-btn {
|
||||
padding: 0 22px;
|
||||
height: 44px;
|
||||
}
|
||||
|
||||
/* —— 移动端:侧栏变抽屉 —— */
|
||||
.shell-backdrop {
|
||||
display: none;
|
||||
}
|
||||
@media (max-width: 767px) {
|
||||
.sidebar {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
width: 82vw;
|
||||
max-width: 320px;
|
||||
transform: translateX(-100%);
|
||||
transition: transform 0.25s cubic-bezier(0.2, 0.8, 0.2, 1);
|
||||
box-shadow: 0 0 40px rgba(var(--shadow) / 0.35);
|
||||
}
|
||||
.shell.drawer-open .sidebar {
|
||||
transform: none;
|
||||
}
|
||||
.shell.drawer-open .shell-backdrop {
|
||||
display: block;
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background: rgba(26, 29, 36, 0.4);
|
||||
z-index: 15;
|
||||
}
|
||||
.ws-menu {
|
||||
display: flex;
|
||||
}
|
||||
.ws-action {
|
||||
padding: 7px 12px;
|
||||
}
|
||||
}
|
||||
|
||||
/* 宽屏:工作区内容容器更宽(实例网格多列) */
|
||||
@media (min-width: 880px) {
|
||||
.ws-page .content {
|
||||
max-width: 940px;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,6 +29,10 @@ export default defineConfig({
|
||||
workbox: {
|
||||
// 桌面反代与 API 不能被 SW 拦截
|
||||
navigateFallbackDenylist: [/^\/desktop/, /^\/api/],
|
||||
// 新版本立即接管 + 清理旧缓存,避免更新后仍跑旧代码(硬刷新绕不过 SW)
|
||||
clientsClaim: true,
|
||||
skipWaiting: true,
|
||||
cleanupOutdatedCaches: true,
|
||||
},
|
||||
}),
|
||||
],
|
||||
|
||||
Reference in New Issue
Block a user