feat: 优化图标分页选择器与中文返回信息
This commit is contained in:
231
README.md
231
README.md
@@ -1,6 +1,6 @@
|
|||||||
# ShortX ToolHub
|
# ShortX ToolHub
|
||||||
|
|
||||||
一个模块化的 ShortX JS 浮窗工具框架,支持广播关闭、子线程模型、日志记录、自动下载与热更新。
|
一个模块化的 ShortX JS 浮窗工具框架,支持子模块自动下载、按 `Last-Modified` 热更新、启动日志记录,以及面向大图标库的分页式 ShortX 图标选择器。
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -10,31 +10,32 @@
|
|||||||
```
|
```
|
||||||
shortx.getShortXDir()/
|
shortx.getShortXDir()/
|
||||||
├── ToolHub/
|
├── ToolHub/
|
||||||
│ └── code/
|
│ ├── code/
|
||||||
│ ├── th_01_base.js
|
│ │ ├── th_01_base.js
|
||||||
│ ├── th_02_core.js
|
│ │ ├── th_02_core.js
|
||||||
│ ├── th_03_icon.js
|
│ │ ├── th_03_icon.js
|
||||||
│ ├── th_04_theme.js
|
│ │ ├── th_04_theme.js
|
||||||
│ ├── th_05_persistence.js
|
│ │ ├── th_05_persistence.js
|
||||||
│ ├── th_06_icon_parser.js
|
│ │ ├── th_06_icon_parser.js
|
||||||
│ ├── th_07_shortcut.js
|
│ │ ├── th_07_shortcut.js
|
||||||
│ ├── th_08_content.js
|
│ │ ├── th_08_content.js
|
||||||
│ ├── th_09_animation.js
|
│ │ ├── th_09_animation.js
|
||||||
│ ├── th_10_shell.js
|
│ │ ├── th_10_shell.js
|
||||||
│ ├── th_11_action.js
|
│ │ ├── th_11_action.js
|
||||||
│ ├── th_12_rebuild.js
|
│ │ ├── th_12_rebuild.js
|
||||||
│ ├── th_13_panel_ui.js
|
│ │ ├── th_13_panel_ui.js
|
||||||
│ ├── th_14_panels.js
|
│ │ ├── th_14_panels.js
|
||||||
│ ├── th_15_extra.js
|
│ │ ├── th_15_extra.js
|
||||||
│ └── th_16_entry.js
|
│ │ └── th_16_entry.js
|
||||||
└── ToolHub/logs/
|
│ └── logs/
|
||||||
└── init.log
|
│ └── init.log
|
||||||
```
|
```
|
||||||
|
|
||||||
### 服务器(项目维护目录)
|
### 服务器(项目维护目录)
|
||||||
```
|
```
|
||||||
ToolHub/
|
ToolHub/
|
||||||
├── ToolHub.js # 入口文件(粘贴到 ShortX 任务)
|
├── ToolHub.js
|
||||||
|
├── README.md
|
||||||
└── code/
|
└── code/
|
||||||
├── th_01_base.js
|
├── th_01_base.js
|
||||||
├── th_02_core.js
|
├── th_02_core.js
|
||||||
@@ -58,135 +59,131 @@ ToolHub/
|
|||||||
|
|
||||||
## 部署步骤
|
## 部署步骤
|
||||||
|
|
||||||
### 1. 创建目录(可省略)
|
### 1. 放置入口文件
|
||||||
|
将 `ToolHub.js` 内容粘贴到 ShortX 任务中。
|
||||||
|
|
||||||
入口文件会自动检测并创建 `ToolHub/code/` 目录,无需手动操作。
|
### 2. 直接运行
|
||||||
|
入口文件会自动:
|
||||||
|
1. 检查并创建 `ToolHub/code/`
|
||||||
|
2. 修复目录权限(`chmod 700` + `chown 1000:1000`)
|
||||||
|
3. 对 16 个子模块做 HEAD 检测
|
||||||
|
4. 缺失或有更新时自动下载覆盖
|
||||||
|
5. 记录启动日志到 `ToolHub/logs/init.log`
|
||||||
|
|
||||||
若需手动创建,在 ShortX 数据根目录下执行:
|
---
|
||||||
```bash
|
|
||||||
mkdir -p ToolHub/code
|
|
||||||
chmod 700 ToolHub/code
|
|
||||||
chown 1000:1000 ToolHub/code
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. 放置入口文件
|
## 返回信息格式
|
||||||
|
|
||||||
将 `ToolHub.js` 的内容粘贴到 ShortX 任务中。
|
成功启动示例:
|
||||||
|
|
||||||
子模块会自动从 git 仓库下载到 `ToolHub/code/`,无需手动复制。
|
|
||||||
|
|
||||||
### 3. 运行
|
|
||||||
|
|
||||||
执行 ShortX 任务,正常返回示例:
|
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
"ok": true,
|
"ok": true,
|
||||||
"started": true,
|
"started": true,
|
||||||
"msg": "已按 WM 专属 HandlerThread 模型启动",
|
"msg": "ToolHub 启动成功:已按 WM 专属 HandlerThread 模型启动",
|
||||||
|
"syncMsg": "本次已覆盖更新 2 个子模块(新增 0 / 覆盖 2):th_14_panels.js、th_16_entry.js",
|
||||||
|
"updatedCount": 2,
|
||||||
|
"updatedModules": ["th_14_panels.js", "th_16_entry.js"],
|
||||||
"closeAction": "shortx.wm.floatball.CLOSE",
|
"closeAction": "shortx.wm.floatball.CLOSE",
|
||||||
"layout": {"cols": 2, "rows": 2}
|
"layout": {"cols": 2, "rows": 2}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
无子模块更新时:
|
||||||
|
```json
|
||||||
## 自动下载与权限管理
|
{
|
||||||
|
"ok": true,
|
||||||
入口文件启动时会自动完成以下操作:
|
"started": true,
|
||||||
|
"msg": "ToolHub 启动成功:已按 WM 专属 HandlerThread 模型启动",
|
||||||
1. **缺失自检**:检查 `ToolHub/code/` 下的 16 个模块文件,缺失则从 git raw URL 自动下载
|
"syncMsg": "子模块已是最新,本次未覆盖更新。",
|
||||||
2. **权限保障**:目录不存在时自动创建并设置 `chmod 700` + `chown 1000:1000`
|
"updatedCount": 0,
|
||||||
3. **权限判断**:通过 `stat` 命令精确检查 uid/gid/mode,不正确才修复
|
"updatedModules": [],
|
||||||
4. **单次检查**:一次启动中只检查一次目录权限,避免重复 shell 开销
|
"closeAction": "shortx.wm.floatball.CLOSE",
|
||||||
|
"layout": {"cols": 2, "rows": 2}
|
||||||
---
|
}
|
||||||
|
|
||||||
## 版本管理与热更新
|
|
||||||
|
|
||||||
每个模块文件第一行必须包含版本注释:
|
|
||||||
```javascript
|
|
||||||
// @version 1.0.0
|
|
||||||
```
|
```
|
||||||
|
|
||||||
入口文件中的 `MODULE_MANIFEST` 定义各模块的期望版本。启动时若本地版本与期望版本不匹配,自动重新下载。
|
如果有非关键模块加载失败,会额外返回:
|
||||||
|
- `loadMsg`
|
||||||
|
- `loadErrors`
|
||||||
|
|
||||||
升级模块时只需:
|
如果启动失败,会额外返回:
|
||||||
1. 更新模块文件中的 `@version` 版本号
|
- `err`
|
||||||
2. 同步更新 `ToolHub.js` 中 `MODULE_MANIFEST` 的对应版本号
|
|
||||||
3. 推送到 git 仓库
|
|
||||||
4. 实机下次启动时自动检测并更新
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 下载校验
|
## 子模块热更新机制
|
||||||
|
|
||||||
- **大小校验**:对比 HTTP `Content-Length` 与实际写入字节数,不匹配则抛异常
|
当前机制不是版本号比对,而是:
|
||||||
- **内容校验**:读取下载文件前 200 字节,检测 `<!DOCTYPE` 或 `<html`,防止下到 404/502 错误页面
|
- 对每个子模块发送 HTTP HEAD 请求
|
||||||
|
- 读取远端 `Last-Modified`
|
||||||
|
- 与本地 `.lm_模块名` 缓存比较
|
||||||
|
- 不一致则重新下载覆盖
|
||||||
|
- 网络检查失败时优先回退到本地已有文件
|
||||||
|
|
||||||
|
因此:
|
||||||
|
- **入口文件无需维护 `MODULE_MANIFEST`**
|
||||||
|
- 模块里的 `@version` 仅可作为人工注释,不参与程序判断
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## ShortX 图标选择器优化
|
||||||
|
|
||||||
|
按钮编辑页里的 ShortX 图标选择器现已改为:
|
||||||
|
- **分页模式**,不再一次性塞入大批图标
|
||||||
|
- **一页显示多少个图标,就按当前 UI 实际可见容量自动计算 pageSize**
|
||||||
|
- 保留 **搜索 / 分类 / 上一页 / 下一页**
|
||||||
|
- 选中图标后自动回填并收起
|
||||||
|
- 收起后再次点击 **展开图标库** 可正常重新打开
|
||||||
|
|
||||||
|
当前交互要点:
|
||||||
|
1. 切到 `ShortX图标` 时自动展开图标库
|
||||||
|
2. 页容量按当前可见行数 × 4 列实时计算
|
||||||
|
3. 搜索、切分类、翻页时都会回到顶部,减少卡顿感和误触
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 日志系统
|
## 日志系统
|
||||||
|
|
||||||
### 启动日志
|
启动日志路径:
|
||||||
路径:`shortx.getShortXDir() + "/ToolHub/logs/init.log"`
|
```text
|
||||||
|
shortx.getShortXDir() + "/ToolHub/logs/init.log"
|
||||||
记录内容:
|
|
||||||
- 目录创建/权限修复
|
|
||||||
- 模块下载开始/结束/异常
|
|
||||||
- 版本不匹配
|
|
||||||
- 模块加载失败
|
|
||||||
- 模块体积超阈警告(>200KB)
|
|
||||||
|
|
||||||
### 运行日志
|
|
||||||
路径:`shortx.getShortXDir() + "/ToolHub/logs/"`
|
|
||||||
|
|
||||||
日志文件按天分割,默认保留 3 天。
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 关闭浮窗
|
|
||||||
|
|
||||||
通过 adb 或 ShortX Shell 执行:
|
|
||||||
```bash
|
|
||||||
am broadcast -a shortx.wm.floatball.CLOSE
|
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
记录内容包括:
|
||||||
|
- 目录创建 / 权限修复
|
||||||
## 模块说明
|
- 更新检测
|
||||||
|
- 下载开始 / 结束 / 异常
|
||||||
| 文件 | 职责 | 线数参考 |
|
- 模块加载失败
|
||||||
|------|------|---------|
|
- 模块体积告警(>200KB)
|
||||||
| `th_01_base.js` | 基础工具函数、Logger、崩溃处理、进程信息获取 | ~1300 |
|
|
||||||
| `th_02_core.js` | FloatBallAppWM 构造函数、基础工具方法(dp/now/clamp) | ~124 |
|
|
||||||
| `th_03_icon.js` | 图标缓存/LRU、悬浮球图标加载(PNG 文件) | ~170 |
|
|
||||||
| `th_04_theme.js` | 屏幕/旋转、UI 样式辅助、莫奈动态取色、主题检测 | ~800 |
|
|
||||||
| `th_05_persistence.js` | 面板/位置/配置持久化、设置面板 schema 与编辑缓存 | ~298 |
|
|
||||||
| `th_06_icon_parser.js` | 快捷方式图标解析、resolveIconDrawable、图标文件路径 | ~485 |
|
|
||||||
| `th_07_shortcut.js` | 内置快捷方式选择器(合并 shortcuts.js)、分组过滤 | ~1175 |
|
|
||||||
| `th_08_content.js` | ContentProvider URI 解析、通用 query、统一入口 | ~209 |
|
|
||||||
| `th_09_animation.js` | 动画/视图管理、吸边停靠、屏幕监控、重建悬浮球 | ~662 |
|
|
||||||
| `th_10_shell.js` | Shell 智能执行(Action 优先 + 广播桥兜底) | ~33 |
|
|
||||||
| `th_11_action.js` | 按钮动作执行(点击/长按/双击) | ~320 |
|
|
||||||
| `th_12_rebuild.js` | 改大小后安全重建悬浮球 | ~76 |
|
|
||||||
| `th_13_panel_ui.js` | 设置面板 UI 组件(SectionHeader、SettingItemView 等) | ~375 |
|
|
||||||
| `th_14_panels.js` | 面板配置工厂、按钮构建器、对话框、文本查看器 | ~2957 |
|
|
||||||
| `th_15_extra.js` | 额外面板:设备信息、网络状态、快捷操作 | ~1598 |
|
|
||||||
| `th_16_entry.js` | 入口面板定义、广播接收器注册、启动流程 | ~324 |
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 模块加载容错
|
## 模块职责
|
||||||
|
|
||||||
- `for` 循环加载 16 个模块,单模块失败记录日志但不阻断后续加载
|
| 文件 | 职责 |
|
||||||
- `th_16_entry.js` 失败时直接抛异常(启动必备)
|
|------|------|
|
||||||
- 错误信息落盘到 `ToolHub/logs/init.log`,便于实机排查
|
| `th_01_base.js` | 基础工具、日志、配置校验、通用辅助 |
|
||||||
|
| `th_02_core.js` | `FloatBallAppWM` 构造、基础状态与核心工具 |
|
||||||
|
| `th_03_icon.js` | 图标缓存、Bitmap 管理、悬浮球图标加载 |
|
||||||
|
| `th_04_theme.js` | 主题、颜色、样式工具 |
|
||||||
|
| `th_05_persistence.js` | 持久化与设置数据层 |
|
||||||
|
| `th_06_icon_parser.js` | 图标解析、ShortX 内置图标扫描与回退 |
|
||||||
|
| `th_07_shortcut.js` | 快捷方式选择器 |
|
||||||
|
| `th_08_content.js` | ContentProvider 读取与通用 query |
|
||||||
|
| `th_09_animation.js` | 面板动画、吸边、显示/隐藏管理 |
|
||||||
|
| `th_10_shell.js` | Shell 执行层 |
|
||||||
|
| `th_11_action.js` | 按钮动作分发与执行 |
|
||||||
|
| `th_12_rebuild.js` | 悬浮球重建逻辑 |
|
||||||
|
| `th_13_panel_ui.js` | 设置面板通用 UI 组件 |
|
||||||
|
| `th_14_panels.js` | 设置面板、按钮编辑器、ShortX 图标分页选择器 |
|
||||||
|
| `th_15_extra.js` | 主面板与附加展示层 |
|
||||||
|
| `th_16_entry.js` | 生命周期、广播注册、启动与销毁 |
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 注意事项
|
## 注意事项
|
||||||
|
|
||||||
- 入口文件通过 `loadScript()` 动态加载子模块,`var` 声明通过间接 `eval` 挂到全局作用域
|
- 入口文件只负责加载与汇总返回信息,不承载业务 UI
|
||||||
- 子模块加载顺序不可更改:base → core → icon → theme → persistence → icon_parser → shortcut → content → animation → shell → action → rebuild → panel_ui → panels → extra → entry
|
- `th_16_entry.js` 属于关键模块,加载失败会直接中止启动
|
||||||
- 调试请查看日志文件,不通过返回 JSON 暴露内部细节
|
- 不建议把调试细节直接塞进返回 JSON,优先写日志;返回信息只保留用户判断启动/更新所需的关键信息
|
||||||
- 单个模块建议不超过 200KB,超过时启动日志会记录 WARN 提示拆分
|
- 若修改了模块结构、返回字段、图标选择器交互,记得同步更新 README 与相关技能说明
|
||||||
|
|||||||
203
ToolHub.js
203
ToolHub.js
@@ -6729,18 +6729,79 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
statusTv: null,
|
statusTv: null,
|
||||||
searchEt: null,
|
searchEt: null,
|
||||||
grid: null,
|
grid: null,
|
||||||
|
gridScroll: null,
|
||||||
pickerWrap: null,
|
pickerWrap: null,
|
||||||
toggleBtn: null,
|
toggleBtn: null,
|
||||||
clearBtn: null,
|
clearBtn: null,
|
||||||
pageSize: 20,
|
pageSize: 0,
|
||||||
currentPage: 0,
|
currentPage: 0,
|
||||||
activeTab: "all",
|
activeTab: "all",
|
||||||
tabButtons: {},
|
tabButtons: {},
|
||||||
pageInfoTv: null,
|
pageInfoTv: null,
|
||||||
prevBtn: null,
|
prevBtn: null,
|
||||||
nextBtn: null
|
nextBtn: null,
|
||||||
|
pageCols: 4,
|
||||||
|
pageRows: 0,
|
||||||
|
cellWidthDp: 76,
|
||||||
|
cellHeightDp: 92,
|
||||||
|
cellMarginDp: 4,
|
||||||
|
lastMeasuredGridHeight: 0
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function getShortXPickerClosedLabel() {
|
||||||
|
return "展开图标库";
|
||||||
|
}
|
||||||
|
|
||||||
|
function getShortXPickerOpenedLabel() {
|
||||||
|
return "收起图标库";
|
||||||
|
}
|
||||||
|
|
||||||
|
function scrollShortXGridToTop() {
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.gridScroll) {
|
||||||
|
shortxPickerState.gridScroll.post(new java.lang.Runnable({
|
||||||
|
run: function() {
|
||||||
|
try { shortxPickerState.gridScroll.fullScroll(android.view.View.FOCUS_UP); } catch(eScroll0) {}
|
||||||
|
try { shortxPickerState.gridScroll.scrollTo(0, 0); } catch(eScroll1) {}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} catch(eScrollWrap) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveShortXPickerPageSize() {
|
||||||
|
var cols = Number(shortxPickerState.pageCols || 4);
|
||||||
|
if (cols < 1) cols = 4;
|
||||||
|
var fallbackHeight = self.dp(520);
|
||||||
|
var rawHeight = 0;
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.gridScroll) rawHeight = Number(shortxPickerState.gridScroll.getHeight() || 0);
|
||||||
|
} catch(eH0) {}
|
||||||
|
if (rawHeight <= 0) rawHeight = fallbackHeight;
|
||||||
|
var cellOuterHeight = self.dp(Number(shortxPickerState.cellHeightDp || 92) + Number(shortxPickerState.cellMarginDp || 4) * 2);
|
||||||
|
if (cellOuterHeight <= 0) cellOuterHeight = self.dp(100);
|
||||||
|
var rows = Math.max(1, Math.floor(rawHeight / cellOuterHeight));
|
||||||
|
var size = Math.max(cols, rows * cols);
|
||||||
|
shortxPickerState.pageRows = rows;
|
||||||
|
shortxPickerState.lastMeasuredGridHeight = rawHeight;
|
||||||
|
shortxPickerState.pageSize = size;
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setShortXPickerExpanded(expanded, doRender) {
|
||||||
|
shortxPickerState.expanded = !!expanded;
|
||||||
|
if (shortxPickerState.pickerWrap) {
|
||||||
|
shortxPickerState.pickerWrap.setVisibility(shortxPickerState.expanded ? android.view.View.VISIBLE : android.view.View.GONE);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText(shortxPickerState.expanded ? getShortXPickerOpenedLabel() : getShortXPickerClosedLabel());
|
||||||
|
} catch(eToggleTxt) {}
|
||||||
|
if (shortxPickerState.expanded && doRender !== false) {
|
||||||
|
resolveShortXPickerPageSize();
|
||||||
|
renderShortXIconGrid();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var shortxQuickRow = new android.widget.LinearLayout(context);
|
var shortxQuickRow = new android.widget.LinearLayout(context);
|
||||||
shortxQuickRow.setOrientation(android.widget.LinearLayout.HORIZONTAL);
|
shortxQuickRow.setOrientation(android.widget.LinearLayout.HORIZONTAL);
|
||||||
shortxQuickRow.setGravity(android.view.Gravity.CENTER_VERTICAL);
|
shortxQuickRow.setGravity(android.view.Gravity.CENTER_VERTICAL);
|
||||||
@@ -6774,14 +6835,9 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxBtnGap.setLayoutParams(new android.widget.LinearLayout.LayoutParams(self.dp(8), 1));
|
shortxBtnGap.setLayoutParams(new android.widget.LinearLayout.LayoutParams(self.dp(8), 1));
|
||||||
shortxQuickRow.addView(shortxBtnGap);
|
shortxQuickRow.addView(shortxBtnGap);
|
||||||
|
|
||||||
var btnBrowseShortXIcon = self.ui.createFlatButton(self, "图标库", C.primary, function() {
|
var btnBrowseShortXIcon = self.ui.createFlatButton(self, getShortXPickerClosedLabel(), C.primary, function() {
|
||||||
self.touchActivity();
|
self.touchActivity();
|
||||||
shortxPickerState.expanded = !shortxPickerState.expanded;
|
setShortXPickerExpanded(!shortxPickerState.expanded, true);
|
||||||
if (shortxPickerState.pickerWrap) {
|
|
||||||
shortxPickerState.pickerWrap.setVisibility(shortxPickerState.expanded ? android.view.View.VISIBLE : android.view.View.GONE);
|
|
||||||
}
|
|
||||||
try { if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText(shortxPickerState.expanded ? "收起" : "图标库"); } catch(eT1) {}
|
|
||||||
if (shortxPickerState.expanded) renderShortXIconGrid();
|
|
||||||
});
|
});
|
||||||
shortxPickerState.toggleBtn = btnBrowseShortXIcon;
|
shortxPickerState.toggleBtn = btnBrowseShortXIcon;
|
||||||
shortxQuickRow.addView(btnBrowseShortXIcon);
|
shortxQuickRow.addView(btnBrowseShortXIcon);
|
||||||
@@ -10935,85 +10991,80 @@ FloatBallAppWM.prototype.startAsync = function(entryProcInfo, closeRule) {
|
|||||||
|
|
||||||
// =======================【执行(入口线程)】======================
|
// =======================【执行(入口线程)】======================
|
||||||
var __out = (function() {
|
var __out = (function() {
|
||||||
|
function optStr(v) {
|
||||||
|
return (v === undefined || v === null) ? "" : String(v);
|
||||||
|
}
|
||||||
|
function summarizeModuleUpdates(list) {
|
||||||
|
var names = [];
|
||||||
|
var created = 0;
|
||||||
|
var overwritten = 0;
|
||||||
|
var i;
|
||||||
|
for (i = 0; i < list.length; i++) {
|
||||||
|
var item = list[i] || {};
|
||||||
|
var name = optStr(item.module);
|
||||||
|
if (name) names.push(name);
|
||||||
|
if (item.isNew) created++; else overwritten++;
|
||||||
|
}
|
||||||
|
if (names.length === 0) {
|
||||||
|
return {
|
||||||
|
count: 0,
|
||||||
|
modules: [],
|
||||||
|
msg: "子模块已是最新,本次未覆盖更新。"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
count: names.length,
|
||||||
|
modules: names,
|
||||||
|
msg: "本次已覆盖更新 " + names.length + " 个子模块(新增 " + created + " / 覆盖 " + overwritten + "):" + names.join("、")
|
||||||
|
};
|
||||||
|
}
|
||||||
|
function summarizeLoadErrors(list) {
|
||||||
|
var names = [];
|
||||||
|
var i;
|
||||||
|
for (i = 0; i < list.length; i++) {
|
||||||
|
var item = list[i] || {};
|
||||||
|
var name = optStr(item.module);
|
||||||
|
if (name) names.push(name);
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
count: names.length,
|
||||||
|
modules: names,
|
||||||
|
msg: names.length ? ("有 " + names.length + " 个子模块加载失败:" + names.join("、")) : "所有子模块加载正常。"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
var entryInfo = getProcessInfo("entry");
|
var entryInfo = getProcessInfo("entry");
|
||||||
|
|
||||||
// # 初始化 logger
|
|
||||||
var logger = new ToolHubLogger(entryInfo);
|
var logger = new ToolHubLogger(entryInfo);
|
||||||
// # 安装崩溃处理
|
installCrashHandler(logger);
|
||||||
var crashHandlerInstalled = installCrashHandler(logger);
|
|
||||||
|
|
||||||
// # 启动 app
|
|
||||||
var app = new FloatBallAppWM(logger);
|
var app = new FloatBallAppWM(logger);
|
||||||
|
|
||||||
// # 计算广播 action(支持规则)
|
|
||||||
var closeRule = String(app.config.ACTION_CLOSE_ALL_RULE || "shortx.wm.floatball.CLOSE");
|
var closeRule = String(app.config.ACTION_CLOSE_ALL_RULE || "shortx.wm.floatball.CLOSE");
|
||||||
|
|
||||||
var startRet = null;
|
var startRet = null;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
startRet = app.startAsync(entryInfo, closeRule);
|
startRet = app.startAsync(entryInfo, closeRule);
|
||||||
} catch (eTop) {
|
} catch (eTop) {
|
||||||
try { logger.fatal("TOP startAsync crash err=" + String(eTop)); } catch (eLog) {}
|
try { logger.fatal("TOP startAsync crash err=" + String(eTop)); } catch (eLog) {}
|
||||||
startRet = { ok: false, err: String(eTop) };
|
startRet = { ok: false, err: String(eTop) };
|
||||||
}
|
}
|
||||||
|
var syncInfo = summarizeModuleUpdates(__moduleUpdates);
|
||||||
// # 中文摘要(方便 ShortX 日志/结果中快速识别)
|
var loadInfo = summarizeLoadErrors(loadErrors);
|
||||||
var _btnCount = (startRet && startRet.buttons != null) ? startRet.buttons : 0;
|
var started = !!(startRet && startRet.ok);
|
||||||
var _layout = (startRet && startRet.layout) ? startRet.layout : { cols: app.config.PANEL_COLS, rows: app.config.PANEL_ROWS };
|
var rawMsg = optStr(startRet && startRet.msg);
|
||||||
var _closeAction = String(startRet && startRet.closeAction ? startRet.closeAction : "shortx.wm.floatball.CLOSE");
|
var out = {
|
||||||
var _logEnabled = !!app.config.LOG_ENABLE;
|
ok: started,
|
||||||
var _logDays = Math.max(1, Math.floor(Number(app.config.LOG_KEEP_DAYS || 3)));
|
started: started,
|
||||||
|
msg: started ? (rawMsg ? ("ToolHub 启动成功:" + rawMsg) : "ToolHub 启动成功") : "ToolHub 启动失败",
|
||||||
// # 返回信息
|
syncMsg: syncInfo.msg,
|
||||||
return {
|
updatedCount: syncInfo.count,
|
||||||
ok: true,
|
updatedModules: syncInfo.modules,
|
||||||
result: startRet,
|
closeAction: optStr(startRet && startRet.closeAction),
|
||||||
process: entryInfo,
|
layout: startRet && startRet.layout || null
|
||||||
crashHandlerInstalled: !!crashHandlerInstalled,
|
|
||||||
log: {
|
|
||||||
enable: _logEnabled,
|
|
||||||
debug: !!app.config.LOG_DEBUG,
|
|
||||||
keepDays: _logDays,
|
|
||||||
dirActive: String(logger.dir || ""),
|
|
||||||
prefix: String(app.config.LOG_PREFIX || "ShortX_ToolHub"),
|
|
||||||
lastInitErr: String(logger.lastInitErr || "")
|
|
||||||
},
|
|
||||||
shell: {
|
|
||||||
useActionFirstDefault: false,
|
|
||||||
hasShellCommandClass: false,
|
|
||||||
bridge: {
|
|
||||||
action: String(app.config.SHELL_BRIDGE_ACTION),
|
|
||||||
extraCmd: String(app.config.SHELL_BRIDGE_EXTRA_CMD),
|
|
||||||
extraFrom: String(app.config.SHELL_BRIDGE_EXTRA_FROM),
|
|
||||||
extraRoot: String(app.config.SHELL_BRIDGE_EXTRA_ROOT),
|
|
||||||
defaultRoot: !!app.config.SHELL_BRIDGE_DEFAULT_ROOT
|
|
||||||
}
|
|
||||||
},
|
|
||||||
content: {
|
|
||||||
maxRows: Number(app.config.CONTENT_MAX_ROWS || 20),
|
|
||||||
viewerTextSp: Number(app.config.CONTENT_VIEWER_TEXT_SP || 12)
|
|
||||||
},
|
|
||||||
suggestCloseShell: "am broadcast -a " + _closeAction,
|
|
||||||
suggestTaskerProfile: {
|
|
||||||
event: "Intent Received",
|
|
||||||
action: String(app.config.SHELL_BRIDGE_ACTION),
|
|
||||||
readExtras: [
|
|
||||||
{ key: String(app.config.SHELL_BRIDGE_EXTRA_CMD), var: "%cmd" },
|
|
||||||
{ key: String(app.config.SHELL_BRIDGE_EXTRA_ROOT), var: "%root" }
|
|
||||||
],
|
|
||||||
exec: "Run Shell (root=%root) cmd=%cmd"
|
|
||||||
},
|
|
||||||
// ========== 中文摘要(供 ShortX 结果查看)==========
|
|
||||||
状态: (startRet && startRet.ok) ? "✅ 启动成功" : "❌ 启动失败",
|
|
||||||
悬浮球: (startRet && startRet.ok) ? "已显示" : "未显示",
|
|
||||||
关闭指令: "am broadcast -a " + _closeAction,
|
|
||||||
按钮数量: String(_btnCount) + " 个",
|
|
||||||
面板布局: String(_layout.cols) + " 列 × " + String(_layout.rows) + " 行",
|
|
||||||
日志: (_logEnabled ? "📝 已启用" : "📝 已禁用") + " · 保留 " + String(_logDays) + " 天",
|
|
||||||
崩溃处理: !!crashHandlerInstalled ? "🛡️ 已安装" : "⚠️ 未安装",
|
|
||||||
线程模型: "HandlerThread(WM 专属)",
|
|
||||||
消息: (startRet && startRet.msg) ? String(startRet.msg) : ""
|
|
||||||
};
|
};
|
||||||
|
if (loadInfo.count > 0) {
|
||||||
|
out.loadMsg = loadInfo.msg;
|
||||||
|
out.loadErrors = loadInfo.modules;
|
||||||
|
}
|
||||||
|
if (!started) out.err = optStr(startRet && startRet.err) || "未知错误";
|
||||||
|
return out;
|
||||||
})();
|
})();
|
||||||
|
|
||||||
JSON.stringify(__out);
|
JSON.stringify(__out);
|
||||||
|
|||||||
@@ -671,18 +671,79 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
statusTv: null,
|
statusTv: null,
|
||||||
searchEt: null,
|
searchEt: null,
|
||||||
grid: null,
|
grid: null,
|
||||||
|
gridScroll: null,
|
||||||
pickerWrap: null,
|
pickerWrap: null,
|
||||||
toggleBtn: null,
|
toggleBtn: null,
|
||||||
clearBtn: null,
|
clearBtn: null,
|
||||||
pageSize: 20,
|
pageSize: 0,
|
||||||
currentPage: 0,
|
currentPage: 0,
|
||||||
activeTab: "all",
|
activeTab: "all",
|
||||||
tabButtons: {},
|
tabButtons: {},
|
||||||
pageInfoTv: null,
|
pageInfoTv: null,
|
||||||
prevBtn: null,
|
prevBtn: null,
|
||||||
nextBtn: null
|
nextBtn: null,
|
||||||
|
pageCols: 4,
|
||||||
|
pageRows: 0,
|
||||||
|
cellWidthDp: 76,
|
||||||
|
cellHeightDp: 92,
|
||||||
|
cellMarginDp: 4,
|
||||||
|
lastMeasuredGridHeight: 0
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function getShortXPickerClosedLabel() {
|
||||||
|
return "展开图标库";
|
||||||
|
}
|
||||||
|
|
||||||
|
function getShortXPickerOpenedLabel() {
|
||||||
|
return "收起图标库";
|
||||||
|
}
|
||||||
|
|
||||||
|
function scrollShortXGridToTop() {
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.gridScroll) {
|
||||||
|
shortxPickerState.gridScroll.post(new java.lang.Runnable({
|
||||||
|
run: function() {
|
||||||
|
try { shortxPickerState.gridScroll.fullScroll(android.view.View.FOCUS_UP); } catch(eScroll0) {}
|
||||||
|
try { shortxPickerState.gridScroll.scrollTo(0, 0); } catch(eScroll1) {}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} catch(eScrollWrap) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveShortXPickerPageSize() {
|
||||||
|
var cols = Number(shortxPickerState.pageCols || 4);
|
||||||
|
if (cols < 1) cols = 4;
|
||||||
|
var fallbackHeight = self.dp(520);
|
||||||
|
var rawHeight = 0;
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.gridScroll) rawHeight = Number(shortxPickerState.gridScroll.getHeight() || 0);
|
||||||
|
} catch(eH0) {}
|
||||||
|
if (rawHeight <= 0) rawHeight = fallbackHeight;
|
||||||
|
var cellOuterHeight = self.dp(Number(shortxPickerState.cellHeightDp || 92) + Number(shortxPickerState.cellMarginDp || 4) * 2);
|
||||||
|
if (cellOuterHeight <= 0) cellOuterHeight = self.dp(100);
|
||||||
|
var rows = Math.max(1, Math.floor(rawHeight / cellOuterHeight));
|
||||||
|
var size = Math.max(cols, rows * cols);
|
||||||
|
shortxPickerState.pageRows = rows;
|
||||||
|
shortxPickerState.lastMeasuredGridHeight = rawHeight;
|
||||||
|
shortxPickerState.pageSize = size;
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setShortXPickerExpanded(expanded, doRender) {
|
||||||
|
shortxPickerState.expanded = !!expanded;
|
||||||
|
if (shortxPickerState.pickerWrap) {
|
||||||
|
shortxPickerState.pickerWrap.setVisibility(shortxPickerState.expanded ? android.view.View.VISIBLE : android.view.View.GONE);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText(shortxPickerState.expanded ? getShortXPickerOpenedLabel() : getShortXPickerClosedLabel());
|
||||||
|
} catch(eToggleTxt) {}
|
||||||
|
if (shortxPickerState.expanded && doRender !== false) {
|
||||||
|
resolveShortXPickerPageSize();
|
||||||
|
renderShortXIconGrid();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var shortxQuickRow = new android.widget.LinearLayout(context);
|
var shortxQuickRow = new android.widget.LinearLayout(context);
|
||||||
shortxQuickRow.setOrientation(android.widget.LinearLayout.HORIZONTAL);
|
shortxQuickRow.setOrientation(android.widget.LinearLayout.HORIZONTAL);
|
||||||
shortxQuickRow.setGravity(android.view.Gravity.CENTER_VERTICAL);
|
shortxQuickRow.setGravity(android.view.Gravity.CENTER_VERTICAL);
|
||||||
@@ -716,17 +777,9 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxBtnGap.setLayoutParams(new android.widget.LinearLayout.LayoutParams(self.dp(8), 1));
|
shortxBtnGap.setLayoutParams(new android.widget.LinearLayout.LayoutParams(self.dp(8), 1));
|
||||||
shortxQuickRow.addView(shortxBtnGap);
|
shortxQuickRow.addView(shortxBtnGap);
|
||||||
|
|
||||||
var btnBrowseShortXIcon = self.ui.createFlatButton(self, "选择图标", C.primary, function() {
|
var btnBrowseShortXIcon = self.ui.createFlatButton(self, getShortXPickerClosedLabel(), C.primary, function() {
|
||||||
self.touchActivity();
|
self.touchActivity();
|
||||||
self.showIconPicker({
|
setShortXPickerExpanded(!shortxPickerState.expanded, true);
|
||||||
onPick: function(iconName) {
|
|
||||||
try {
|
|
||||||
var shortName = self.normalizeShortXIconName(iconName, false);
|
|
||||||
inputShortXIcon.input.setText(shortName);
|
|
||||||
updateShortXIconPreview();
|
|
||||||
} catch(ePick) {}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
shortxPickerState.toggleBtn = btnBrowseShortXIcon;
|
shortxPickerState.toggleBtn = btnBrowseShortXIcon;
|
||||||
shortxQuickRow.addView(btnBrowseShortXIcon);
|
shortxQuickRow.addView(btnBrowseShortXIcon);
|
||||||
@@ -752,7 +805,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxPickerState.pickerWrap = shortxPickerWrap;
|
shortxPickerState.pickerWrap = shortxPickerWrap;
|
||||||
|
|
||||||
var shortxPickerHead = new android.widget.TextView(context);
|
var shortxPickerHead = new android.widget.TextView(context);
|
||||||
shortxPickerHead.setText("ShortX 图标库(支持搜索,点击即回填)");
|
shortxPickerHead.setText("ShortX 图标库(分页模式,支持搜索 / 分类 / 点击即回填)");
|
||||||
shortxPickerHead.setTextColor(subTextColor);
|
shortxPickerHead.setTextColor(subTextColor);
|
||||||
shortxPickerHead.setTextSize(android.util.TypedValue.COMPLEX_UNIT_SP, 12);
|
shortxPickerHead.setTextSize(android.util.TypedValue.COMPLEX_UNIT_SP, 12);
|
||||||
shortxPickerHead.setPadding(self.dp(12), self.dp(10), self.dp(12), self.dp(6));
|
shortxPickerHead.setPadding(self.dp(12), self.dp(10), self.dp(12), self.dp(6));
|
||||||
@@ -833,6 +886,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
|
|
||||||
function goShortXPage(delta) {
|
function goShortXPage(delta) {
|
||||||
shortxPickerState.currentPage = Math.max(0, Number(shortxPickerState.currentPage || 0) + Number(delta || 0));
|
shortxPickerState.currentPage = Math.max(0, Number(shortxPickerState.currentPage || 0) + Number(delta || 0));
|
||||||
|
scrollShortXGridToTop();
|
||||||
renderShortXIconGrid();
|
renderShortXIconGrid();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -866,6 +920,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxPickerState.activeTab = def.key;
|
shortxPickerState.activeTab = def.key;
|
||||||
shortxPickerState.currentPage = 0;
|
shortxPickerState.currentPage = 0;
|
||||||
applyShortXTabStyles();
|
applyShortXTabStyles();
|
||||||
|
scrollShortXGridToTop();
|
||||||
renderShortXIconGrid();
|
renderShortXIconGrid();
|
||||||
});
|
});
|
||||||
tabBtn.setPadding(self.dp(10), self.dp(4), self.dp(10), self.dp(4));
|
tabBtn.setPadding(self.dp(10), self.dp(4), self.dp(10), self.dp(4));
|
||||||
@@ -883,6 +938,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
var shortxGridScrollLp = new android.widget.LinearLayout.LayoutParams(android.widget.LinearLayout.LayoutParams.MATCH_PARENT, self.dp(520));
|
var shortxGridScrollLp = new android.widget.LinearLayout.LayoutParams(android.widget.LinearLayout.LayoutParams.MATCH_PARENT, self.dp(520));
|
||||||
shortxGridScrollLp.setMargins(self.dp(8), 0, self.dp(8), self.dp(8));
|
shortxGridScrollLp.setMargins(self.dp(8), 0, self.dp(8), self.dp(8));
|
||||||
shortxPickerWrap.addView(shortxGridScroll, shortxGridScrollLp);
|
shortxPickerWrap.addView(shortxGridScroll, shortxGridScrollLp);
|
||||||
|
shortxPickerState.gridScroll = shortxGridScroll;
|
||||||
|
|
||||||
var shortxGrid = new android.widget.GridLayout(context);
|
var shortxGrid = new android.widget.GridLayout(context);
|
||||||
try { shortxGrid.setColumnCount(4); } catch(eGC0) {}
|
try { shortxGrid.setColumnCount(4); } catch(eGC0) {}
|
||||||
@@ -915,7 +971,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
try {
|
try {
|
||||||
if (!shortxPickerState.grid) return;
|
if (!shortxPickerState.grid) return;
|
||||||
shortxPickerState.grid.removeAllViews();
|
shortxPickerState.grid.removeAllViews();
|
||||||
try { shortxPickerState.grid.setColumnCount(4); } catch(eColSet) {}
|
try { shortxPickerState.grid.setColumnCount(Number(shortxPickerState.pageCols || 4)); } catch(eColSet) {}
|
||||||
var icons = self.getShortXIconCatalog();
|
var icons = self.getShortXIconCatalog();
|
||||||
shortxPickerState.iconList = icons;
|
shortxPickerState.iconList = icons;
|
||||||
var query = "";
|
var query = "";
|
||||||
@@ -935,7 +991,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
filtered.push(item);
|
filtered.push(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var pageSize = Number(shortxPickerState.pageSize || 20);
|
var pageSize = resolveShortXPickerPageSize();
|
||||||
if (pageSize < 1) pageSize = 20;
|
if (pageSize < 1) pageSize = 20;
|
||||||
var totalPages = filtered.length > 0 ? Math.ceil(filtered.length / pageSize) : 1;
|
var totalPages = filtered.length > 0 ? Math.ceil(filtered.length / pageSize) : 1;
|
||||||
if (shortxPickerState.currentPage >= totalPages) shortxPickerState.currentPage = totalPages - 1;
|
if (shortxPickerState.currentPage >= totalPages) shortxPickerState.currentPage = totalPages - 1;
|
||||||
@@ -948,13 +1004,13 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
var errMsg = self._shortxIconCatalogError ? String(self._shortxIconCatalogError) : "未知原因";
|
var errMsg = self._shortxIconCatalogError ? String(self._shortxIconCatalogError) : "未知原因";
|
||||||
shortxPickerState.statusTv.setText("ShortX 图标反射失败/为空:" + errMsg);
|
shortxPickerState.statusTv.setText("ShortX 图标反射失败/为空:" + errMsg);
|
||||||
} else if (!query) {
|
} else if (!query) {
|
||||||
shortxPickerState.statusTv.setText("分类[" + shortxPickerState.activeTab + "] 共 " + filtered.length + " 个,当前第 " + (shortxPickerState.currentPage + 1) + "/" + totalPages + " 页。");
|
shortxPickerState.statusTv.setText("分类[" + shortxPickerState.activeTab + "] 共 " + filtered.length + " 个,每页 " + pageSize + " 个(" + shortxPickerState.pageRows + " 行 × " + shortxPickerState.pageCols + " 列),当前第 " + (shortxPickerState.currentPage + 1) + "/" + totalPages + " 页。");
|
||||||
} else {
|
} else {
|
||||||
shortxPickerState.statusTv.setText("分类[" + shortxPickerState.activeTab + "] 搜索 [" + query + "] 命中 " + totalMatch + " 个,当前第 " + (shortxPickerState.currentPage + 1) + "/" + totalPages + " 页。");
|
shortxPickerState.statusTv.setText("分类[" + shortxPickerState.activeTab + "] 搜索 [" + query + "] 命中 " + totalMatch + " 个,每页 " + pageSize + " 个,当前第 " + (shortxPickerState.currentPage + 1) + "/" + totalPages + " 页。");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (shortxPickerState.pageInfoTv) {
|
if (shortxPickerState.pageInfoTv) {
|
||||||
shortxPickerState.pageInfoTv.setText((filtered.length > 0 ? (shortxPickerState.currentPage + 1) : 0) + " / " + totalPages + " · " + filtered.length + "项");
|
shortxPickerState.pageInfoTv.setText((filtered.length > 0 ? (shortxPickerState.currentPage + 1) : 0) + " / " + totalPages + " · " + filtered.length + "项 · 每页" + pageSize + "个");
|
||||||
}
|
}
|
||||||
try { shortxPickerState.prevBtn.setEnabled(shortxPickerState.currentPage > 0); } catch(ePrev) {}
|
try { shortxPickerState.prevBtn.setEnabled(shortxPickerState.currentPage > 0); } catch(ePrev) {}
|
||||||
try { shortxPickerState.nextBtn.setEnabled(shortxPickerState.currentPage < totalPages - 1); } catch(eNext) {}
|
try { shortxPickerState.nextBtn.setEnabled(shortxPickerState.currentPage < totalPages - 1); } catch(eNext) {}
|
||||||
@@ -968,9 +1024,9 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
cell.setGravity(android.view.Gravity.CENTER_HORIZONTAL);
|
cell.setGravity(android.view.Gravity.CENTER_HORIZONTAL);
|
||||||
cell.setPadding(self.dp(8), self.dp(8), self.dp(8), self.dp(8));
|
cell.setPadding(self.dp(8), self.dp(8), self.dp(8), self.dp(8));
|
||||||
var lp = new android.widget.GridLayout.LayoutParams();
|
var lp = new android.widget.GridLayout.LayoutParams();
|
||||||
lp.width = self.dp(76);
|
lp.width = self.dp(Number(shortxPickerState.cellWidthDp || 76));
|
||||||
lp.height = self.dp(92);
|
lp.height = self.dp(Number(shortxPickerState.cellHeightDp || 92));
|
||||||
lp.setMargins(self.dp(4), self.dp(4), self.dp(4), self.dp(4));
|
lp.setMargins(self.dp(Number(shortxPickerState.cellMarginDp || 4)), self.dp(Number(shortxPickerState.cellMarginDp || 4)), self.dp(Number(shortxPickerState.cellMarginDp || 4)), self.dp(Number(shortxPickerState.cellMarginDp || 4)));
|
||||||
cell.setLayoutParams(lp);
|
cell.setLayoutParams(lp);
|
||||||
var isSelected = selectedShort && selectedShort === String(entry.shortName);
|
var isSelected = selectedShort && selectedShort === String(entry.shortName);
|
||||||
cell.setBackground(self.ui.createRoundDrawable(self.withAlpha(isSelected ? C.primary : cardColor, isSelected ? 0.18 : 0.96), self.dp(12)));
|
cell.setBackground(self.ui.createRoundDrawable(self.withAlpha(isSelected ? C.primary : cardColor, isSelected ? 0.18 : 0.96), self.dp(12)));
|
||||||
@@ -999,9 +1055,8 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
self.touchActivity();
|
self.touchActivity();
|
||||||
try { inputShortXIcon.input.setText(String(entry.shortName)); } catch(eSetI) {}
|
try { inputShortXIcon.input.setText(String(entry.shortName)); } catch(eSetI) {}
|
||||||
updateShortXIconPreview();
|
updateShortXIconPreview();
|
||||||
shortxPickerState.expanded = false;
|
scrollShortXGridToTop();
|
||||||
if (shortxPickerState.pickerWrap) shortxPickerState.pickerWrap.setVisibility(android.view.View.GONE);
|
setShortXPickerExpanded(false, false);
|
||||||
try { if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText("图标库"); } catch(eSetT) {}
|
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
shortxPickerState.grid.addView(cell);
|
shortxPickerState.grid.addView(cell);
|
||||||
@@ -1016,6 +1071,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxSearchEt.addTextChangedListener(new JavaAdapter(android.text.TextWatcher, {
|
shortxSearchEt.addTextChangedListener(new JavaAdapter(android.text.TextWatcher, {
|
||||||
afterTextChanged: function(s) {
|
afterTextChanged: function(s) {
|
||||||
shortxPickerState.currentPage = 0;
|
shortxPickerState.currentPage = 0;
|
||||||
|
scrollShortXGridToTop();
|
||||||
renderShortXIconGrid();
|
renderShortXIconGrid();
|
||||||
},
|
},
|
||||||
beforeTextChanged: function(s, st, c, a) {},
|
beforeTextChanged: function(s, st, c, a) {},
|
||||||
@@ -1031,6 +1087,19 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
}));
|
}));
|
||||||
} catch(eTwIcon1) {}
|
} catch(eTwIcon1) {}
|
||||||
|
|
||||||
|
try {
|
||||||
|
shortxGridScroll.getViewTreeObserver().addOnGlobalLayoutListener(new android.view.ViewTreeObserver.OnGlobalLayoutListener({
|
||||||
|
onGlobalLayout: function() {
|
||||||
|
if (!shortxPickerState.expanded) return;
|
||||||
|
var oldSize = Number(shortxPickerState.pageSize || 0);
|
||||||
|
var newSize = resolveShortXPickerPageSize();
|
||||||
|
if (newSize > 0 && newSize !== oldSize) {
|
||||||
|
shortxPickerState.currentPage = 0;
|
||||||
|
renderShortXIconGrid();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
} catch(eGridLayoutWatch) {}
|
||||||
// # ShortX 图标颜色(默认跟随主题)
|
// # ShortX 图标颜色(默认跟随主题)
|
||||||
var defaultTint = targetBtn.iconTint || "";
|
var defaultTint = targetBtn.iconTint || "";
|
||||||
if (!defaultTint) {
|
if (!defaultTint) {
|
||||||
@@ -1063,7 +1132,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
shortxPickerWrap.setVisibility(android.view.View.GONE);
|
shortxPickerWrap.setVisibility(android.view.View.GONE);
|
||||||
inputShortXIconTint.view.setVisibility(android.view.View.GONE);
|
inputShortXIconTint.view.setVisibility(android.view.View.GONE);
|
||||||
shortxPickerState.expanded = false;
|
shortxPickerState.expanded = false;
|
||||||
try { if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText("图标库"); } catch(eBt0) {}
|
try { if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText(getShortXPickerClosedLabel()); } catch(eBt0) {}
|
||||||
// 清空另一种方式的值
|
// 清空另一种方式的值
|
||||||
inputShortXIcon.input.setText("");
|
inputShortXIcon.input.setText("");
|
||||||
inputShortXIconTint.input.setText("");
|
inputShortXIconTint.input.setText("");
|
||||||
@@ -1072,10 +1141,6 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
inputShortXIcon.view.setVisibility(android.view.View.VISIBLE);
|
inputShortXIcon.view.setVisibility(android.view.View.VISIBLE);
|
||||||
shortxQuickRow.setVisibility(android.view.View.VISIBLE);
|
shortxQuickRow.setVisibility(android.view.View.VISIBLE);
|
||||||
inputShortXIconTint.view.setVisibility(android.view.View.VISIBLE);
|
inputShortXIconTint.view.setVisibility(android.view.View.VISIBLE);
|
||||||
// 不再展开内联面板
|
|
||||||
shortxPickerState.expanded = false;
|
|
||||||
if (shortxPickerState.pickerWrap) shortxPickerWrap.setVisibility(android.view.View.GONE);
|
|
||||||
try { if (shortxPickerState.toggleBtn) shortxPickerState.toggleBtn.setText("选择图标"); } catch(eBt1) {}
|
|
||||||
// 清空另一种方式的值
|
// 清空另一种方式的值
|
||||||
inputIconPath.input.setText("");
|
inputIconPath.input.setText("");
|
||||||
// # ShortX 图标颜色默认跟随主题
|
// # ShortX 图标颜色默认跟随主题
|
||||||
@@ -1087,7 +1152,7 @@ FloatBallAppWM.prototype.buildButtonEditorPanelView = function() {
|
|||||||
}
|
}
|
||||||
} catch(e) {}
|
} catch(e) {}
|
||||||
updateShortXIconPreview();
|
updateShortXIconPreview();
|
||||||
renderShortXIconGrid();
|
setShortXPickerExpanded(true, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user