@switchbot/homebridge-switchbot 5.0.0-beta.153 → 5.0.0-beta.155
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.github/workflows/release.yml +63 -15
- package/.github/workflows/stale.yml +2 -4
- package/CHANGELOG.md +21 -29
- package/MIGRATION.md +6 -6
- package/README.md +5 -3
- package/dist/device-types.js +7 -7
- package/dist/device-types.js.map +1 -1
- package/dist/deviceFactory.d.ts +1 -1
- package/dist/deviceFactory.d.ts.map +1 -1
- package/dist/deviceFactory.js +20 -20
- package/dist/deviceFactory.js.map +1 -1
- package/dist/homebridge-ui/device-types.js +246 -0
- package/dist/homebridge-ui/device-types.js.map +1 -0
- package/dist/homebridge-ui/deviceCommandMapper.js +319 -0
- package/dist/homebridge-ui/deviceCommandMapper.js.map +1 -0
- package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -1
- package/dist/homebridge-ui/endpoints/devices.js +64 -1
- package/dist/homebridge-ui/endpoints/devices.js.map +1 -1
- package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -1
- package/dist/homebridge-ui/endpoints/discovery.js +5 -1
- package/dist/homebridge-ui/endpoints/discovery.js.map +1 -1
- package/dist/homebridge-ui/errors.js +32 -0
- package/dist/homebridge-ui/errors.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/config.js +90 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/config.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js +144 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js +219 -0
- package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/server.js +11 -0
- package/dist/homebridge-ui/homebridge-ui/server.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js +108 -0
- package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js +111 -0
- package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js.map +1 -0
- package/dist/homebridge-ui/homebridge-ui/utils/logger.js +17 -0
- package/dist/homebridge-ui/homebridge-ui/utils/logger.js.map +1 -0
- package/dist/homebridge-ui/public/js/api.d.ts.map +1 -1
- package/dist/homebridge-ui/public/js/api.js +24 -11
- package/dist/homebridge-ui/public/js/api.js.map +1 -1
- package/dist/homebridge-ui/public/js/api.ts +24 -12
- package/dist/homebridge-ui/public/js/app.js +117 -267
- package/dist/homebridge-ui/public/js/app.js.map +3 -3
- package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -1
- package/dist/homebridge-ui/public/js/devices.js +2 -0
- package/dist/homebridge-ui/public/js/devices.js.map +1 -1
- package/dist/homebridge-ui/public/js/devices.ts +2 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts +5 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -1
- package/dist/homebridge-ui/public/js/discovery.js +79 -245
- package/dist/homebridge-ui/public/js/discovery.js.map +1 -1
- package/dist/homebridge-ui/public/js/discovery.ts +88 -247
- package/dist/homebridge-ui/public/js/render.d.ts.map +1 -1
- package/dist/homebridge-ui/public/js/render.js +2 -1
- package/dist/homebridge-ui/public/js/render.js.map +1 -1
- package/dist/homebridge-ui/public/js/render.ts +2 -1
- package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -1
- package/dist/homebridge-ui/public/js/toast.js +15 -6
- package/dist/homebridge-ui/public/js/toast.js.map +1 -1
- package/dist/homebridge-ui/public/js/toast.ts +14 -7
- package/dist/homebridge-ui/settings.js +8 -0
- package/dist/homebridge-ui/settings.js.map +1 -0
- package/dist/homebridge-ui/switchbotClient.js +247 -0
- package/dist/homebridge-ui/switchbotClient.js.map +1 -0
- package/dist/homebridge-ui/utils/config-parser.d.ts +4 -0
- package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -1
- package/dist/homebridge-ui/utils/config-parser.js +21 -0
- package/dist/homebridge-ui/utils/config-parser.js.map +1 -1
- package/dist/switchbotClient.d.ts +7 -1
- package/dist/switchbotClient.d.ts.map +1 -1
- package/dist/switchbotClient.js +82 -10
- package/dist/switchbotClient.js.map +1 -1
- package/docs/assets/main.js +1 -1
- package/docs/index.html +10 -4
- package/docs/variables/default.html +1 -1
- package/eslint.config.js +9 -10
- package/package.json +26 -24
- package/src/device-types.js +246 -0
- package/src/device-types.js.map +1 -0
- package/src/device-types.ts +7 -7
- package/src/deviceCommandMapper.js +319 -0
- package/src/deviceCommandMapper.js.map +1 -0
- package/src/deviceFactory.ts +22 -21
- package/src/errors.js +32 -0
- package/src/errors.js.map +1 -0
- package/src/homebridge-ui/endpoints/devices.ts +66 -1
- package/src/homebridge-ui/endpoints/discovery.ts +5 -1
- package/src/homebridge-ui/public/js/api.ts +24 -12
- package/src/homebridge-ui/public/js/devices.ts +2 -0
- package/src/homebridge-ui/public/js/discovery.ts +88 -247
- package/src/homebridge-ui/public/js/render.ts +2 -1
- package/src/homebridge-ui/public/js/toast.ts +14 -7
- package/src/homebridge-ui/utils/config-parser.ts +17 -0
- package/src/settings.js +8 -0
- package/src/settings.js.map +1 -0
- package/src/switchbotClient.js +247 -0
- package/src/switchbotClient.js.map +1 -0
- package/src/switchbotClient.ts +95 -10
- package/test/client/switchbotClient.spec.ts +42 -1
- package/test/e2e/run-e2e.spec.ts +1 -0
- package/tsconfig.ui.json +11 -0
- package/.github/workflows/beta-release.yml +0 -52
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"devices.d.ts","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/devices.ts"],"names":[],"mappings":"AAMA,wBAAsB,iBAAiB,CAAC,MAAM,EAAE,GAAG,EAAE,OAAO,GAAE;IAAE,OAAO,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAO,GAAG,OAAO,CAAC;IAAE,KAAK,EAAE,OAAO,CAAA;CAAE,CAAC,
|
|
1
|
+
{"version":3,"file":"devices.d.ts","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/devices.ts"],"names":[],"mappings":"AAMA,wBAAsB,iBAAiB,CAAC,MAAM,EAAE,GAAG,EAAE,OAAO,GAAE;IAAE,OAAO,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAO,GAAG,OAAO,CAAC;IAAE,KAAK,EAAE,OAAO,CAAA;CAAE,CAAC,CAkF3I;AAED,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,IAAI,CAAC,CAUzD;AAED,wBAAsB,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC,CAG3D"}
|
|
@@ -60,6 +60,8 @@ export async function addDeviceToConfig(device, options = {}) {
|
|
|
60
60
|
}
|
|
61
61
|
}
|
|
62
62
|
if (refresh) {
|
|
63
|
+
// Force reload of config from disk after add
|
|
64
|
+
await syncParentPluginConfigFromDisk(true);
|
|
63
65
|
await loadConfiguredDevices();
|
|
64
66
|
}
|
|
65
67
|
return { added: !alreadyExists };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"devices.js","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/devices.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,8BAA8B,EAAE,MAAM,UAAU,CAAA;AAClF,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAA;AACnC,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,YAAY,CAAA;AACnD,OAAO,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAA;AAC9C,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,YAAY,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAE9E,MAAM,CAAC,KAAK,UAAU,iBAAiB,CAAC,MAAW,EAAE,UAAuD,EAAE;IAC5G,MAAM,EAAE,OAAO,GAAG,IAAI,EAAE,UAAU,GAAG,IAAI,EAAE,GAAG,OAAO,CAAA;IACrD,IAAI,CAAC;QACH,MAAM,EAAE,sBAAsB,EAAE,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,CAAA;QAE9D,MAAM,YAAY,GAAQ,MAAM,sBAAsB,CAAC,MAAM,CAAC,CAAA;QAC9D,IAAI,CAAC,YAAY,IAAI,OAAO,YAAY,KAAK,QAAQ,IAAI,CAAC,YAAY,CAAC,gBAAgB,IAAI,CAAC,YAAY,CAAC,gBAAgB,EAAE,CAAC;YAC1H,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAA;QACzB,CAAC;QAED,UAAU,EAAE,CAAA;QACZ,KAAK,CAAC,IAAI,CAAC,0BAA0B,EAAE,MAAM,CAAC,CAAA;QAC9C,2CAA2C;QAC3C,IAAI,QAAQ,GAAG,YAAY,CAAC,gBAAgB,CAAA;QAC5C,IAAI,CAAC,QAAQ,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;YAC1C,QAAQ,GAAG,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,EAAE,CAAA;YACnC,KAAK,CAAC,IAAI,CAAC,6BAA6B,YAAY,CAAC,gBAAgB,wBAAwB,QAAQ,GAAG,CAAC,CAAA;QAC3G,CAAC;QACD,MAAM,IAAI,GAAG,MAAM,SAAS,CAAC,MAAM,CAAC,EAAE,EAAE,QAAQ,EAAE,YAAY,CAAC,gBAAgB,EAAE;YAC/E,OAAO,EAAE,YAAY,CAAC,OAAO;YAC7B,KAAK,EAAE,MAAM,CAAC,KAAK;YACnB,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,aAAa,EAAE,YAAY,CAAC,aAAa;YACzC,KAAK,EAAE,YAAY,CAAC,KAAK;SAC1B,CAAC,CAAA;QACF,KAAK,CAAC,IAAI,CAAC,sBAAsB,EAAE,IAAI,CAAC,CAAA;QAExC,MAAM,aAAa,GAAG,CAAC,CAAC,IAAI,EAAE,aAAa,CAAA;QAC3C,MAAM,OAAO,GAAG,IAAI,EAAE,OAAO;eACxB,CAAC,aAAa;gBACf,CAAC,CAAC,WAAW,YAAY,CAAC,gBAAgB,qBAAqB;gBAC/D,CAAC,CAAC,WAAW,YAAY,CAAC,gBAAgB,uBAAuB,CAAC,CAAA;QAEtE,IAAI,aAAa,EAAE,CAAC;YAClB,SAAS,CAAC,OAAO,CAAC,CAAA;QACpB,CAAC;aAAM,CAAC;YACN,YAAY,CAAC,OAAO,CAAC,CAAA;QACvB,CAAC;QAED,IAAI,UAAU,EAAE,CAAC;YACf,MAAM,MAAM,GAAG,QAAQ,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAA;YACxD,IAAI,MAAM,EAAE,CAAC;gBACX,MAAM,CAAC,WAAW,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,OAAO,CAAA;gBAC5D,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,OAAO,CAAC,CAAA;gBAChC,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,aAAa,CAAC,CAAA;gBAEnC,6EAA6E;gBAC7E,IAAI,CAAC,aAAa,EAAE,CAAC;oBACnB,MAAM,MAAM,GAAG,MAAM,8BAA8B,CAAC,IAAI,CAAC,CAAA;oBACzD,MAAM,CAAC,WAAW,IAAI,MAAM;wBAC1B,CAAC,CAAC,gCAAgC;wBAClC,CAAC,CAAC,qEAAqE,CAAA;oBAEzE,IAAI,MAAM,EAAE,CAAC;wBACX,YAAY,CAAC,8CAA8C,CAAC,CAAA;oBAC9D,CAAC;yBAAM,CAAC;wBACN,YAAY,CAAC,kEAAkE,CAAC,CAAA;oBAClF,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,OAAO,EAAE,CAAC;YACZ,MAAM,qBAAqB,EAAE,CAAA;QAC/B,CAAC;QAED,OAAO,EAAE,KAAK,EAAE,CAAC,aAAa,EAAE,CAAA;IAClC,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,MAAM,GAAG,GAAG,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAA;QACtD,KAAK,CAAC,KAAK,CAAC,mBAAmB,EAAE,GAAG,CAAC,CAAA;QACrC,MAAM,MAAM,GAAG,QAAQ,CAAC,cAAc,CAAC,gBAAgB,CAAC,CAAA;QACxD,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,CAAC,WAAW,GAAG,YAAY,GAAG,EAAE,CAAA;YACtC,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;QAC/B,CAAC;QACD,UAAU,CAAC,GAAG,CAAC,CAAA;QACf,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAA;IACzB,CAAC;YAAS,CAAC;QACT,UAAU,EAAE,CAAA;IACd,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,mBAAmB;IACvC,MAAM,YAAY,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,CAAA;IAC5D,IAAI,CAAC,YAAY,EAAE,CAAC;QAClB,OAAM;IACR,CAAC;IAED,YAAY,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE;QAChD,MAAM,EAAE,0BAA0B,EAAE,GAAG,MAAM,MAAM,CAAC,qBAAqB,CAAC,CAAA;QAC1E,MAAM,0BAA0B,EAAE,CAAA;IACpC,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,qBAAqB;IACzC,MAAM,IAAI,GAAG,MAAM,YAAY,EAAE,CAAA;IACjC,gBAAgB,CAAC,IAAI,CAAC,CAAA;AACxB,CAAC"}
|
|
1
|
+
{"version":3,"file":"devices.js","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/devices.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,3 +1,8 @@
|
|
|
1
|
+
declare global {
|
|
2
|
+
interface Window {
|
|
3
|
+
_discoverySelectedIds: Set<string>;
|
|
4
|
+
}
|
|
5
|
+
}
|
|
1
6
|
export declare function initializeDiscoverySettings(): Promise<void>;
|
|
2
7
|
export declare function discoverDevices(): Promise<void>;
|
|
3
8
|
export declare function addDeviceToConfig(device: any): Promise<void>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"discovery.d.ts","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/discovery.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"discovery.d.ts","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/discovery.ts"],"names":[],"mappings":"AAeA,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,MAAM;QACd,qBAAqB,EAAE,GAAG,CAAC,MAAM,CAAC,CAAA;KACnC;CACF;AAmOD,wBAAsB,2BAA2B,IAAI,OAAO,CAAC,IAAI,CAAC,CA8FjE;AAsDD,wBAAsB,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC,CAiuBrD;AAwMD,wBAAsB,iBAAiB,CAAC,MAAM,EAAE,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC,CAGlE"}
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
// Extend the Window interface to include _discoverySelectedIds for type safety
|
|
1
2
|
// Batch enable/disable helper (true module scope for UI access)
|
|
2
3
|
import { addDevicesInBulk, discoverDevices as apiDiscoverDevices, fetchBluetoothStatus,
|
|
3
4
|
// fetchDevices, // removed unused import
|
|
@@ -7,45 +8,6 @@ import { uiLog } from './logger.js';
|
|
|
7
8
|
import { hideBusyUi, showBusyUi } from './modal.js';
|
|
8
9
|
import { getDiscoveryPreferences, renderDiscoveredDevices, setDiscoveryPreferences } from './render.js';
|
|
9
10
|
import { toastError, toastInfo, toastSuccess, toastWarning } from './toast.js';
|
|
10
|
-
async function batchSetDeviceEnabled(selectedIds, enabled) {
|
|
11
|
-
// Fetch current config
|
|
12
|
-
const resp = await homebridge.request('/platform-config', {});
|
|
13
|
-
if (!resp || resp.success === false || !resp.data) {
|
|
14
|
-
throw new Error('Failed to load config');
|
|
15
|
-
}
|
|
16
|
-
const config = resp.data;
|
|
17
|
-
// Homebridge config may be an array of platforms, find SwitchBot
|
|
18
|
-
const configArr = Array.isArray(config) ? config : [config];
|
|
19
|
-
const platformIdx = configArr.findIndex((c) => (c.platform || c.name || '').toLowerCase().includes('switchbot'));
|
|
20
|
-
if (platformIdx === -1) {
|
|
21
|
-
throw new Error('SwitchBot platform config not found');
|
|
22
|
-
}
|
|
23
|
-
const platformConfig = configArr[platformIdx];
|
|
24
|
-
if (!Array.isArray(platformConfig.devices)) {
|
|
25
|
-
throw new TypeError('No devices array in config');
|
|
26
|
-
}
|
|
27
|
-
let changed = false;
|
|
28
|
-
for (const dev of platformConfig.devices) {
|
|
29
|
-
const id = String(dev.deviceId || dev.id || '').trim().toLowerCase();
|
|
30
|
-
if (selectedIds.has(id)) {
|
|
31
|
-
if (dev.enabled !== enabled) {
|
|
32
|
-
dev.enabled = enabled;
|
|
33
|
-
changed = true;
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
if (changed) {
|
|
38
|
-
if (typeof homebridge.updatePluginConfig === 'function') {
|
|
39
|
-
await homebridge.updatePluginConfig(configArr);
|
|
40
|
-
}
|
|
41
|
-
else {
|
|
42
|
-
throw new TypeError('homebridge.updatePluginConfig is not available');
|
|
43
|
-
}
|
|
44
|
-
if (typeof homebridge.savePluginConfig === 'function') {
|
|
45
|
-
await homebridge.savePluginConfig();
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
11
|
function normalizeId(value) {
|
|
50
12
|
return String(value ?? '').trim().toLowerCase();
|
|
51
13
|
}
|
|
@@ -325,10 +287,10 @@ function getDiscoveryGroupByPreference() {
|
|
|
325
287
|
if (stored === 'hub' || stored === 'type') {
|
|
326
288
|
return stored;
|
|
327
289
|
}
|
|
328
|
-
return '
|
|
290
|
+
return 'type'; // Default to Device Type grouping
|
|
329
291
|
}
|
|
330
292
|
catch (_e) {
|
|
331
|
-
return '
|
|
293
|
+
return 'type';
|
|
332
294
|
}
|
|
333
295
|
}
|
|
334
296
|
function setDiscoveryGroupByPreference(groupBy) {
|
|
@@ -439,25 +401,23 @@ export async function discoverDevices() {
|
|
|
439
401
|
const preferences = getDiscoveryPreferences();
|
|
440
402
|
let groupBy = getDiscoveryGroupByPreference();
|
|
441
403
|
let hideAdded = getDiscoveryHideAddedPreference();
|
|
442
|
-
let controlsInitialized = false;
|
|
443
404
|
// Use persistent selection state across renders
|
|
444
405
|
if (!window._discoverySelectedIds) {
|
|
445
406
|
window._discoverySelectedIds = new Set();
|
|
446
407
|
}
|
|
447
408
|
const selectedIds = window._discoverySelectedIds;
|
|
409
|
+
let controlsInitialized = false;
|
|
448
410
|
// --- Real-time RSSI polling loop ---
|
|
449
411
|
// (Moved inside main try block after bleSettings is defined)
|
|
450
412
|
// Batch enable/disable helper (moved to module scope for UI access)
|
|
451
413
|
async function batchSetDeviceEnabled(selectedIds, enabled) {
|
|
452
414
|
// Fetch current config
|
|
453
|
-
|
|
454
|
-
if (
|
|
455
|
-
throw new
|
|
415
|
+
// Fetch current config using Homebridge UI API
|
|
416
|
+
if (typeof homebridge.getPluginConfig !== 'function') {
|
|
417
|
+
throw new TypeError('homebridge.getPluginConfig is not available');
|
|
456
418
|
}
|
|
457
|
-
const
|
|
458
|
-
|
|
459
|
-
const configArr = Array.isArray(config) ? config : [config];
|
|
460
|
-
const platformIdx = configArr.findIndex((c) => (c.platform || c.name || '').toLowerCase().includes('switchbot'));
|
|
419
|
+
const configArr = await homebridge.getPluginConfig();
|
|
420
|
+
const platformIdx = Array.isArray(configArr) ? configArr.findIndex(c => (c.platform || c.name || '').toLowerCase().includes('switchbot')) : -1;
|
|
461
421
|
if (platformIdx === -1) {
|
|
462
422
|
throw new Error('SwitchBot platform config not found');
|
|
463
423
|
}
|
|
@@ -488,14 +448,53 @@ export async function discoverDevices() {
|
|
|
488
448
|
}
|
|
489
449
|
}
|
|
490
450
|
const ensureDiscoveryControls = async () => {
|
|
451
|
+
// --- Select All / Deselect All controls ---
|
|
452
|
+
const selectAllBtn = document.createElement('button');
|
|
453
|
+
selectAllBtn.textContent = 'Select All';
|
|
454
|
+
selectAllBtn.style.fontSize = '13px';
|
|
455
|
+
selectAllBtn.style.padding = '6px 18px';
|
|
456
|
+
selectAllBtn.style.borderRadius = '6px';
|
|
457
|
+
selectAllBtn.style.background = '#f3f4f6';
|
|
458
|
+
selectAllBtn.style.color = '#1d4ed8';
|
|
459
|
+
selectAllBtn.style.border = '1px solid #d1d5db';
|
|
460
|
+
selectAllBtn.style.cursor = 'pointer';
|
|
461
|
+
selectAllBtn.style.marginRight = '8px';
|
|
462
|
+
selectAllBtn.onclick = () => {
|
|
463
|
+
// Add all visible device IDs to selectedIds
|
|
464
|
+
for (const d of discoveredDevices) {
|
|
465
|
+
selectedIds.add(normalizeId(d.id));
|
|
466
|
+
}
|
|
467
|
+
window.dispatchEvent(new Event('discovery-selection-changed'));
|
|
468
|
+
void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
469
|
+
};
|
|
470
|
+
const deselectAllBtn = document.createElement('button');
|
|
471
|
+
deselectAllBtn.textContent = 'Deselect All';
|
|
472
|
+
deselectAllBtn.style.fontSize = '13px';
|
|
473
|
+
deselectAllBtn.style.padding = '6px 18px';
|
|
474
|
+
deselectAllBtn.style.borderRadius = '6px';
|
|
475
|
+
deselectAllBtn.style.background = '#f3f4f6';
|
|
476
|
+
deselectAllBtn.style.color = '#ef4444';
|
|
477
|
+
deselectAllBtn.style.border = '1px solid #d1d5db';
|
|
478
|
+
deselectAllBtn.style.cursor = 'pointer';
|
|
479
|
+
deselectAllBtn.onclick = () => {
|
|
480
|
+
// Remove all visible device IDs from selectedIds
|
|
481
|
+
for (const d of discoveredDevices) {
|
|
482
|
+
selectedIds.delete(normalizeId(d.id));
|
|
483
|
+
}
|
|
484
|
+
window.dispatchEvent(new Event('discovery-selection-changed'));
|
|
485
|
+
void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
486
|
+
};
|
|
487
|
+
// Insert select/deselect all controls above the action buttons
|
|
488
|
+
const selectControlsRow = document.createElement('div');
|
|
489
|
+
selectControlsRow.style.display = 'flex';
|
|
490
|
+
selectControlsRow.style.gap = '10px';
|
|
491
|
+
selectControlsRow.style.margin = '0 0 10px 0';
|
|
492
|
+
selectControlsRow.appendChild(selectAllBtn);
|
|
493
|
+
selectControlsRow.appendChild(deselectAllBtn);
|
|
491
494
|
if (controlsInitialized) {
|
|
492
495
|
return;
|
|
493
496
|
}
|
|
494
|
-
// Always use persistent selectedIds
|
|
495
|
-
if (!window._discoverySelectedIds) {
|
|
496
|
-
window._discoverySelectedIds = new Set();
|
|
497
|
-
}
|
|
498
|
-
const selectedIds = window._discoverySelectedIds;
|
|
497
|
+
// Always use persistent selectedIds (already defined in outer scope)
|
|
499
498
|
const controlsDiv = document.createElement('div');
|
|
500
499
|
controlsDiv.style.cssText = 'margin-bottom: 12px; display: flex; gap: 12px; flex-wrap: wrap; align-items: center;';
|
|
501
500
|
const filterLabel = document.createElement('label');
|
|
@@ -564,16 +563,28 @@ export async function discoverDevices() {
|
|
|
564
563
|
setDiscoveryPreferences(preferences);
|
|
565
564
|
void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
566
565
|
};
|
|
567
|
-
const groupLabel = document.createElement('label');
|
|
568
|
-
groupLabel.style.fontSize = '12px';
|
|
569
|
-
groupLabel.style.fontWeight = '500';
|
|
570
|
-
groupLabel.style.marginLeft = '8px';
|
|
571
|
-
groupLabel.textContent = 'Group:';
|
|
572
566
|
const groupSelect = document.createElement('select');
|
|
573
567
|
groupSelect.style.fontSize = '11px';
|
|
574
568
|
groupSelect.style.padding = '4px 8px';
|
|
575
569
|
groupSelect.style.borderRadius = '3px';
|
|
576
|
-
|
|
570
|
+
// Set default value to 'type' if no stored preference
|
|
571
|
+
if (!localStorage.getItem(DISCOVERY_GROUP_BY_KEY)) {
|
|
572
|
+
groupSelect.value = 'type';
|
|
573
|
+
}
|
|
574
|
+
else {
|
|
575
|
+
groupSelect.value = groupBy;
|
|
576
|
+
}
|
|
577
|
+
const groupLabel = document.createElement('label');
|
|
578
|
+
groupLabel.style.fontSize = '12px';
|
|
579
|
+
groupLabel.style.fontWeight = '500';
|
|
580
|
+
groupLabel.style.marginLeft = '8px';
|
|
581
|
+
// Set label text to match selected group
|
|
582
|
+
const groupLabelTextMap = {
|
|
583
|
+
connection: 'Connection',
|
|
584
|
+
hub: 'Hub',
|
|
585
|
+
type: 'Device Type',
|
|
586
|
+
};
|
|
587
|
+
groupLabel.textContent = `Group: ${groupLabelTextMap[groupSelect.value] || 'Connection'}`;
|
|
577
588
|
const groupOptions = [
|
|
578
589
|
{ label: 'Connection', value: 'connection' },
|
|
579
590
|
{ label: 'Hub', value: 'hub' },
|
|
@@ -588,6 +599,7 @@ export async function discoverDevices() {
|
|
|
588
599
|
groupSelect.onchange = () => {
|
|
589
600
|
groupBy = groupSelect.value;
|
|
590
601
|
setDiscoveryGroupByPreference(groupBy);
|
|
602
|
+
groupLabel.textContent = `Group: ${groupLabelTextMap[groupSelect.value] || 'Connection'}`;
|
|
591
603
|
void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
592
604
|
};
|
|
593
605
|
const hideAddedLabel = document.createElement('label');
|
|
@@ -774,6 +786,7 @@ export async function discoverDevices() {
|
|
|
774
786
|
topActionRow.appendChild(disableSelectedBtn);
|
|
775
787
|
// Clear list and append controls in correct order
|
|
776
788
|
list.innerHTML = '';
|
|
789
|
+
list.appendChild(selectControlsRow);
|
|
777
790
|
list.appendChild(topActionRow);
|
|
778
791
|
list.appendChild(controlsDiv);
|
|
779
792
|
let deviceListContainer = document.getElementById('discoveredDevices');
|
|
@@ -1150,25 +1163,12 @@ async function updateDiscoveryView(allDevices, preferences, groupBy, hideAdded,
|
|
|
1150
1163
|
toastError('Discovery UI error: device list container missing. Please reload the page.');
|
|
1151
1164
|
}
|
|
1152
1165
|
}
|
|
1153
|
-
//
|
|
1154
|
-
// Remove any duplicate 'Add Selected' buttons from legacy UI
|
|
1155
|
-
// Robust NodeList iteration: classic for loop to avoid formatter and transpiler issues
|
|
1156
|
-
const legacyAddSelectedBtns = [];
|
|
1157
|
-
const allBtns = document.querySelectorAll('button');
|
|
1158
|
-
for (let i = 0; i < allBtns.length; i++) {
|
|
1159
|
-
const btn = allBtns[i];
|
|
1160
|
-
if (btn.textContent && btn.textContent.trim() === 'Add Selected') {
|
|
1161
|
-
legacyAddSelectedBtns.push(btn);
|
|
1162
|
-
}
|
|
1163
|
-
}
|
|
1164
|
-
for (let i = 0; i < legacyAddSelectedBtns.length; i++) {
|
|
1165
|
-
legacyAddSelectedBtns[i].remove();
|
|
1166
|
-
}
|
|
1167
|
-
// Listen for selection changes to update batch button states
|
|
1166
|
+
// Only update the enabled/disabled state of batch action buttons (created in ensureDiscoveryControls)
|
|
1168
1167
|
function updateBatchButtonStates() {
|
|
1168
|
+
// These buttons are created in ensureDiscoveryControls and should have unique IDs
|
|
1169
1169
|
const addSelectedBtn = document.getElementById('addSelectedBtn');
|
|
1170
|
-
const enableSelectedBtn = document.
|
|
1171
|
-
const disableSelectedBtn = document.
|
|
1170
|
+
const enableSelectedBtn = document.getElementById('enableSelectedBtn');
|
|
1171
|
+
const disableSelectedBtn = document.getElementById('disableSelectedBtn');
|
|
1172
1172
|
const hasSelection = selectedIds.size > 0;
|
|
1173
1173
|
if (addSelectedBtn) {
|
|
1174
1174
|
addSelectedBtn.disabled = !hasSelection;
|
|
@@ -1182,174 +1182,8 @@ async function updateDiscoveryView(allDevices, preferences, groupBy, hideAdded,
|
|
|
1182
1182
|
}
|
|
1183
1183
|
window.removeEventListener('discovery-selection-changed', updateBatchButtonStates);
|
|
1184
1184
|
window.addEventListener('discovery-selection-changed', updateBatchButtonStates);
|
|
1185
|
-
//
|
|
1186
|
-
|
|
1187
|
-
if (!batchControls) {
|
|
1188
|
-
batchControls = document.createElement('div');
|
|
1189
|
-
batchControls.id = 'batchImportControls';
|
|
1190
|
-
batchControls.style.display = 'flex';
|
|
1191
|
-
batchControls.style.flexWrap = 'wrap';
|
|
1192
|
-
batchControls.style.alignItems = 'center';
|
|
1193
|
-
batchControls.style.gap = '18px';
|
|
1194
|
-
batchControls.style.margin = '8px 0 18px 0';
|
|
1195
|
-
batchControls.style.width = '100%';
|
|
1196
|
-
// Button container for horizontal alignment
|
|
1197
|
-
const buttonRow = document.createElement('div');
|
|
1198
|
-
buttonRow.style.display = 'flex';
|
|
1199
|
-
buttonRow.style.flexWrap = 'nowrap';
|
|
1200
|
-
buttonRow.style.alignItems = 'center';
|
|
1201
|
-
buttonRow.style.justifyContent = 'center';
|
|
1202
|
-
buttonRow.style.gap = '16px';
|
|
1203
|
-
buttonRow.style.width = '100%';
|
|
1204
|
-
// Add Selected to Config button
|
|
1205
|
-
const addSelectedBtn = document.createElement('button');
|
|
1206
|
-
addSelectedBtn.id = 'addSelectedBtn';
|
|
1207
|
-
addSelectedBtn.textContent = 'Add Selected to Config';
|
|
1208
|
-
addSelectedBtn.disabled = selectedIds.size === 0;
|
|
1209
|
-
addSelectedBtn.style.fontWeight = '600';
|
|
1210
|
-
// Shared style for all red action buttons
|
|
1211
|
-
const redButtonStyle = {
|
|
1212
|
-
width: '220px',
|
|
1213
|
-
padding: '12px 0',
|
|
1214
|
-
fontSize: '17px',
|
|
1215
|
-
marginBottom: '0',
|
|
1216
|
-
boxShadow: '0 2px 8px 0 rgba(220,38,38,0.10)',
|
|
1217
|
-
borderRadius: '8px',
|
|
1218
|
-
};
|
|
1219
|
-
Object.assign(addSelectedBtn.style, redButtonStyle);
|
|
1220
|
-
addSelectedBtn.style.background = '#ef4444';
|
|
1221
|
-
addSelectedBtn.style.color = 'white';
|
|
1222
|
-
addSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
|
|
1223
|
-
addSelectedBtn.onmouseenter = function () {
|
|
1224
|
-
addSelectedBtn.style.background = '#dc2626';
|
|
1225
|
-
};
|
|
1226
|
-
addSelectedBtn.onmouseleave = function () {
|
|
1227
|
-
addSelectedBtn.style.background = '#ef4444';
|
|
1228
|
-
};
|
|
1229
|
-
addSelectedBtn.onclick = async () => {
|
|
1230
|
-
if (!selectedIds.size) {
|
|
1231
|
-
return;
|
|
1232
|
-
}
|
|
1233
|
-
addSelectedBtn.disabled = true;
|
|
1234
|
-
addSelectedBtn.textContent = 'Adding...';
|
|
1235
|
-
try {
|
|
1236
|
-
const selectedDevices = visibleDevices.filter(d => selectedIds.has(normalizeId(d.id)));
|
|
1237
|
-
const bulkResult = await addDevicesInBulk(selectedDevices.map(d => ({
|
|
1238
|
-
deviceId: d.id,
|
|
1239
|
-
name: d.name,
|
|
1240
|
-
type: d.type,
|
|
1241
|
-
rssi: d.rssi,
|
|
1242
|
-
address: d.address,
|
|
1243
|
-
model: d.model,
|
|
1244
|
-
})));
|
|
1245
|
-
uiLog.info('Batch add response:', bulkResult);
|
|
1246
|
-
if (!bulkResult || bulkResult.success === false) {
|
|
1247
|
-
throw new Error(bulkResult?.data?.message || 'Batch add failed');
|
|
1248
|
-
}
|
|
1249
|
-
const addedCount = bulkResult?.addedCount ?? bulkResult?.data?.addedCount ?? 0;
|
|
1250
|
-
const skippedCount = bulkResult?.skippedCount ?? bulkResult?.data?.skippedCount ?? 0;
|
|
1251
|
-
toastSuccess(`Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`);
|
|
1252
|
-
await loadConfiguredDevices();
|
|
1253
|
-
selectedIds.clear();
|
|
1254
|
-
addSelectedBtn.disabled = true;
|
|
1255
|
-
addSelectedBtn.textContent = 'Add Selected to Config';
|
|
1256
|
-
await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
1257
|
-
}
|
|
1258
|
-
catch (e) {
|
|
1259
|
-
uiLog.error('Batch add error:', e);
|
|
1260
|
-
toastError(e instanceof Error ? e.message : 'Failed to add devices');
|
|
1261
|
-
addSelectedBtn.disabled = false;
|
|
1262
|
-
addSelectedBtn.textContent = 'Add Selected to Config';
|
|
1263
|
-
}
|
|
1264
|
-
};
|
|
1265
|
-
// Enable Selected button
|
|
1266
|
-
const enableSelectedBtn = document.createElement('button');
|
|
1267
|
-
enableSelectedBtn.textContent = 'Enable Selected';
|
|
1268
|
-
enableSelectedBtn.style.fontWeight = '600';
|
|
1269
|
-
Object.assign(enableSelectedBtn.style, redButtonStyle);
|
|
1270
|
-
enableSelectedBtn.style.background = '#ef4444';
|
|
1271
|
-
enableSelectedBtn.style.color = 'white';
|
|
1272
|
-
enableSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
|
|
1273
|
-
enableSelectedBtn.onmouseenter = function () {
|
|
1274
|
-
enableSelectedBtn.style.background = '#dc2626';
|
|
1275
|
-
};
|
|
1276
|
-
enableSelectedBtn.onmouseleave = function () {
|
|
1277
|
-
enableSelectedBtn.style.background = '#ef4444';
|
|
1278
|
-
};
|
|
1279
|
-
enableSelectedBtn.disabled = selectedIds.size === 0;
|
|
1280
|
-
enableSelectedBtn.onclick = async () => {
|
|
1281
|
-
if (!selectedIds.size) {
|
|
1282
|
-
return;
|
|
1283
|
-
}
|
|
1284
|
-
enableSelectedBtn.disabled = true;
|
|
1285
|
-
try {
|
|
1286
|
-
showBusyUi();
|
|
1287
|
-
await batchSetDeviceEnabled(selectedIds, true);
|
|
1288
|
-
toastSuccess('Selected devices enabled');
|
|
1289
|
-
await loadConfiguredDevices();
|
|
1290
|
-
await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
1291
|
-
}
|
|
1292
|
-
catch (e) {
|
|
1293
|
-
uiLog.error('Batch enable error:', e);
|
|
1294
|
-
toastError(e instanceof Error ? e.message : 'Failed to enable devices');
|
|
1295
|
-
}
|
|
1296
|
-
finally {
|
|
1297
|
-
hideBusyUi();
|
|
1298
|
-
enableSelectedBtn.disabled = false;
|
|
1299
|
-
}
|
|
1300
|
-
};
|
|
1301
|
-
// Disable Selected button
|
|
1302
|
-
const disableSelectedBtn = document.createElement('button');
|
|
1303
|
-
disableSelectedBtn.textContent = 'Disable Selected';
|
|
1304
|
-
disableSelectedBtn.style.fontWeight = '600';
|
|
1305
|
-
Object.assign(disableSelectedBtn.style, redButtonStyle);
|
|
1306
|
-
disableSelectedBtn.style.background = '#ef4444';
|
|
1307
|
-
disableSelectedBtn.style.color = 'white';
|
|
1308
|
-
disableSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
|
|
1309
|
-
disableSelectedBtn.onmouseenter = function () {
|
|
1310
|
-
disableSelectedBtn.style.background = '#dc2626';
|
|
1311
|
-
};
|
|
1312
|
-
disableSelectedBtn.onmouseleave = function () {
|
|
1313
|
-
disableSelectedBtn.style.background = '#ef4444';
|
|
1314
|
-
};
|
|
1315
|
-
disableSelectedBtn.disabled = selectedIds.size === 0;
|
|
1316
|
-
disableSelectedBtn.onclick = async () => {
|
|
1317
|
-
if (!selectedIds.size) {
|
|
1318
|
-
return;
|
|
1319
|
-
}
|
|
1320
|
-
disableSelectedBtn.disabled = true;
|
|
1321
|
-
try {
|
|
1322
|
-
showBusyUi();
|
|
1323
|
-
await batchSetDeviceEnabled(selectedIds, false);
|
|
1324
|
-
toastSuccess('Selected devices disabled');
|
|
1325
|
-
await loadConfiguredDevices();
|
|
1326
|
-
await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
|
|
1327
|
-
}
|
|
1328
|
-
catch (e) {
|
|
1329
|
-
uiLog.error('Batch disable error:', e);
|
|
1330
|
-
toastError(e instanceof Error ? e.message : 'Failed to disable devices');
|
|
1331
|
-
}
|
|
1332
|
-
finally {
|
|
1333
|
-
hideBusyUi();
|
|
1334
|
-
disableSelectedBtn.disabled = false;
|
|
1335
|
-
}
|
|
1336
|
-
};
|
|
1337
|
-
buttonRow.appendChild(addSelectedBtn);
|
|
1338
|
-
buttonRow.appendChild(enableSelectedBtn);
|
|
1339
|
-
buttonRow.appendChild(disableSelectedBtn);
|
|
1340
|
-
batchControls.appendChild(buttonRow);
|
|
1341
|
-
const listContainer = document.getElementById('discoveredList');
|
|
1342
|
-
if (listContainer) {
|
|
1343
|
-
listContainer.insertBefore(batchControls, listContainer.firstChild);
|
|
1344
|
-
}
|
|
1345
|
-
}
|
|
1346
|
-
else {
|
|
1347
|
-
// Update button state if already present
|
|
1348
|
-
const addSelectedBtn = document.getElementById('addSelectedBtn');
|
|
1349
|
-
if (addSelectedBtn) {
|
|
1350
|
-
addSelectedBtn.disabled = selectedIds.size === 0;
|
|
1351
|
-
}
|
|
1352
|
-
}
|
|
1185
|
+
// Initial state update
|
|
1186
|
+
updateBatchButtonStates();
|
|
1353
1187
|
// Update status with count
|
|
1354
1188
|
const status = document.getElementById('discoverStatus');
|
|
1355
1189
|
if (status) {
|