@switchbot/homebridge-switchbot 5.0.0-beta.152 → 5.0.0-beta.154

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 (44) hide show
  1. package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -1
  2. package/dist/homebridge-ui/endpoints/devices.js +64 -1
  3. package/dist/homebridge-ui/endpoints/devices.js.map +1 -1
  4. package/dist/homebridge-ui/public/js/api.d.ts.map +1 -1
  5. package/dist/homebridge-ui/public/js/api.js +24 -11
  6. package/dist/homebridge-ui/public/js/api.js.map +1 -1
  7. package/dist/homebridge-ui/public/js/api.ts +24 -12
  8. package/dist/homebridge-ui/public/js/app.js +118 -263
  9. package/dist/homebridge-ui/public/js/app.js.map +3 -3
  10. package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -1
  11. package/dist/homebridge-ui/public/js/devices.js +2 -0
  12. package/dist/homebridge-ui/public/js/devices.js.map +1 -1
  13. package/dist/homebridge-ui/public/js/devices.ts +2 -0
  14. package/dist/homebridge-ui/public/js/discovery.d.ts +5 -0
  15. package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -1
  16. package/dist/homebridge-ui/public/js/discovery.js +80 -249
  17. package/dist/homebridge-ui/public/js/discovery.js.map +1 -1
  18. package/dist/homebridge-ui/public/js/discovery.ts +89 -251
  19. package/dist/homebridge-ui/public/js/modals.d.ts.map +1 -1
  20. package/dist/homebridge-ui/public/js/modals.js +4 -2
  21. package/dist/homebridge-ui/public/js/modals.js.map +1 -1
  22. package/dist/homebridge-ui/public/js/modals.ts +6 -2
  23. package/dist/homebridge-ui/public/js/render.d.ts.map +1 -1
  24. package/dist/homebridge-ui/public/js/render.js +2 -1
  25. package/dist/homebridge-ui/public/js/render.js.map +1 -1
  26. package/dist/homebridge-ui/public/js/render.ts +2 -1
  27. package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -1
  28. package/dist/homebridge-ui/public/js/toast.js +15 -6
  29. package/dist/homebridge-ui/public/js/toast.js.map +1 -1
  30. package/dist/homebridge-ui/public/js/toast.ts +14 -7
  31. package/dist/homebridge-ui/utils/config-parser.d.ts +4 -0
  32. package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -1
  33. package/dist/homebridge-ui/utils/config-parser.js +21 -0
  34. package/dist/homebridge-ui/utils/config-parser.js.map +1 -1
  35. package/docs/variables/default.html +1 -1
  36. package/package.json +4 -4
  37. package/src/homebridge-ui/endpoints/devices.ts +66 -1
  38. package/src/homebridge-ui/public/js/api.ts +24 -12
  39. package/src/homebridge-ui/public/js/devices.ts +2 -0
  40. package/src/homebridge-ui/public/js/discovery.ts +89 -251
  41. package/src/homebridge-ui/public/js/modals.ts +6 -2
  42. package/src/homebridge-ui/public/js/render.ts +2 -1
  43. package/src/homebridge-ui/public/js/toast.ts +14 -7
  44. package/src/homebridge-ui/utils/config-parser.ts +17 -0
