6.8 KiB
更新中心列表图标设计
背景
当前 Electron 更新中心已经可以展示更新项、来源、迁移标记、进度和日志,但更新列表仍然只有文字信息,没有应用图标。对于 APM 包、传统 deb 包和迁移项,纯文字列表会降低识别效率,尤其在批量更新和搜索场景下不够直观。
仓库现状里已经存在多套可复用的图标来源逻辑:
- 主商店卡片通过远程商店 URL 拼接
icon.png。 - 已安装应用列表支持本地图标和远程 URL 双来源。
- 旧 Qt 更新器会为 APM 更新项解析 desktop 与 entries/icons,并在无本地图标时继续使用其他数据源。
目标是在更新中心列表中加入应用图标,同时保持最小改动、兼容当前后端结构,并遵循“本地解析优先,其次远程 URL,最后占位图标”的策略。
目标
- 在更新中心列表中为每个更新项展示应用图标。
- 图标来源优先级为:本地解析 > 远程 URL > 前端占位图标。
- 前后端仅增加一个最小公共字段,不引入复杂的图标对象结构。
- 图标缺失或加载失败时,界面仍然保持稳定、整齐、不闪烁。
非目标
- 不为图标来源新增额外网络探测请求。
- 不在本次设计中重构应用详情页、已安装列表或主商店卡片的图标逻辑。
- 不在 UI 中展示“图标来源”说明文字。
方案概览
采用“主进程解析来源、渲染层只展示”的方案:
- 更新中心主进程在加载更新项时解析图标来源,并将结果写入更新项的
icon字段。 - 渲染层更新列表只消费
item.icon,不参与解析来源。 - 前端负责单次图片加载失败回退到占位图标。
数据结构变化
主进程
修改:electron/main/backend/update-center/types.ts
为 UpdateCenterItem 增加:
icon?: string;
渲染层
修改:src/global/typedefinition.ts
为 UpdateCenterItem 增加:
icon?: string;
Service 映射
修改:electron/main/backend/update-center/service.ts
在主进程 snapshot -> renderer snapshot 的映射中透传 icon 字段。
图标来源策略
优先级
每个更新项统一按以下顺序取图标:
- 本地图标路径
- 远程商店图标 URL
- 前端占位图标
1. 本地图标路径
传统 deb / Spark 更新项
优先复用仓库中已有的 desktop 文件扫描与 Icon= 解析思路,来源参考:
electron/main/backend/install-manager.ts
解析策略:
- 从已安装包对应的 desktop 文件中读取
Icon=。 - 如果解析结果为绝对路径,直接返回。
- 如果解析结果为图标名,则尝试根据系统图标路径补全。
- 若无法得到有效路径,则继续下一层来源。
APM 更新项
优先复用旧 Qt 更新器已存在的 APM 图标解析逻辑,来源参考:
spark-update-tool/src/aptssupdater.cpp
解析策略:
- 查找 APM 包的
entries/applications/*.desktop。 - 从 desktop 的
Icon=字段中解析图标。 - 若
Icon=为绝对路径,直接返回。 - 若
Icon=为图标名,则尝试拼接 APM 包内entries/icons/...路径。 - 若仍无结果,则继续下一层来源。
2. 远程商店图标 URL
如果本地图标解析失败,则为更新项生成远程图标 URL。
实现原则:
- 不主动探测 URL 是否可用。
- 仅按现有商店规则拼接 URL,并交给浏览器加载。
- 浏览器加载失败后由前端回退占位图标。
对 Spark/传统 deb:
- 使用当前商店已有的远程图标拼接规则。
- 若更新项可以推断出对应 category 和 arch,则拼接:
${APM_STORE_BASE_URL}/${arch}/${category}/${pkgname}/icon.png
对 APM:
- 若仓库中已有 APM 对应商店资源约定,则使用同样的
icon.png规则。 - 若当前数据无法可靠推断 category,则允许直接跳过远程 URL,进入前端占位图标。
3. 占位图标
如果主进程未能提供 icon,或者前端加载失败,则使用统一占位图标。
占位规则:
- 图标尺寸与正常图标一致。
- 使用仓库现有品牌资源或统一默认应用图标。
- 不因失败状态改变列表布局高度或间距。
模块边界
新增:
electron/main/backend/update-center/icons.ts
职责:
resolveUpdateItemIcon()resolveApmIcon()resolveDesktopIcon()buildRemoteFallbackIconUrl()
该模块只负责“根据更新项得到一个 icon?: string”,不参与更新队列、安装、刷新、忽略等逻辑。
数据流
主进程加载更新项
- 查询并合并更新项。
- 对每个更新项执行图标解析。
- 将解析到的
icon字段写入UpdateCenterItem。 - 由
service.ts将该字段透传到渲染层 snapshot。
渲染层展示
UpdateCenterItem.vue读取item.icon。- 如果
item.icon为本地绝对路径,则转成file://URL。 - 如果
item.icon为远程 URL,则直接作为图片地址使用。 - 若图片加载失败,则切换为占位图标,并记住失败状态避免重复尝试。
UI 设计
列表项布局
在更新列表中新增一个固定图标位:
- 位置:复选框后、应用信息前。
- 尺寸:
40x40。 - 样式:圆角矩形,视觉与商店应用卡片图标一致。
- 图标位固定占位,避免有图和无图的项出现布局跳动。
失败回退
前端仅做一次失败回退:
- 优先渲染
item.icon。 - 触发
@error后切换为占位图。 - 记录该项失败状态,避免反复向无效地址重新请求。
测试方案
主进程测试
新增或扩展测试覆盖:
- 本地图标优先于远程 URL。
- APM 更新项可解析包内 desktop/icons。
- 传统 deb 更新项可解析 desktop
Icon=。 - 无本地图标时能生成远程 URL 或返回空值。
组件测试
扩展 UpdateCenterItem.vue 组件测试:
- 有
item.icon时渲染图片。 - 图片加载失败时回退到占位图。
- 图标存在时不影响当前状态标签、迁移标签、进度条显示。
风险与约束
- 更新项当前不一定总能推断出 category,因此远程 URL 兜底对部分项可能不可用;这是可接受的,因为前端还有占位图兜底。
- 本地图标解析涉及多个来源路径,必须限制在读取路径和拼接路径,不做额外昂贵的同步探测。
- APM 图标路径依赖当前系统安装结构,若个别包结构不标准,应直接退回远程或占位图,而不是阻断更新列表。
决策总结
- 更新中心增加单字段
icon?: string,不引入复杂图标对象。 - 主进程解析图标来源,渲染层只负责展示和失败回退。
- 图标来源顺序固定为:本地解析 > 远程 URL > 占位图。
- UI 仅新增稳定图标位,不改变现有更新列表信息层级。