@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.
Files changed (102) hide show
  1. package/.github/workflows/release.yml +63 -15
  2. package/.github/workflows/stale.yml +2 -4
  3. package/CHANGELOG.md +21 -29
  4. package/MIGRATION.md +6 -6
  5. package/README.md +5 -3
  6. package/dist/device-types.js +7 -7
  7. package/dist/device-types.js.map +1 -1
  8. package/dist/deviceFactory.d.ts +1 -1
  9. package/dist/deviceFactory.d.ts.map +1 -1
  10. package/dist/deviceFactory.js +20 -20
  11. package/dist/deviceFactory.js.map +1 -1
  12. package/dist/homebridge-ui/device-types.js +246 -0
  13. package/dist/homebridge-ui/device-types.js.map +1 -0
  14. package/dist/homebridge-ui/deviceCommandMapper.js +319 -0
  15. package/dist/homebridge-ui/deviceCommandMapper.js.map +1 -0
  16. package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -1
  17. package/dist/homebridge-ui/endpoints/devices.js +64 -1
  18. package/dist/homebridge-ui/endpoints/devices.js.map +1 -1
  19. package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -1
  20. package/dist/homebridge-ui/endpoints/discovery.js +5 -1
  21. package/dist/homebridge-ui/endpoints/discovery.js.map +1 -1
  22. package/dist/homebridge-ui/errors.js +32 -0
  23. package/dist/homebridge-ui/errors.js.map +1 -0
  24. package/dist/homebridge-ui/homebridge-ui/endpoints/config.js +90 -0
  25. package/dist/homebridge-ui/homebridge-ui/endpoints/config.js.map +1 -0
  26. package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js +144 -0
  27. package/dist/homebridge-ui/homebridge-ui/endpoints/devices.js.map +1 -0
  28. package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js +219 -0
  29. package/dist/homebridge-ui/homebridge-ui/endpoints/discovery.js.map +1 -0
  30. package/dist/homebridge-ui/homebridge-ui/server.js +11 -0
  31. package/dist/homebridge-ui/homebridge-ui/server.js.map +1 -0
  32. package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js +108 -0
  33. package/dist/homebridge-ui/homebridge-ui/utils/config-parser.js.map +1 -0
  34. package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js +111 -0
  35. package/dist/homebridge-ui/homebridge-ui/utils/device-migration.js.map +1 -0
  36. package/dist/homebridge-ui/homebridge-ui/utils/logger.js +17 -0
  37. package/dist/homebridge-ui/homebridge-ui/utils/logger.js.map +1 -0
  38. package/dist/homebridge-ui/public/js/api.d.ts.map +1 -1
  39. package/dist/homebridge-ui/public/js/api.js +24 -11
  40. package/dist/homebridge-ui/public/js/api.js.map +1 -1
  41. package/dist/homebridge-ui/public/js/api.ts +24 -12
  42. package/dist/homebridge-ui/public/js/app.js +117 -267
  43. package/dist/homebridge-ui/public/js/app.js.map +3 -3
  44. package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -1
  45. package/dist/homebridge-ui/public/js/devices.js +2 -0
  46. package/dist/homebridge-ui/public/js/devices.js.map +1 -1
  47. package/dist/homebridge-ui/public/js/devices.ts +2 -0
  48. package/dist/homebridge-ui/public/js/discovery.d.ts +5 -0
  49. package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -1
  50. package/dist/homebridge-ui/public/js/discovery.js +79 -245
  51. package/dist/homebridge-ui/public/js/discovery.js.map +1 -1
  52. package/dist/homebridge-ui/public/js/discovery.ts +88 -247
  53. package/dist/homebridge-ui/public/js/render.d.ts.map +1 -1
  54. package/dist/homebridge-ui/public/js/render.js +2 -1
  55. package/dist/homebridge-ui/public/js/render.js.map +1 -1
  56. package/dist/homebridge-ui/public/js/render.ts +2 -1
  57. package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -1
  58. package/dist/homebridge-ui/public/js/toast.js +15 -6
  59. package/dist/homebridge-ui/public/js/toast.js.map +1 -1
  60. package/dist/homebridge-ui/public/js/toast.ts +14 -7
  61. package/dist/homebridge-ui/settings.js +8 -0
  62. package/dist/homebridge-ui/settings.js.map +1 -0
  63. package/dist/homebridge-ui/switchbotClient.js +247 -0
  64. package/dist/homebridge-ui/switchbotClient.js.map +1 -0
  65. package/dist/homebridge-ui/utils/config-parser.d.ts +4 -0
  66. package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -1
  67. package/dist/homebridge-ui/utils/config-parser.js +21 -0
  68. package/dist/homebridge-ui/utils/config-parser.js.map +1 -1
  69. package/dist/switchbotClient.d.ts +7 -1
  70. package/dist/switchbotClient.d.ts.map +1 -1
  71. package/dist/switchbotClient.js +82 -10
  72. package/dist/switchbotClient.js.map +1 -1
  73. package/docs/assets/main.js +1 -1
  74. package/docs/index.html +10 -4
  75. package/docs/variables/default.html +1 -1
  76. package/eslint.config.js +9 -10
  77. package/package.json +26 -24
  78. package/src/device-types.js +246 -0
  79. package/src/device-types.js.map +1 -0
  80. package/src/device-types.ts +7 -7
  81. package/src/deviceCommandMapper.js +319 -0
  82. package/src/deviceCommandMapper.js.map +1 -0
  83. package/src/deviceFactory.ts +22 -21
  84. package/src/errors.js +32 -0
  85. package/src/errors.js.map +1 -0
  86. package/src/homebridge-ui/endpoints/devices.ts +66 -1
  87. package/src/homebridge-ui/endpoints/discovery.ts +5 -1
  88. package/src/homebridge-ui/public/js/api.ts +24 -12
  89. package/src/homebridge-ui/public/js/devices.ts +2 -0
  90. package/src/homebridge-ui/public/js/discovery.ts +88 -247
  91. package/src/homebridge-ui/public/js/render.ts +2 -1
  92. package/src/homebridge-ui/public/js/toast.ts +14 -7
  93. package/src/homebridge-ui/utils/config-parser.ts +17 -0
  94. package/src/settings.js +8 -0
  95. package/src/settings.js.map +1 -0
  96. package/src/switchbotClient.js +247 -0
  97. package/src/switchbotClient.js.map +1 -0
  98. package/src/switchbotClient.ts +95 -10
  99. package/test/client/switchbotClient.spec.ts +42 -1
  100. package/test/e2e/run-e2e.spec.ts +1 -0
  101. package/tsconfig.ui.json +11 -0
  102. 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,CAgF3I;AAED,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,IAAI,CAAC,CAUzD;AAED,wBAAsB,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC,CAG3D"}
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":"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"}
1
+ {"version":3,"file":"devices.js","sourceRoot":"","sources":["../../../../src/homebridge-ui/public/js/devices.ts"],"names":[],"mappings":"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"}
@@ -67,6 +67,8 @@ export async function addDeviceToConfig(device: any, options: { refresh?: boolea
67
67
  }
68
68
 
69
69
  if (refresh) {
70
+ // Force reload of config from disk after add
71
+ await syncParentPluginConfigFromDisk(true)
70
72
  await loadConfiguredDevices()
71
73
  }
72
74
 
@@ -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":"AAsRA,wBAAsB,2BAA2B,IAAI,OAAO,CAAC,IAAI,CAAC,CA8FjE;AAsDD,wBAAsB,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC,CA6qBrD;AA2XD,wBAAsB,iBAAiB,CAAC,MAAM,EAAE,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC,CAGlE"}
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 'connection';
290
+ return 'type'; // Default to Device Type grouping
329
291
  }
330
292
  catch (_e) {
331
- return 'connection';
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
- const resp = await homebridge.request('/platform-config', {});
454
- if (!resp || resp.success === false || !resp.data) {
455
- throw new Error('Failed to load config');
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 config = resp.data;
458
- // Homebridge config may be an array of platforms, find SwitchBot
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
- groupSelect.value = groupBy;
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
- // --- Batch Import Controls ---
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.querySelector('button')?.parentElement?.querySelector('button[aria-label="Enable Selected"]');
1171
- const disableSelectedBtn = document.querySelector('button')?.parentElement?.querySelector('button[aria-label="Disable Selected"]');
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
- // Add 'Add Selected to Config' button if not present, and align all red action buttons in a row
1186
- let batchControls = document.getElementById('batchImportControls');
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) {