@@ -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,CAgrBrD;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) {
@@ -401,10 +363,7 @@ export async function discoverDevices() {
401
363
  let phase = 'Preparing discovery...';
402
364
  let cancelled = false;
403
365
  // --- Real-time RSSI polling additions ---
404
- // Get BLE scan duration from settings or UI
405
- const bleScanDurationSeconds = scanSelect ? Number(scanSelect.value) : 5;
406
- const pollIntervalMs = 1000; // 1s polling
407
- const scanEndTime = Date.now() + bleScanDurationSeconds * 1000;
366
+ // (bleScanDurationSeconds is now only used in bleSettings below)
408
367
  const setPhase = (nextPhase) => {
409
368
  phase = nextPhase;
410
369
  phaseStartedAt = Date.now();
@@ -442,25 +401,23 @@ export async function discoverDevices() {
442
401
  const preferences = getDiscoveryPreferences();
443
402
  let groupBy = getDiscoveryGroupByPreference();
444
403
  let hideAdded = getDiscoveryHideAddedPreference();
445
- let controlsInitialized = false;
446
404
  // Use persistent selection state across renders
447
405
  if (!window._discoverySelectedIds) {
448
406
  window._discoverySelectedIds = new Set();
449
407
  }
450
408
  const selectedIds = window._discoverySelectedIds;
409
+ let controlsInitialized = false;
451
410
  // --- Real-time RSSI polling loop ---
452
411
  // (Moved inside main try block after bleSettings is defined)
453
412
  // Batch enable/disable helper (moved to module scope for UI access)
454
413
  async function batchSetDeviceEnabled(selectedIds, enabled) {
455
414
  // Fetch current config
456
- const resp = await homebridge.request('/platform-config', {});
457
- if (!resp || resp.success === false || !resp.data) {
458
- 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');
459
418
  }
460
- const config = resp.data;
461
- // Homebridge config may be an array of platforms, find SwitchBot
462
- const configArr = Array.isArray(config) ? config : [config];
463
- 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;
464
421
  if (platformIdx === -1) {
465
422
  throw new Error('SwitchBot platform config not found');
466
423
  }
@@ -491,14 +448,53 @@ export async function discoverDevices() {
491
448
  }
492
449
  }
493
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);
494
494
  if (controlsInitialized) {
495
495
  return;
496
496
  }
497
- // Always use persistent selectedIds
498
- if (!window._discoverySelectedIds) {
499
- window._discoverySelectedIds = new Set();
500
- }
501
- const selectedIds = window._discoverySelectedIds;
497
+ // Always use persistent selectedIds (already defined in outer scope)
502
498
  const controlsDiv = document.createElement('div');
503
499
  controlsDiv.style.cssText = 'margin-bottom: 12px; display: flex; gap: 12px; flex-wrap: wrap; align-items: center;';
504
500
  const filterLabel = document.createElement('label');
@@ -567,16 +563,28 @@ export async function discoverDevices() {
567
563
  setDiscoveryPreferences(preferences);
568
564
  void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
569
565
  };
570
- const groupLabel = document.createElement('label');
571
- groupLabel.style.fontSize = '12px';
572
- groupLabel.style.fontWeight = '500';
573
- groupLabel.style.marginLeft = '8px';
574
- groupLabel.textContent = 'Group:';
575
566
  const groupSelect = document.createElement('select');
576
567
  groupSelect.style.fontSize = '11px';
577
568
  groupSelect.style.padding = '4px 8px';
578
569
  groupSelect.style.borderRadius = '3px';
579
- 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'}`;
580
588
  const groupOptions = [
581
589
  { label: 'Connection', value: 'connection' },
582
590
  { label: 'Hub', value: 'hub' },
@@ -591,6 +599,7 @@ export async function discoverDevices() {
591
599
  groupSelect.onchange = () => {
592
600
  groupBy = groupSelect.value;
593
601
  setDiscoveryGroupByPreference(groupBy);
602
+ groupLabel.textContent = `Group: ${groupLabelTextMap[groupSelect.value] || 'Connection'}`;
594
603
  void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds);
595
604
  };
596
605
  const hideAddedLabel = document.createElement('label');
@@ -777,6 +786,7 @@ export async function discoverDevices() {
777
786
  topActionRow.appendChild(disableSelectedBtn);
778
787
  // Clear list and append controls in correct order
779
788
  list.innerHTML = '';
789
+ list.appendChild(selectControlsRow);
780
790
  list.appendChild(topActionRow);
781
791
  list.appendChild(controlsDiv);
782
792
  let deviceListContainer = document.getElementById('discoveredDevices');
@@ -1153,25 +1163,12 @@ async function updateDiscoveryView(allDevices, preferences, groupBy, hideAdded,
1153
1163
  toastError('Discovery UI error: device list container missing. Please reload the page.');
1154
1164
  }
1155
1165
  }
1156
- // --- Batch Import Controls ---
1157
- // Remove any duplicate 'Add Selected' buttons from legacy UI
1158
- // Robust NodeList iteration: classic for loop to avoid formatter and transpiler issues
1159
- const legacyAddSelectedBtns = [];
1160
- const allBtns = document.querySelectorAll('button');
1161
- for (let i = 0; i < allBtns.length; i++) {
1162
- const btn = allBtns[i];
1163
- if (btn.textContent && btn.textContent.trim() === 'Add Selected') {
1164
- legacyAddSelectedBtns.push(btn);
1165
- }
1166
- }
1167
- for (let i = 0; i < legacyAddSelectedBtns.length; i++) {
1168
- legacyAddSelectedBtns[i].remove();
1169
- }
1170
- // Listen for selection changes to update batch button states
1166
+ // Only update the enabled/disabled state of batch action buttons (created in ensureDiscoveryControls)
1171
1167
  function updateBatchButtonStates() {
1168
+ // These buttons are created in ensureDiscoveryControls and should have unique IDs
1172
1169
  const addSelectedBtn = document.getElementById('addSelectedBtn');
1173
- const enableSelectedBtn = document.querySelector('button')?.parentElement?.querySelector('button[aria-label="Enable Selected"]');
1174
- const disableSelectedBtn = document.querySelector('button')?.parentElement?.querySelector('button[aria-label="Disable Selected"]');
1170
+ const enableSelectedBtn = document.getElementById('enableSelectedBtn');
1171
+ const disableSelectedBtn = document.getElementById('disableSelectedBtn');
1175
1172
  const hasSelection = selectedIds.size > 0;
1176
1173
  if (addSelectedBtn) {
1177
1174
  addSelectedBtn.disabled = !hasSelection;
@@ -1185,174 +1182,8 @@ async function updateDiscoveryView(allDevices, preferences, groupBy, hideAdded,
1185
1182
  }
1186
1183
  window.removeEventListener('discovery-selection-changed', updateBatchButtonStates);
1187
1184
  window.addEventListener('discovery-selection-changed', updateBatchButtonStates);
1188
- // Add 'Add Selected to Config' button if not present, and align all red action buttons in a row
1189
- let batchControls = document.getElementById('batchImportControls');
1190
- if (!batchControls) {
1191
- batchControls = document.createElement('div');
1192
- batchControls.id = 'batchImportControls';
1193
- batchControls.style.display = 'flex';
1194
- batchControls.style.flexWrap = 'wrap';
1195
- batchControls.style.alignItems = 'center';
1196
- batchControls.style.gap = '18px';
1197
- batchControls.style.margin = '8px 0 18px 0';
1198
- batchControls.style.width = '100%';
1199
- // Button container for horizontal alignment
1200
- const buttonRow = document.createElement('div');
1201
- buttonRow.style.display = 'flex';
1202
- buttonRow.style.flexWrap = 'nowrap';
1203
- buttonRow.style.alignItems = 'center';
1204
- buttonRow.style.justifyContent = 'center';
1205
- buttonRow.style.gap = '16px';
1206
- buttonRow.style.width = '100%';
1207
- // Add Selected to Config button
1208
- const addSelectedBtn = document.createElement('button');
1209
- addSelectedBtn.id = 'addSelectedBtn';
1210
- addSelectedBtn.textContent = 'Add Selected to Config';
1211
- addSelectedBtn.disabled = selectedIds.size === 0;
1212
- addSelectedBtn.style.fontWeight = '600';
1213
- // Shared style for all red action buttons
1214
- const redButtonStyle = {
1215
- width: '220px',
1216
- padding: '12px 0',
1217
- fontSize: '17px',
1218
- marginBottom: '0',
1219
- boxShadow: '0 2px 8px 0 rgba(220,38,38,0.10)',
1220
- borderRadius: '8px',
1221
- };
1222
- Object.assign(addSelectedBtn.style, redButtonStyle);
1223
- addSelectedBtn.style.background = '#ef4444';
1224
- addSelectedBtn.style.color = 'white';
1225
- addSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
1226
- addSelectedBtn.onmouseenter = function () {
1227
- addSelectedBtn.style.background = '#dc2626';
1228
- };
1229
- addSelectedBtn.onmouseleave = function () {
1230
- addSelectedBtn.style.background = '#ef4444';
1231
- };
1232
- addSelectedBtn.onclick = async () => {
1233
- if (!selectedIds.size) {
1234
- return;
1235
- }
1236
- addSelectedBtn.disabled = true;
1237
- addSelectedBtn.textContent = 'Adding...';
1238
- try {
1239
- const selectedDevices = visibleDevices.filter(d => selectedIds.has(normalizeId(d.id)));
1240
- const bulkResult = await addDevicesInBulk(selectedDevices.map(d => ({
1241
- deviceId: d.id,
1242
- name: d.name,
1243
- type: d.type,
1244
- rssi: d.rssi,
1245
- address: d.address,
1246
- model: d.model,
1247
- })));
1248
- uiLog.info('Batch add response:', bulkResult);
1249
- if (!bulkResult || bulkResult.success === false) {
1250
- throw new Error(bulkResult?.data?.message || 'Batch add failed');
1251
- }
1252
- const addedCount = bulkResult?.addedCount ?? bulkResult?.data?.addedCount ?? 0;
1253
- const skippedCount = bulkResult?.skippedCount ?? bulkResult?.data?.skippedCount ?? 0;
1254
- toastSuccess(`Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`);
1255
- await loadConfiguredDevices();
1256
- selectedIds.clear();
1257
- addSelectedBtn.disabled = true;
1258
- addSelectedBtn.textContent = 'Add Selected to Config';
1259
- await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
1260
- }
1261
- catch (e) {
1262
- uiLog.error('Batch add error:', e);
1263
- toastError(e instanceof Error ? e.message : 'Failed to add devices');
1264
- addSelectedBtn.disabled = false;
1265
- addSelectedBtn.textContent = 'Add Selected to Config';
1266
- }
1267
- };
1268
- // Enable Selected button
1269
- const enableSelectedBtn = document.createElement('button');
1270
- enableSelectedBtn.textContent = 'Enable Selected';
1271
- enableSelectedBtn.style.fontWeight = '600';
1272
- Object.assign(enableSelectedBtn.style, redButtonStyle);
1273
- enableSelectedBtn.style.background = '#ef4444';
1274
- enableSelectedBtn.style.color = 'white';
1275
- enableSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
1276
- enableSelectedBtn.onmouseenter = function () {
1277
- enableSelectedBtn.style.background = '#dc2626';
1278
- };
1279
- enableSelectedBtn.onmouseleave = function () {
1280
- enableSelectedBtn.style.background = '#ef4444';
1281
- };
1282
- enableSelectedBtn.disabled = selectedIds.size === 0;
1283
- enableSelectedBtn.onclick = async () => {
1284
- if (!selectedIds.size) {
1285
- return;
1286
- }
1287
- enableSelectedBtn.disabled = true;
1288
- try {
1289
- showBusyUi();
1290
- await batchSetDeviceEnabled(selectedIds, true);
1291
- toastSuccess('Selected devices enabled');
1292
- await loadConfiguredDevices();
1293
- await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
1294
- }
1295
- catch (e) {
1296
- uiLog.error('Batch enable error:', e);
1297
- toastError(e instanceof Error ? e.message : 'Failed to enable devices');
1298
- }
1299
- finally {
1300
- hideBusyUi();
1301
- enableSelectedBtn.disabled = false;
1302
- }
1303
- };
1304
- // Disable Selected button
1305
- const disableSelectedBtn = document.createElement('button');
1306
- disableSelectedBtn.textContent = 'Disable Selected';
1307
- disableSelectedBtn.style.fontWeight = '600';
1308
- Object.assign(disableSelectedBtn.style, redButtonStyle);
1309
- disableSelectedBtn.style.background = '#ef4444';
1310
- disableSelectedBtn.style.color = 'white';
1311
- disableSelectedBtn.style.transition = 'background 0.2s, box-shadow 0.2s';
1312
- disableSelectedBtn.onmouseenter = function () {
1313
- disableSelectedBtn.style.background = '#dc2626';
1314
- };
1315
- disableSelectedBtn.onmouseleave = function () {
1316
- disableSelectedBtn.style.background = '#ef4444';
1317
- };
1318
- disableSelectedBtn.disabled = selectedIds.size === 0;
1319
- disableSelectedBtn.onclick = async () => {
1320
- if (!selectedIds.size) {
1321
- return;
1322
- }
1323
- disableSelectedBtn.disabled = true;
1324
- try {
1325
- showBusyUi();
1326
- await batchSetDeviceEnabled(selectedIds, false);
1327
- toastSuccess('Selected devices disabled');
1328
- await loadConfiguredDevices();
1329
- await updateDiscoveryView(allDevices, preferences, groupBy, hideAdded, selectedIds);
1330
- }
1331
- catch (e) {
1332
- uiLog.error('Batch disable error:', e);
1333
- toastError(e instanceof Error ? e.message : 'Failed to disable devices');
1334
- }
1335
- finally {
1336
- hideBusyUi();
1337
- disableSelectedBtn.disabled = false;
1338
- }
1339
- };
1340
- buttonRow.appendChild(addSelectedBtn);
1341
- buttonRow.appendChild(enableSelectedBtn);
1342
- buttonRow.appendChild(disableSelectedBtn);
1343
- batchControls.appendChild(buttonRow);
1344
- const listContainer = document.getElementById('discoveredList');
1345
- if (listContainer) {
1346
- listContainer.insertBefore(batchControls, listContainer.firstChild);
1347
- }
1348
- }
1349
- else {
1350
- // Update button state if already present
1351
- const addSelectedBtn = document.getElementById('addSelectedBtn');
1352
- if (addSelectedBtn) {
1353
- addSelectedBtn.disabled = selectedIds.size === 0;
1354
- }
1355
- }
1185
+ // Initial state update
1186
+ updateBatchButtonStates();
1356
1187
  // Update status with count
1357
1188
  const status = document.getElementById('discoverStatus');
1358
1189
  if (status) {